/**************************************************************************** ** ** Copyright (C) 2022 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtCore module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:COMM$ ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** $QT_END_LICENSE$ ** ** ** ** ** ** ** ** ** ** ** ** ** ** ** ** ** ** ** ******************************************************************************/ #ifndef QOFFSETSTRINGARRAY_P_H #define QOFFSETSTRINGARRAY_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include "private/qglobal_p.h" #include #include #include QT_BEGIN_NAMESPACE namespace QtPrivate { template struct OffsetSequenceHelper : OffsetSequenceHelper { }; template struct OffsetSequenceHelper<1, Last, I, S, Idx...> : IndexesList { // the unary + before std::numeric_limits below is required. Otherwise we get on g++-10.2: // error: comparison is always false due to limited range of data type [-Werror=type-limits] static const constexpr auto Length = Last + I; using Type = typename std::conditional< Last <= +std::numeric_limits::max(), quint8, typename std::conditional< Last <= std::numeric_limits::max(), quint16, int>::type >::type; }; template struct OffsetSequence : OffsetSequenceHelper { }; template struct StaticString { const char data[N]; }; template<> struct StaticString<0> { static constexpr int size() noexcept { return 0; } }; template struct StaticStringBuilder; template struct StaticStringBuilder, IndexesList> { QT_WARNING_PUSH QT_WARNING_DISABLE_MSVC(4100) // The formal parameter is not referenced in the body of the function. // The unreferenced parameter is ignored. // It happens when 'rs' is StaticString<0> template static constexpr StaticString concatenate( const char (&ls)[N1], const StaticString &rs) noexcept { return StaticString{{ls[I1]..., rs.data[I2]...}}; } QT_WARNING_POP }; template constexpr StaticString<0> staticString() noexcept { return StaticString<0>{}; } QT_WARNING_PUSH QT_WARNING_DISABLE_MSVC(4503) template constexpr StaticString staticString(const char (&s)[I], const char (&...sx)[Ix]) noexcept { return StaticStringBuilder< makeIndexSequence, makeIndexSequence>::concatenate(s, staticString(sx...)); } QT_WARNING_POP } // namespace QtPrivate QT_WARNING_PUSH #if defined(Q_CC_GNU_ONLY) && __GNUC__ == 9 QT_WARNING_DISABLE_GCC("-Wstringop-overflow") #endif template class QOffsetStringArray { public: using Type = T; template constexpr QOffsetStringArray(const QtPrivate::StaticString &str, QtPrivate::IndexesList) noexcept : m_string(str), m_offsets{Ox...} { } constexpr inline const char *operator[](const int index) const noexcept { return m_string.data + m_offsets[qBound(int(0), index, SizeOffsets - 1)]; } constexpr inline const char *at(const int index) const noexcept { return m_string.data + m_offsets[index]; } constexpr inline const char *str() const { return m_string.data; } constexpr inline const T *offsets() const { return m_offsets; } constexpr inline int count() const { return SizeOffsets; } static constexpr const auto sizeString = SizeString; static constexpr const auto sizeOffsets = SizeOffsets; private: QtPrivate::StaticString m_string; const T m_offsets[SizeOffsets]; }; QT_WARNING_POP template constexpr QOffsetStringArray qOffsetStringArray( const QtPrivate::StaticString &string, QtPrivate::IndexesList offsets) noexcept { return QOffsetStringArray( string, offsets); } template struct QOffsetStringArrayRet { using Offsets = QtPrivate::OffsetSequence; using Type = QOffsetStringArray; }; template constexpr auto qOffsetStringArray(const char (&...strings)[Nx]) noexcept -> typename QOffsetStringArrayRet::Type { using Offsets = QtPrivate::OffsetSequence; return qOffsetStringArray( QtPrivate::staticString(strings...), Offsets{}); } QT_END_NAMESPACE #endif // QOFFSETSTRINGARRAY_P_H