/*- * Copyright 2021 Vsevolod Stakhov * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef RSPAMD_UTIL_HXX #define RSPAMD_UTIL_HXX #pragma once #include #include #include #include #include /* * Common C++ utilities */ namespace rspamd { /* * Creates std::array from a standard C style array with automatic size calculation */ template constexpr auto array_of(Ts&&... t) -> std::array>, sizeof...(Ts)> { using T = typename std::decay_t>; return {{ std::forward(t)... }}; } template && std::is_constructible_v, bool> = false> constexpr auto find_map(const C &c, const K &k) -> std::optional> { auto f = c.find(k); if (f != c.end()) { return std::cref(f->second); } return std::nullopt; } template inline constexpr auto make_string_view_from_it(_It begin, _It end) { using result_type = std::string_view; return result_type{((begin != end) ? &*begin : nullptr), (typename result_type::size_type)std::max(std::distance(begin, end), (typename result_type::difference_type)0) }; } /** * Enumerate for range loop */ template ())), typename = decltype(std::end(std::declval()))> constexpr auto enumerate(T && iterable) { struct iterator { size_t i; TIter iter; bool operator != (const iterator & other) const { return iter != other.iter; } void operator ++ () { ++i; ++iter; } auto operator * () const { return std::tie(i, *iter); } }; struct iterable_wrapper { T iterable; auto begin() { return iterator{ 0, std::begin(iterable) }; } auto end() { return iterator{ 0, std::end(iterable) }; } }; return iterable_wrapper{ std::forward(iterable) }; } /** * Allocator that cleans up memory in a secure way on destruction * @tparam T */ template class secure_mem_allocator : public std::allocator { public: using value_type = typename std::allocator::value_type; using size_type = typename std::allocator::size_type; template struct rebind { typedef secure_mem_allocator other; }; secure_mem_allocator() noexcept = default; secure_mem_allocator(const secure_mem_allocator &_) noexcept : std::allocator(_) {} template explicit secure_mem_allocator(const secure_mem_allocator&) noexcept {} void deallocate(value_type *p, size_type num) noexcept { rspamd_explicit_memzero((void *)p, num); std::allocator::deallocate(p, num); } }; } #endif //RSPAMD_UTIL_HXX