/* Copyright (C) 1996-1997 Id Software, Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA See file, 'COPYING', for details. */ #pragma once #include #include #include // for memcpy() #include #include #include #include #include #include #include // for std::apply() int32_t Q_strncasecmp(const std::string_view &a, const std::string_view &b, size_t maxcount); int32_t Q_strcasecmp(const std::string_view &a, const std::string_view &b); bool string_iequals(const std::string_view &a, const std::string_view &b); // mxd struct case_insensitive_hash { std::size_t operator()(const std::string &s) const noexcept; }; struct case_insensitive_equal { bool operator()(const std::string &l, const std::string &r) const noexcept; }; struct case_insensitive_less { bool operator()(const std::string &l, const std::string &r) const noexcept; }; /** * standard C natural string compare * @param s1 left string * @param s2 right string * @return -1 when s1 < s2, 0 when s1 == s2, 1 when s1 > s2 */ int natstrcmp(const char *s1, const char *s2, bool case_sensitive = true); /** * STL natural less-than string compare * @param s1 left string * @param s2 right string * @return true when natural s1 < s2 */ bool natstrlt(const char *s1, const char *s2, bool case_sensitive = true); /** * @param s1 left string * @param s2 right string * std::string variant of natstrlt. * @return true when natural s1 < s2 */ bool stlnatstrlt(const std::string &s1, const std::string &s2, bool case_sensitive = true); struct natural_equal { bool operator()(const std::string &l, const std::string &r) const noexcept; }; struct natural_less { bool operator()(const std::string &l, const std::string &r) const noexcept; }; struct natural_case_insensitive_equal { bool operator()(const std::string &l, const std::string &r) const noexcept; }; struct natural_case_insensitive_less { bool operator()(const std::string &l, const std::string &r) const noexcept; }; std::string_view::const_iterator string_ifind(std::string_view haystack, std::string_view needle); bool string_icontains(std::string_view haystack, std::string_view needle); #include using qclock = std::chrono::high_resolution_clock; using duration = std::chrono::duration; using time_point = std::chrono::time_point; time_point I_FloatTime(); /* * ============================================================================ * BYTE ORDER FUNCTIONS * ============================================================================ */ #include // Binary streams; by default, streams use the native endianness // (unchanged bytes) but can be changed to a specific endianness // with the manipulator below. namespace detail { int32_t endian_i(); // 0 is the default for iwords enum class st_en : long { na = 0, le = 1, be = 2, }; bool need_swap(std::ios_base &os); template inline void write_swapped(std::ostream &s, const T &val) { const char *pVal = reinterpret_cast(&val); for (int32_t i = sizeof(T) - 1; i >= 0; i--) { s.write(&pVal[i], 1); } } template inline void read_swapped(std::istream &s, T &val) { char *pRetVal = reinterpret_cast(&val); for (int32_t i = sizeof(T) - 1; i >= 0; i--) { s.read(&pRetVal[i], 1); } } } // namespace detail template inline std::ios_base &endianness(std::ios_base &os) { os.iword(detail::endian_i()) = static_cast(e) + 1; return os; } // blank type used for paddings template struct padding { }; struct padding_n { size_t n; constexpr padding_n(size_t np) : n(np) { } }; // using <= for ostream and >= for istream template inline std::ostream &operator<=(std::ostream &s, const padding &) { for (size_t i = 0; i < n; i++) { s.put(0); } return s; } inline std::ostream &operator<=(std::ostream &s, const padding_n &p) { for (size_t i = 0; i < p.n; i++) { s.put(0); } return s; } inline std::ostream &operator<=(std::ostream &s, const char &c) { s.write(&c, sizeof(c)); return s; } inline std::ostream &operator<=(std::ostream &s, const int8_t &c) { s.write(reinterpret_cast(&c), sizeof(c)); return s; } inline std::ostream &operator<=(std::ostream &s, const uint8_t &c) { s.write(reinterpret_cast(&c), sizeof(c)); return s; } inline std::ostream &operator<=(std::ostream &s, const uint16_t &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } inline std::ostream &operator<=(std::ostream &s, const int16_t &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } inline std::ostream &operator<=(std::ostream &s, const uint32_t &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } inline std::ostream &operator<=(std::ostream &s, const int32_t &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } inline std::ostream &operator<=(std::ostream &s, const uint64_t &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } inline std::ostream &operator<=(std::ostream &s, const int64_t &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } inline std::ostream &operator<=(std::ostream &s, const float &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } inline std::ostream &operator<=(std::ostream &s, const double &c) { if (!detail::need_swap(s)) s.write(reinterpret_cast(&c), sizeof(c)); else detail::write_swapped(s, c); return s; } template inline std::ostream &operator<=(std::ostream &s, const std::array &c) { for (auto &v : c) s <= v; return s; } template inline std::ostream &operator<=(std::ostream &s, std::tuple tuple) { std::apply([&s](auto &&...args) { ((s <= args), ...); }, tuple); return s; } template inline std::enable_if_t, std::ostream &> operator<=( std::ostream &s, const T &obj) { // A big ugly, but, this skips us needing a const version of stream_data() s <= const_cast(obj).stream_data(); return s; } template inline std::enable_if_t, std::ostream &> operator<=( std::ostream &s, const T &obj) { obj.stream_write(s); return s; } template inline std::enable_if_t, std::ostream &> operator<=(std::ostream &s, const T &obj) { s <= reinterpret_cast &>(obj); return s; } template inline std::istream &operator>=(std::istream &s, padding &) { s.seekg(n, std::ios_base::cur); return s; } template inline std::istream &operator>=(std::istream &s, padding_n &p) { s.seekg(p.n, std::ios_base::cur); return s; } inline std::istream &operator>=(std::istream &s, char &c) { s.read(&c, sizeof(c)); return s; } inline std::istream &operator>=(std::istream &s, int8_t &c) { s.read(reinterpret_cast(&c), sizeof(c)); return s; } inline std::istream &operator>=(std::istream &s, uint8_t &c) { s.read(reinterpret_cast(&c), sizeof(c)); return s; } inline std::istream &operator>=(std::istream &s, uint16_t &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } inline std::istream &operator>=(std::istream &s, int16_t &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } inline std::istream &operator>=(std::istream &s, uint32_t &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } inline std::istream &operator>=(std::istream &s, int32_t &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } inline std::istream &operator>=(std::istream &s, uint64_t &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } inline std::istream &operator>=(std::istream &s, int64_t &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } inline std::istream &operator>=(std::istream &s, float &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } inline std::istream &operator>=(std::istream &s, double &c) { if (!detail::need_swap(s)) s.read(reinterpret_cast(&c), sizeof(c)); else detail::read_swapped(s, c); return s; } template inline std::istream &operator>=(std::istream &s, std::array &c) { for (auto &v : c) s >= v; return s; } template inline std::istream &operator>=(std::istream &s, std::tuple tuple) { std::apply([&s](auto &&...args) { ((s >= args), ...); }, tuple); return s; } template inline std::enable_if_t, std::istream &> operator>=( std::istream &s, T &obj) { s >= obj.stream_data(); return s; } template inline std::enable_if_t, std::istream &> operator>=( std::istream &s, T &obj) { obj.stream_read(s); return s; } template inline std::enable_if_t, std::istream &> operator>=(std::istream &s, T &obj) { s >= reinterpret_cast &>(obj); return s; } // Memory streams, because C++ doesn't supply these. struct membuf : std::streambuf { public: // construct membuf for reading and/or writing membuf(void *base, size_t size, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out); // construct membuf for reading membuf(const void *base, size_t size, std::ios_base::openmode which = std::ios_base::in); protected: void setpptrs(char *first, char *next, char *end); // seek operations pos_type seekpos(pos_type off, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out) override; pos_type seekoff(off_type off, std::ios_base::seekdir dir, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out) override; // put stuff std::streamsize xsputn(const char_type *s, std::streamsize n) override; int_type overflow(int_type ch) override; // get stuff std::streamsize xsgetn(char_type *s, std::streamsize n) override; int_type underflow() override; }; struct memstream : virtual membuf, std::ostream, std::istream { memstream(void *base, size_t size, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out | std::ios_base::binary); memstream(const void *base, size_t size, std::ios_base::openmode which = std::ios_base::in | std::ios_base::binary); }; struct omemstream : virtual membuf, std::ostream { omemstream(void *base, size_t size, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out | std::ios_base::binary); }; struct imemstream : virtual membuf, std::istream { imemstream( const void *base, size_t size, std::ios_base::openmode which = std::ios_base::in | std::ios_base::binary); }; // A very strange stream buffer that just stores the written size. // It can only write, not read. struct omemsizebuf : std::streambuf { public: // construct membuf for writing omemsizebuf(std::ios_base::openmode which = std::ios_base::out); protected: void setpptrs(char *first, char *next, char *end); // seek operations pos_type seekpos(pos_type off, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out) override; pos_type seekoff(off_type off, std::ios_base::seekdir dir, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out) override; // put stuff std::streamsize xsputn(const char_type *s, std::streamsize n) override; int_type overflow(int_type ch) override; }; struct omemsizestream : virtual omemsizebuf, std::ostream { omemsizestream(std::ios_base::openmode which = std::ios_base::out | std::ios_base::binary); }; void CRC_Init(uint16_t &crcvalue); void CRC_ProcessByte(uint16_t &crcvalue, uint8_t data); uint16_t CRC_Block(const uint8_t *start, int count); std::vector StringToVector(const std::string &str); template T deserialize(const std::vector &bytes) { auto stream = imemstream(bytes.data(), bytes.size()); stream >> endianness; T result; stream >= result; return result; }