/* * nghttp2 - HTTP/2 C Library * * Copyright (c) 2015 Tatsuhiro Tsujikawa * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #ifndef TEMPLATE_H #define TEMPLATE_H #include "nghttp2_config.h" #include #include #include #include #include #include #include #include #include #include namespace nghttp2 { // std::forward is constexpr since C++14 template constexpr std::array< typename std::decay::type>::type, sizeof...(T)> make_array(T &&...t) { return std::array< typename std::decay::type>::type, sizeof...(T)>{{std::forward(t)...}}; } template constexpr size_t array_size(T (&)[N]) { return N; } template constexpr size_t str_size(T (&)[N]) { return N - 1; } // inspired by , but our // template can take functions returning other than void. template struct Defer { Defer(F &&f, T &&...t) : f(std::bind(std::forward(f), std::forward(t)...)) {} Defer(Defer &&o) noexcept : f(std::move(o.f)) {} ~Defer() { f(); } using ResultType = typename std::result_of::type( typename std::decay::type...)>::type; std::function f; }; template Defer defer(F &&f, T &&...t) { return Defer(std::forward(f), std::forward(t)...); } template bool test_flags(T t, F flags) { return (t & flags) == flags; } // doubly linked list of element T*. T must have field T *dlprev and // T *dlnext, which point to previous element and next element in the // list respectively. template struct DList { DList() : head(nullptr), tail(nullptr), len(0) {} DList(const DList &) = delete; DList &operator=(const DList &) = delete; DList(DList &&other) noexcept : head{std::exchange(other.head, nullptr)}, tail{std::exchange(other.tail, nullptr)}, len{std::exchange(other.len, 0)} {} DList &operator=(DList &&other) noexcept { if (this == &other) { return *this; } head = std::exchange(other.head, nullptr); tail = std::exchange(other.tail, nullptr); len = std::exchange(other.len, 0); return *this; } void append(T *t) { ++len; if (tail) { tail->dlnext = t; t->dlprev = tail; tail = t; return; } head = tail = t; } void remove(T *t) { --len; auto p = t->dlprev; auto n = t->dlnext; if (p) { p->dlnext = n; } if (head == t) { head = n; } if (n) { n->dlprev = p; } if (tail == t) { tail = p; } t->dlprev = t->dlnext = nullptr; } bool empty() const { return head == nullptr; } size_t size() const { return len; } T *head, *tail; size_t len; }; template void dlist_delete_all(DList &dl) { for (auto e = dl.head; e;) { auto next = e->dlnext; delete e; e = next; } } // User-defined literals for K, M, and G (powers of 1024) constexpr unsigned long long operator"" _k(unsigned long long k) { return k * 1024; } constexpr unsigned long long operator"" _m(unsigned long long m) { return m * 1024 * 1024; } constexpr unsigned long long operator"" _g(unsigned long long g) { return g * 1024 * 1024 * 1024; } // User-defined literals for time, converted into double in seconds // hours constexpr double operator"" _h(unsigned long long h) { return h * 60 * 60; } // minutes constexpr double operator"" _min(unsigned long long min) { return min * 60; } // seconds constexpr double operator"" _s(unsigned long long s) { return s; } // milliseconds constexpr double operator"" _ms(unsigned long long ms) { return ms / 1000.; } // Returns a copy of NULL-terminated string [first, last). template std::unique_ptr strcopy(InputIt first, InputIt last) { auto res = std::make_unique(last - first + 1); *std::copy(first, last, res.get()) = '\0'; return res; } // Returns a copy of NULL-terminated string |val|. inline std::unique_ptr strcopy(const char *val) { return strcopy(val, val + strlen(val)); } inline std::unique_ptr strcopy(const char *val, size_t n) { return strcopy(val, val + n); } // Returns a copy of val.c_str(). inline std::unique_ptr strcopy(const std::string &val) { return strcopy(std::begin(val), std::end(val)); } inline std::unique_ptr strcopy(const std::unique_ptr &val) { if (!val) { return nullptr; } return strcopy(val.get()); } inline std::unique_ptr strcopy(const std::unique_ptr &val, size_t n) { if (!val) { return nullptr; } return strcopy(val.get(), val.get() + n); } // ImmutableString represents string that is immutable unlike // std::string. It has c_str() and size() functions to mimic // std::string. It manages buffer by itself. Just like std::string, // c_str() returns NULL-terminated string, but NULL character may // appear before the final terminal NULL. class ImmutableString { public: using traits_type = std::char_traits; using value_type = traits_type::char_type; using allocator_type = std::allocator; using size_type = std::allocator_traits::size_type; using difference_type = std::allocator_traits::difference_type; using const_reference = const value_type &; using const_pointer = const value_type *; using const_iterator = const_pointer; using const_reverse_iterator = std::reverse_iterator; ImmutableString() : len(0), base("") {} ImmutableString(const char *s, size_t slen) : len(slen), base(copystr(s, s + len)) {} explicit ImmutableString(const char *s) : len(strlen(s)), base(copystr(s, s + len)) {} explicit ImmutableString(const std::string &s) : len(s.size()), base(copystr(std::begin(s), std::end(s))) {} template ImmutableString(InputIt first, InputIt last) : len(std::distance(first, last)), base(copystr(first, last)) {} ImmutableString(const ImmutableString &other) : len(other.len), base(copystr(std::begin(other), std::end(other))) {} ImmutableString(ImmutableString &&other) noexcept : len{std::exchange(other.len, 0)}, base{std::exchange(other.base, "")} {} ~ImmutableString() { if (len) { delete[] base; } } ImmutableString &operator=(const ImmutableString &other) { if (this == &other) { return *this; } if (len) { delete[] base; } len = other.len; base = copystr(std::begin(other), std::end(other)); return *this; } ImmutableString &operator=(ImmutableString &&other) noexcept { if (this == &other) { return *this; } if (len) { delete[] base; } len = std::exchange(other.len, 0); base = std::exchange(other.base, ""); return *this; } template static ImmutableString from_lit(const char (&s)[N]) { return ImmutableString(s, N - 1); } const_iterator begin() const { return base; }; const_iterator cbegin() const { return base; }; const_iterator end() const { return base + len; }; const_iterator cend() const { return base + len; }; const_reverse_iterator rbegin() const { return const_reverse_iterator{base + len}; } const_reverse_iterator crbegin() const { return const_reverse_iterator{base + len}; } const_reverse_iterator rend() const { return const_reverse_iterator{base}; } const_reverse_iterator crend() const { return const_reverse_iterator{base}; } const char *c_str() const { return base; } size_type size() const { return len; } bool empty() const { return len == 0; } const_reference operator[](size_type pos) const { return *(base + pos); } private: template const char *copystr(InputIt first, InputIt last) { if (first == last) { return ""; } auto res = new char[std::distance(first, last) + 1]; *std::copy(first, last, res) = '\0'; return res; } size_type len; const char *base; }; inline bool operator==(const ImmutableString &lhs, const ImmutableString &rhs) { return lhs.size() == rhs.size() && std::equal(std::begin(lhs), std::end(lhs), std::begin(rhs)); } inline bool operator==(const ImmutableString &lhs, const std::string &rhs) { return lhs.size() == rhs.size() && std::equal(std::begin(lhs), std::end(lhs), std::begin(rhs)); } inline bool operator==(const std::string &lhs, const ImmutableString &rhs) { return rhs == lhs; } inline bool operator==(const ImmutableString &lhs, const char *rhs) { return lhs.size() == strlen(rhs) && std::equal(std::begin(lhs), std::end(lhs), rhs); } inline bool operator==(const char *lhs, const ImmutableString &rhs) { return rhs == lhs; } inline bool operator!=(const ImmutableString &lhs, const ImmutableString &rhs) { return !(lhs == rhs); } inline bool operator!=(const ImmutableString &lhs, const std::string &rhs) { return !(lhs == rhs); } inline bool operator!=(const std::string &lhs, const ImmutableString &rhs) { return !(rhs == lhs); } inline bool operator!=(const ImmutableString &lhs, const char *rhs) { return !(lhs == rhs); } inline bool operator!=(const char *lhs, const ImmutableString &rhs) { return !(rhs == lhs); } inline std::ostream &operator<<(std::ostream &o, const ImmutableString &s) { return o.write(s.c_str(), s.size()); } inline std::string &operator+=(std::string &lhs, const ImmutableString &rhs) { lhs.append(rhs.c_str(), rhs.size()); return lhs; } // StringRef is a reference to a string owned by something else. So // it behaves like simple string, but it does not own pointer. When // it is default constructed, it has empty string. You can freely // copy or move around this struct, but never free its pointer. str() // function can be used to export the content as std::string. class StringRef { public: using traits_type = std::char_traits; using value_type = traits_type::char_type; using allocator_type = std::allocator; using size_type = std::allocator_traits::size_type; using difference_type = std::allocator_traits::difference_type; using const_reference = const value_type &; using const_pointer = const value_type *; using const_iterator = const_pointer; using const_reverse_iterator = std::reverse_iterator; constexpr StringRef() : base(""), len(0) {} explicit StringRef(const std::string &s) : base(s.c_str()), len(s.size()) {} explicit StringRef(const ImmutableString &s) : base(s.c_str()), len(s.size()) {} explicit StringRef(const char *s) : base(s), len(strlen(s)) {} constexpr StringRef(const char *s, size_t n) : base(s), len(n) {} template constexpr StringRef(const CharT *s, size_t n) : base(reinterpret_cast(s)), len(n) {} template StringRef(InputIt first, InputIt last) : base(reinterpret_cast(&*first)), len(std::distance(first, last)) {} template StringRef(InputIt *first, InputIt *last) : base(reinterpret_cast(first)), len(std::distance(first, last)) {} template constexpr static StringRef from_lit(const CharT (&s)[N]) { return StringRef{s, N - 1}; } static StringRef from_maybe_nullptr(const char *s) { if (s == nullptr) { return StringRef(); } return StringRef(s); } constexpr const_iterator begin() const { return base; }; constexpr const_iterator cbegin() const { return base; }; constexpr const_iterator end() const { return base + len; }; constexpr const_iterator cend() const { return base + len; }; const_reverse_iterator rbegin() const { return const_reverse_iterator{base + len}; } const_reverse_iterator crbegin() const { return const_reverse_iterator{base + len}; } const_reverse_iterator rend() const { return const_reverse_iterator{base}; } const_reverse_iterator crend() const { return const_reverse_iterator{base}; } constexpr const char *c_str() const { return base; } constexpr size_type size() const { return len; } constexpr bool empty() const { return len == 0; } constexpr const_reference operator[](size_type pos) const { return *(base + pos); } std::string str() const { return std::string(base, len); } const uint8_t *byte() const { return reinterpret_cast(base); } private: const char *base; size_type len; }; inline bool operator==(const StringRef &lhs, const StringRef &rhs) { return lhs.size() == rhs.size() && std::equal(std::begin(lhs), std::end(lhs), std::begin(rhs)); } inline bool operator==(const StringRef &lhs, const std::string &rhs) { return lhs.size() == rhs.size() && std::equal(std::begin(lhs), std::end(lhs), std::begin(rhs)); } inline bool operator==(const std::string &lhs, const StringRef &rhs) { return rhs == lhs; } inline bool operator==(const StringRef &lhs, const char *rhs) { return lhs.size() == strlen(rhs) && std::equal(std::begin(lhs), std::end(lhs), rhs); } inline bool operator==(const StringRef &lhs, const ImmutableString &rhs) { return lhs.size() == rhs.size() && std::equal(std::begin(lhs), std::end(lhs), std::begin(rhs)); } inline bool operator==(const ImmutableString &lhs, const StringRef &rhs) { return rhs == lhs; } inline bool operator==(const char *lhs, const StringRef &rhs) { return rhs == lhs; } inline bool operator!=(const StringRef &lhs, const StringRef &rhs) { return !(lhs == rhs); } inline bool operator!=(const StringRef &lhs, const std::string &rhs) { return !(lhs == rhs); } inline bool operator!=(const std::string &lhs, const StringRef &rhs) { return !(rhs == lhs); } inline bool operator!=(const StringRef &lhs, const char *rhs) { return !(lhs == rhs); } inline bool operator!=(const char *lhs, const StringRef &rhs) { return !(rhs == lhs); } inline bool operator<(const StringRef &lhs, const StringRef &rhs) { return std::lexicographical_compare(std::begin(lhs), std::end(lhs), std::begin(rhs), std::end(rhs)); } inline std::ostream &operator<<(std::ostream &o, const StringRef &s) { return o.write(s.c_str(), s.size()); } inline std::string &operator+=(std::string &lhs, const StringRef &rhs) { lhs.append(rhs.c_str(), rhs.size()); return lhs; } inline int run_app(std::function app, int argc, char **argv) { try { return app(argc, argv); } catch (const std::bad_alloc &) { fputs("Out of memory\n", stderr); } catch (const std::exception &x) { fprintf(stderr, "Caught %s:\n%s\n", typeid(x).name(), x.what()); } catch (...) { fputs("Unknown exception caught\n", stderr); } return EXIT_FAILURE; } } // namespace nghttp2 namespace std { template <> struct hash { std::size_t operator()(const nghttp2::StringRef &s) const noexcept { // 32 bit FNV-1a: // https://tools.ietf.org/html/draft-eastlake-fnv-16#section-6.1.1 uint32_t h = 2166136261u; for (auto c : s) { h ^= static_cast(c); h += (h << 1) + (h << 4) + (h << 7) + (h << 8) + (h << 24); } return h; } }; } // namespace std #endif // TEMPLATE_H