QOffsetStringArray: fix compilation error with MSVC15/17

MSVC 2015/2017 has a bug with constexpr.
It is internal compiler error when we are using QtPrivate::IndexSequence
with more than 1024 items.

Example:

template<int ... I>
void foo(QtPrivate::IndexSequence<I...>) { }

void bar() {
    foo(QtPrivate::makeIndexSequence<1024>()); // works
    foo(QtPrivate::makeIndexSequence<1025>()); // error: C1202: recursive type or function dependency context too complex
}

Workaround:

template<typename>
struct Foo;

template<int ... I>
struct Foo<QtPrivate::IndexSequence<I...>>{
    static constexpr void call() { }
};

void bar() {
    Foo<QtPrivate::makeIndexSequence<1025>>::call(); // works
}

Change-Id: Iedbf5291ac4a9f51f8b5f6b76b83baba920c092d
Reviewed-by: Thiago Macieira <thiago.macieira@intel.com>
This commit is contained in:
Mikhail Svetkin 2018-08-30 11:49:44 +02:00
parent f4d9dadabb
commit e8dacba47a

View File

@ -84,47 +84,34 @@ struct StaticString
{ {
const char data[N]; const char data[N];
constexpr StaticString(const char (&literal)[N]) noexcept static constexpr int size() noexcept
: StaticString(literal, makeIndexSequence<N> {})
{ }
template<int ... Idx>
constexpr StaticString(const char (&literal)[N],
IndexesList<Idx...>) noexcept
: data{literal[Idx]...}
{ }
template<typename ... T>
constexpr StaticString(const T ... c) noexcept : data{c...} { }
constexpr char operator[](int i) const noexcept
{
return data[i];
}
template<int N2>
constexpr StaticString<N + N2> operator+(const StaticString<N2> &rs) const noexcept
{
return concatenate(rs, makeIndexSequence<N>{}, makeIndexSequence<N2>{});
}
template<int N2, int ... I1, int ... I2>
constexpr StaticString<N + N2> concatenate(const StaticString<N2> &rs,
IndexesList<I1...>,
IndexesList<I2...>) const noexcept
{
return StaticString<N + N2>(data[I1]..., rs[I2]...);
}
constexpr int size() const noexcept
{ {
return N; return N;
} }
}; };
template<> template<>
struct StaticString<0> struct StaticString<0>
{ {
static constexpr int size() noexcept
{
return 0;
}
};
template<typename, typename>
struct StaticStringBuilder;
template<int ... I1, int ... I2>
struct StaticStringBuilder<IndexesList<I1...>, IndexesList<I2...>>
{
template<int N1, typename T2>
static constexpr StaticString<N1 + T2::size()> concatenate(
const char (&ls)[N1], const T2 &rs) noexcept
{
return {ls[I1]..., rs.data[I2]...};
}
}; };
template<int Sum> template<int Sum>
@ -136,7 +123,9 @@ constexpr StaticString<0> staticString() noexcept
template<int Sum, int I, int ... Ix> template<int Sum, int I, int ... Ix>
constexpr StaticString<Sum> staticString(const char (&s)[I], const char (&...sx)[Ix]) noexcept constexpr StaticString<Sum> staticString(const char (&s)[I], const char (&...sx)[Ix]) noexcept
{ {
return StaticString<I>(s) + staticString<Sum - I>(sx...); return StaticStringBuilder<
makeIndexSequence<I>,
makeIndexSequence<Sum - I>>::concatenate(s, staticString<Sum - I>(sx...));
} }
} // namespace QtPrivate } // namespace QtPrivate
@ -146,25 +135,24 @@ class QOffsetStringArray
public: public:
using Type = T; using Type = T;
template<int ... Cx, int ... Ox> template<int ... Ox>
constexpr QOffsetStringArray(const QtPrivate::StaticString<SizeString> &str, constexpr QOffsetStringArray(const QtPrivate::StaticString<SizeString> &str,
QtPrivate::IndexesList<Cx...>,
QtPrivate::IndexesList<SizeString, Ox...>) noexcept QtPrivate::IndexesList<SizeString, Ox...>) noexcept
: m_string{str[Cx]...}, : m_string{str},
m_offsets{Ox...} m_offsets{Ox...}
{ } { }
constexpr inline const char *operator[](const int index) const noexcept constexpr inline const char *operator[](const int index) const noexcept
{ {
return m_string + m_offsets[qBound(int(0), index, SizeOffsets - 1)]; return m_string.data + m_offsets[qBound(int(0), index, SizeOffsets - 1)];
} }
constexpr inline const char *at(const int index) const noexcept constexpr inline const char *at(const int index) const noexcept
{ {
return m_string + m_offsets[index]; return m_string.data + m_offsets[index];
} }
constexpr inline const char *str() const { return m_string; } constexpr inline const char *str() const { return m_string.data; }
constexpr inline const T *offsets() const { return m_offsets; } constexpr inline const T *offsets() const { return m_offsets; }
constexpr inline int count() const { return SizeOffsets; }; constexpr inline int count() const { return SizeOffsets; };
@ -172,7 +160,7 @@ public:
static constexpr const auto sizeOffsets = SizeOffsets; static constexpr const auto sizeOffsets = SizeOffsets;
private: private:
const char m_string[SizeString]; QtPrivate::StaticString<SizeString> m_string;
const T m_offsets[SizeOffsets]; const T m_offsets[SizeOffsets];
}; };
@ -183,7 +171,6 @@ constexpr QOffsetStringArray<T, N, sizeof ... (Ox)> qOffsetStringArray(
{ {
return QOffsetStringArray<T, N, sizeof ... (Ox)>( return QOffsetStringArray<T, N, sizeof ... (Ox)>(
string, string,
QtPrivate::makeIndexSequence<N> {},
offsets); offsets);
} }
@ -202,7 +189,6 @@ constexpr auto qOffsetStringArray(const char (&...strings)[Nx]) noexcept -> type
QtPrivate::staticString<Offsets::Length>(strings...), Offsets{}); QtPrivate::staticString<Offsets::Length>(strings...), Offsets{});
} }
QT_END_NAMESPACE QT_END_NAMESPACE
#endif // QOFFSETSTRINGARRAY_P_H #endif // QOFFSETSTRINGARRAY_P_H