/* * 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 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 conexpr 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< typename std::decay::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)); } // 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 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) { fputs("Exception caught: ", stderr); fputs(x.what(), stderr); fputs("\n", stderr); } catch (...) { fputs("Unknown exception caught\n", stderr); } return EXIT_FAILURE; } } // namespace nghttp2 #endif // TEMPLATE_H