8bf66a678a
git-subtree-dir: externals/robin-map git-subtree-split: 5cf53c6f5d81ba31a475f66ac4a61c6f54e476d3
327 lines
9.1 KiB
C++
327 lines
9.1 KiB
C++
/**
|
|
* MIT License
|
|
*
|
|
* Copyright (c) 2017 Thibaut Goetghebuer-Planchon <tessil@gmx.com>
|
|
*
|
|
* 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 TSL_UTILS_H
|
|
#define TSL_UTILS_H
|
|
|
|
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <ostream>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
#include <tsl/robin_hash.h>
|
|
|
|
#ifdef TSL_RH_NO_EXCEPTIONS
|
|
#define TSL_RH_CHECK_THROW(S, E)
|
|
#else
|
|
#define TSL_RH_CHECK_THROW(S, E) BOOST_CHECK_THROW(S, E)
|
|
#endif
|
|
|
|
template<typename T>
|
|
class identity_hash {
|
|
public:
|
|
std::size_t operator()(const T& value) const {
|
|
return static_cast<std::size_t>(value);
|
|
}
|
|
};
|
|
|
|
template<unsigned int MOD>
|
|
class mod_hash {
|
|
public:
|
|
template<typename T>
|
|
std::size_t operator()(const T& value) const {
|
|
return std::hash<T>()(value) % MOD;
|
|
}
|
|
};
|
|
|
|
class self_reference_member_test {
|
|
public:
|
|
self_reference_member_test() : m_value(std::to_string(-1)), m_value_ptr(&m_value) {
|
|
}
|
|
|
|
explicit self_reference_member_test(std::int64_t value) : m_value(std::to_string(value)), m_value_ptr(&m_value) {
|
|
}
|
|
|
|
self_reference_member_test(const self_reference_member_test& other) : m_value(*other.m_value_ptr), m_value_ptr(&m_value) {
|
|
}
|
|
|
|
self_reference_member_test(self_reference_member_test&& other) : m_value(*other.m_value_ptr), m_value_ptr(&m_value) {
|
|
}
|
|
|
|
self_reference_member_test& operator=(const self_reference_member_test& other) {
|
|
m_value = *other.m_value_ptr;
|
|
m_value_ptr = &m_value;
|
|
|
|
return *this;
|
|
}
|
|
|
|
self_reference_member_test& operator=(self_reference_member_test&& other) {
|
|
m_value = *other.m_value_ptr;
|
|
m_value_ptr = &m_value;
|
|
|
|
return *this;
|
|
}
|
|
|
|
std::string value() const {
|
|
return *m_value_ptr;
|
|
}
|
|
|
|
friend std::ostream& operator<<(std::ostream& stream, const self_reference_member_test& value) {
|
|
stream << *value.m_value_ptr;
|
|
|
|
return stream;
|
|
}
|
|
|
|
friend bool operator==(const self_reference_member_test& lhs, const self_reference_member_test& rhs) {
|
|
return *lhs.m_value_ptr == *rhs.m_value_ptr;
|
|
}
|
|
|
|
friend bool operator!=(const self_reference_member_test& lhs, const self_reference_member_test& rhs) {
|
|
return !(lhs == rhs);
|
|
}
|
|
|
|
friend bool operator<(const self_reference_member_test& lhs, const self_reference_member_test& rhs) {
|
|
return *lhs.m_value_ptr < *rhs.m_value_ptr;
|
|
}
|
|
private:
|
|
std::string m_value;
|
|
std::string* m_value_ptr;
|
|
};
|
|
|
|
|
|
class move_only_test {
|
|
public:
|
|
explicit move_only_test(std::int64_t value) : m_value(new std::string(std::to_string(value))) {
|
|
}
|
|
|
|
move_only_test(const move_only_test&) = delete;
|
|
move_only_test(move_only_test&&) = default;
|
|
move_only_test& operator=(const move_only_test&) = delete;
|
|
move_only_test& operator=(move_only_test&&) = default;
|
|
|
|
friend std::ostream& operator<<(std::ostream& stream, const move_only_test& value) {
|
|
if(value.m_value == nullptr) {
|
|
stream << "null";
|
|
}
|
|
else {
|
|
stream << *value.m_value;
|
|
}
|
|
|
|
return stream;
|
|
}
|
|
|
|
friend bool operator==(const move_only_test& lhs, const move_only_test& rhs) {
|
|
if(lhs.m_value == nullptr || rhs.m_value == nullptr) {
|
|
return lhs.m_value == nullptr && rhs.m_value == nullptr;
|
|
}
|
|
else {
|
|
return *lhs.m_value == *rhs.m_value;
|
|
}
|
|
}
|
|
|
|
friend bool operator!=(const move_only_test& lhs, const move_only_test& rhs) {
|
|
return !(lhs == rhs);
|
|
}
|
|
|
|
friend bool operator<(const move_only_test& lhs, const move_only_test& rhs) {
|
|
if(lhs.m_value == nullptr && rhs.m_value == nullptr) {
|
|
return false;
|
|
}
|
|
else if(lhs.m_value == nullptr) {
|
|
return true;
|
|
}
|
|
else if(rhs.m_value == nullptr) {
|
|
return false;
|
|
}
|
|
else {
|
|
return *lhs.m_value < *rhs.m_value;
|
|
}
|
|
}
|
|
|
|
const std::string& value() const {
|
|
return *m_value;
|
|
}
|
|
|
|
private:
|
|
std::unique_ptr<std::string> m_value;
|
|
};
|
|
|
|
|
|
class copy_only_test {
|
|
public:
|
|
explicit copy_only_test(std::int64_t value): m_value(std::to_string(value)) {
|
|
}
|
|
|
|
copy_only_test(const copy_only_test& other): m_value(other.m_value) {
|
|
}
|
|
|
|
copy_only_test& operator=(const copy_only_test& other) {
|
|
m_value = other.m_value;
|
|
|
|
return *this;
|
|
}
|
|
|
|
~copy_only_test() {
|
|
}
|
|
|
|
|
|
friend std::ostream& operator<<(std::ostream& stream, const copy_only_test& value) {
|
|
stream << value.m_value;
|
|
|
|
return stream;
|
|
}
|
|
|
|
friend bool operator==(const copy_only_test& lhs, const copy_only_test& rhs) {
|
|
return lhs.m_value == rhs.m_value;
|
|
}
|
|
|
|
friend bool operator!=(const copy_only_test& lhs, const copy_only_test& rhs) {
|
|
return !(lhs == rhs);
|
|
}
|
|
|
|
friend bool operator<(const copy_only_test& lhs, const copy_only_test& rhs) {
|
|
return lhs.m_value < rhs.m_value;
|
|
}
|
|
|
|
std::string value() const {
|
|
return m_value;
|
|
}
|
|
|
|
private:
|
|
std::string m_value;
|
|
};
|
|
|
|
|
|
|
|
namespace std {
|
|
template<>
|
|
struct hash<self_reference_member_test> {
|
|
std::size_t operator()(const self_reference_member_test& val) const {
|
|
return std::hash<std::string>()(val.value());
|
|
}
|
|
};
|
|
|
|
template<>
|
|
struct hash<move_only_test> {
|
|
std::size_t operator()(const move_only_test& val) const {
|
|
return std::hash<std::string>()(val.value());
|
|
}
|
|
};
|
|
|
|
template<>
|
|
struct hash<copy_only_test> {
|
|
std::size_t operator()(const copy_only_test& val) const {
|
|
return std::hash<std::string>()(val.value());
|
|
}
|
|
};
|
|
}
|
|
|
|
|
|
class utils {
|
|
public:
|
|
template<typename T>
|
|
static T get_key(std::size_t counter);
|
|
|
|
template<typename T>
|
|
static T get_value(std::size_t counter);
|
|
|
|
template<typename HMap>
|
|
static HMap get_filled_hash_map(std::size_t nb_elements);
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
inline std::int64_t utils::get_key<std::int64_t>(std::size_t counter) {
|
|
return tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter);
|
|
}
|
|
|
|
template<>
|
|
inline self_reference_member_test utils::get_key<self_reference_member_test>(std::size_t counter) {
|
|
return self_reference_member_test(tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter));
|
|
}
|
|
|
|
template<>
|
|
inline std::string utils::get_key<std::string>(std::size_t counter) {
|
|
return "Key " + std::to_string(counter);
|
|
}
|
|
|
|
template<>
|
|
inline move_only_test utils::get_key<move_only_test>(std::size_t counter) {
|
|
return move_only_test(tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter));
|
|
}
|
|
|
|
template<>
|
|
inline copy_only_test utils::get_key<copy_only_test>(std::size_t counter) {
|
|
return copy_only_test(tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter));
|
|
}
|
|
|
|
|
|
|
|
|
|
template<>
|
|
inline std::int64_t utils::get_value<std::int64_t>(std::size_t counter) {
|
|
return tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter*2);
|
|
}
|
|
|
|
template<>
|
|
inline self_reference_member_test utils::get_value<self_reference_member_test>(std::size_t counter) {
|
|
return self_reference_member_test(tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter*2));
|
|
}
|
|
|
|
template<>
|
|
inline std::string utils::get_value<std::string>(std::size_t counter) {
|
|
return "Value " + std::to_string(counter);
|
|
}
|
|
|
|
template<>
|
|
inline move_only_test utils::get_value<move_only_test>(std::size_t counter) {
|
|
return move_only_test(tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter*2));
|
|
}
|
|
|
|
template<>
|
|
inline copy_only_test utils::get_value<copy_only_test>(std::size_t counter) {
|
|
return copy_only_test(tsl::detail_robin_hash::numeric_cast<std::int64_t>(counter*2));
|
|
}
|
|
|
|
|
|
|
|
template<typename HMap>
|
|
inline HMap utils::get_filled_hash_map(std::size_t nb_elements) {
|
|
using key_t = typename HMap::key_type; using value_t = typename HMap:: mapped_type;
|
|
|
|
HMap map;
|
|
map.reserve(nb_elements);
|
|
|
|
for(std::size_t i = 0; i < nb_elements; i++) {
|
|
map.insert({utils::get_key<key_t>(i), utils::get_value<value_t>(i)});
|
|
}
|
|
|
|
return map;
|
|
}
|
|
|
|
#endif
|