// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #pragma once #define HAVE_IRCD_UTIL_H namespace ircd { /// Utilities for IRCd. /// /// This is an inline namespace: everything declared in it will be /// accessible in ircd::. By first opening it here as inline all /// subsequent openings of this namespace do not have to use the inline /// keyword but will still be inlined to ircd::. inline namespace util {} } // // Fundamental macros // #define IRCD_EXPCAT(a, b) a ## b #define IRCD_CONCAT(a, b) IRCD_EXPCAT(a, b) #define IRCD_UNIQUE(a) IRCD_CONCAT(a, __COUNTER__) #include "typography.h" #include "identity.h" #include "unit_literal.h" #include "construction.h" #include "unwind.h" #include "reentrance.h" #include "enum.h" #include "custom_ptr.h" #include "va_rtti.h" #include "unique_iterator.h" #include "scope_count.h" #include "scope_restore.h" #include "instance_list.h" #include "instance_multimap.h" #include "instance_map.h" #include "callbacks.h" #include "bswap.h" #include "tuple.h" #include "timer.h" #include "life_guard.h" #include "pubsetbuf.h" #include "string.h" #include "pointers.h" #include "iterator.h" #include "nothrow.h" #include "what.h" #include "u2a.h" #include "pretty.h" #include "hash.h" #include "closure.h" #include "env.h" #include "test.h" #include "boolean.h" #include "maybe.h" #include "all.h" #include "compare_exchange.h" #include "bitset.h" // Unsorted section namespace ircd { inline namespace util { // // Misc size() participants. // size_t size(std::ostream &s); template constexpr size_t size(const char (&buf)[SIZE]) noexcept { return SIZE; } template constexpr size_t size(const std::array &buf) noexcept { return SIZE; } template constexpr size_t size(const std::array &buf) noexcept { return SIZE; } template constexpr typename std::enable_if::value, size_t>::type size(const T &val) noexcept { return sizeof(T); } // // Misc data() participants // template constexpr const char * data(const char (&buf)[SIZE]) noexcept { return buf; } template constexpr char * data(char (&buf)[SIZE]) noexcept { return buf; } template constexpr typename std::enable_if(), const uint8_t *>::type data(const T &val) noexcept { return reinterpret_cast(&val); } template constexpr typename std::enable_if(), uint8_t *>::type data(T &val) noexcept { return reinterpret_cast(&val); } // // Misc bang participants // inline auto operator!(const std::string &str) { return str.empty(); } inline auto operator!(const std::string_view &str) { return str.empty(); } /// Like std::next() but with out_of_range exception /// template typename std::enable_if() || is_input_iterator(), It>::type at(It &&start, It &&stop, ssize_t i) { for(; start != stop; --i, std::advance(start, 1)) if(!i) return std::move(start); throw std::out_of_range { "at(a, b, i): 'i' out of range" }; } /// Like std::find() -> std::distance(), but with out_of_range exception /// template typename std::enable_if() || is_input_iterator(), size_t>::type index(const It &start, const It &stop, value &&val) { const auto it { std::find(start, stop, std::forward(val)) }; if(likely(it != stop)) return std::distance(start, it); throw std::out_of_range { "index(a, b, val): 'val' not contained in set" }; } // // Some functors for STL // template struct keys { auto &operator()(typename container::reference v) const { return v.first; } }; template struct values { auto &operator()(typename container::reference v) const { return v.second; } }; /// Iterator based until() matching std::for_each except the function /// returns a bool to continue rather than void. /// template bool until(it_a a, const it_b &b, boolean_function&& f) { for(; a != b; ++a) if(!f(*a)) return false; return true; } template T minmax(T ret, const T &min, const T &max) { ret = std::max(ret, min); ret = std::min(ret, max); return ret; } } // namespace util } // namespace ircd