/* * 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 namespace nghttp2 { template typename std::enable_if::value, std::unique_ptr>::type make_unique(U &&... u) { return std::unique_ptr(new T(std::forward(u)...)); } template typename std::enable_if::value, std::unique_ptr>::type make_unique(size_t size) { return std::unique_ptr(new typename std::remove_extent::type[size]()); } // 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) : 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) {} DList(const DList &) = delete; DList &operator=(const DList &) = delete; DList(DList &&other) : head(other.head), tail(other.tail) { other.head = other.tail = nullptr; } DList &operator=(DList &&other) { if (this == &other) { return *this; } head = other.head; tail = other.tail; other.head = other.tail = nullptr; return *this; } void append(T *t) { if (tail) { tail->dlnext = t; t->dlprev = tail; tail = t; return; } head = tail = t; } void remove(T *t) { 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; } T *head, *tail; }; 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 constexpr double operator"" _h(unsigned long long h) { return h * 60 * 60; } constexpr double operator"" _min(unsigned long long min) { return min * 60; } // Returns a copy of NULL-terminated string [first, last). template std::unique_ptr strcopy(InputIt first, InputIt last) { auto res = 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); } // VString represents 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. struct VString { VString() : len(0) {} VString(const char *s, size_t slen) : len(slen), base(strcopy(s, len)) {} VString(const char *s) : len(strlen(s)), base(strcopy(s, len)) {} VString(const std::string &s) : len(s.size()), base(strcopy(s)) {} template VString(InputIt first, InputIt last) : len(std::distance(first, last)), base(strcopy(first, last)) {} VString(const VString &other) : len(other.len), base(strcopy(other.base, other.len)) {} VString(VString &&) = default; VString &operator=(const VString &other) { if (this == &other) { return *this; } len = other.len; base = strcopy(other.base, other.len); return *this; } VString &operator=(VString &&other) = default; const char *c_str() const { return base.get(); } size_t size() const { return len; } size_t len; std::unique_ptr base; }; // StringAdaptor 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. struct StringAdaptor { StringAdaptor() : base(""), len(0) {} template StringAdaptor(const T &s) : base(s.c_str()), len(s.size()) {} StringAdaptor(const char *s) : base(s), len(strlen(s)) {} const char *c_str() const { return base; } size_t size() const { return len; } std::string str() const { return std::string(base, len); } const char *base; size_t len; }; 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 #endif // TEMPLATE_H