/* * Copyright 2017 Leonid Yuriev * and other libmdbx authors: please see AUTHORS file. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted only as authorized by the OpenLDAP * Public License. * * A copy of this license is available in the file LICENSE in the * top-level directory of the distribution or, alternatively, at * . */ #pragma once #include "base.h" #if !defined(__BYTE_ORDER__) || !defined(__ORDER_LITTLE_ENDIAN__) || \ !defined(__ORDER_BIG_ENDIAN__) #ifndef _MSC_VER #include /* for endianness */ #endif #if defined(__BYTE_ORDER) && defined(__LITTLE_ENDIAN) && defined(__BIG_ENDIAN) #define __ORDER_LITTLE_ENDIAN__ __LITTLE_ENDIAN #define __ORDER_BIG_ENDIAN__ __BIG_ENDIAN #define __BYTE_ORDER__ __BYTE_ORDER #else #define __ORDER_LITTLE_ENDIAN__ 1234 #define __ORDER_BIG_ENDIAN__ 4321 #if defined(__LITTLE_ENDIAN__) || defined(_LITTLE_ENDIAN) || \ defined(__ARMEL__) || defined(__THUMBEL__) || defined(__AARCH64EL__) || \ defined(__MIPSEL__) || defined(_MIPSEL) || defined(__MIPSEL) || \ defined(__i386) || defined(__x86_64__) || defined(_M_IX86) || \ defined(_M_X64) || defined(i386) || defined(_X86_) || defined(__i386__) || \ defined(_X86_64_) || defined(_M_ARM) || defined(_M_ARM64) || \ defined(__e2k__) #define __BYTE_ORDER__ __ORDER_LITTLE_ENDIAN__ #elif defined(__BIG_ENDIAN__) || defined(_BIG_ENDIAN) || defined(__ARMEB__) || \ defined(__THUMBEB__) || defined(__AARCH64EB__) || defined(__MIPSEB__) || \ defined(_MIPSEB) || defined(__MIPSEB) || defined(_M_IA64) #define __BYTE_ORDER__ __ORDER_BIG_ENDIAN__ #else #error __BYTE_ORDER__ should be defined. #endif #endif #endif #if __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__ && \ __BYTE_ORDER__ != __ORDER_BIG_ENDIAN__ #error Unsupported byte order. #endif #if __GNUC_PREREQ(4, 4) || defined(__clang__) #if __GNUC_PREREQ(4, 5) || defined(__clang__) #define unreachable() __builtin_unreachable() #endif #define bswap64(v) __builtin_bswap64(v) #define bswap32(v) __builtin_bswap32(v) #if __GNUC_PREREQ(4, 8) || __has_builtin(__builtin_bswap16) #define bswap16(v) __builtin_bswap16(v) #endif #elif defined(_MSC_VER) #if _MSC_FULL_VER < 190024215 #pragma message( \ "It is recommended to use Visual Studio 2015 (MSC 19.0) or newer.") #endif #define unreachable() __assume(0) #define bswap64(v) _byteswap_uint64(v) #define bswap32(v) _byteswap_ulong(v) #define bswap16(v) _byteswap_ushort(v) #define rot64(v, s) _rotr64(v, s) #define rot32(v, s) _rotr(v, s) #if defined(_M_ARM64) || defined(_M_X64) || defined(_M_IA64) #pragma intrinsic(_umul128) #define mul_64x64_128(a, b, ph) _umul128(a, b, ph) #pragma intrinsic(__umulh) #define mul_64x64_high(a, b) __umulh(a, b) #endif #if defined(_M_IX86) #pragma intrinsic(__emulu) #define mul_32x32_64(a, b) __emulu(a, b) #elif defined(_M_ARM) #define mul_32x32_64(a, b) _arm_umull(a, b) #endif #endif /* compiler */ #ifndef unreachable #define unreachable() \ do { \ } while (1) #endif #ifndef bswap64 #ifdef __bswap_64 #define bswap64(v) __bswap_64(v) #else static __inline uint64_t bswap64(uint64_t v) { return v << 56 | v >> 56 | ((v << 40) & 0x00ff000000000000ull) | ((v << 24) & 0x0000ff0000000000ull) | ((v << 8) & 0x000000ff00000000ull) | ((v >> 8) & 0x00000000ff000000ull) | ((v >> 24) & 0x0000000000ff0000ull) | ((v >> 40) & 0x000000000000ff00ull); } #endif #endif /* bswap64 */ #ifndef bswap32 #ifdef __bswap_32 #define bswap32(v) __bswap_32(v) #else static __inline uint32_t bswap32(uint32_t v) { return v << 24 | v >> 24 | ((v << 8) & 0x00ff0000) | ((v >> 8) & 0x0000ff00); } #endif #endif /* bswap32 */ #ifndef bswap16 #ifdef __bswap_16 #define bswap16(v) __bswap_16(v) #else static __inline uint16_t bswap16(uint16_t v) { return v << 8 | v >> 8; } #endif #endif /* bswap16 */ #define is_byteorder_le() (__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__) #define is_byteorder_be() (__BYTE_ORDER__ == __ORDER_BIG_ENDIAN__) #ifndef htole16 #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ #define htobe16(v) bswap16(v) #define htole16(v) (v) #define be16toh(v) bswap16(v) #define le16toh(v) (v) #else #define htobe16(v) (v) #define htole16(v) bswap16(v) #define be16toh(v) (v) #define le16toh(v) bswap16(v) #endif #endif /* htole16 */ #ifndef htole32 #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ #define htobe32(v) bswap32(v) #define htole32(v) (v) #define be32toh(v) bswap32(v) #define le32toh(v) (v) #else #define htobe32(v) (v) #define htole32(v) bswap32(v) #define be32toh(v) (v) #define le32toh(v) bswap32(v) #endif #endif /* htole32 */ #ifndef htole64 #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ #define htobe64(v) bswap64(v) #define htole64(v) (v) #define be64toh(v) bswap64(v) #define le64toh(v) (v) #else #define htobe64(v) (v) #define htole64(v) bswap_64(v) #define be64toh(v) (v) #define le64toh(v) bswap_64(v) #endif #endif /* htole64 */ namespace unaligned { template static __inline T load(const void *ptr) { #if defined(_MSC_VER) && \ (defined(_M_ARM64) || defined(_M_X64) || defined(_M_IA64)) return *(const T __unaligned *)ptr; #elif UNALIGNED_OK return *(const T *)ptr; #else T local; #if defined(__GNUC__) || defined(__clang__) __builtin_memcpy(&local, (const T *)ptr, sizeof(T)); #else memcpy(&local, (const T *)ptr, sizeof(T)); #endif /* __GNUC__ || __clang__ */ return local; #endif /* UNALIGNED_OK */ } template static __inline void store(void *ptr, const T &value) { #if defined(_MSC_VER) && \ (defined(_M_ARM64) || defined(_M_X64) || defined(_M_IA64)) *((T __unaligned *)ptr) = value; #elif UNALIGNED_OK *(volatile T *)ptr = value; #else #if defined(__GNUC__) || defined(__clang__) __builtin_memcpy(ptr, &value, sizeof(T)); #else memcpy(ptr, &value, sizeof(T)); #endif /* __GNUC__ || __clang__ */ #endif /* UNALIGNED_OK */ } } /* namespace unaligned */ //----------------------------------------------------------------------------- #ifndef rot64 static __inline uint64_t rot64(uint64_t v, unsigned s) { return (v >> s) | (v << (64 - s)); } #endif /* rot64 */ #ifndef mul_32x32_64 static __inline uint64_t mul_32x32_64(uint32_t a, uint32_t b) { return a * (uint64_t)b; } #endif /* mul_32x32_64 */ #ifndef mul_64x64_128 static __inline unsigned add_with_carry(uint64_t *sum, uint64_t addend) { *sum += addend; return *sum < addend; } static __inline uint64_t mul_64x64_128(uint64_t a, uint64_t b, uint64_t *h) { #if defined(__SIZEOF_INT128__) || \ (defined(_INTEGRAL_MAX_BITS) && _INTEGRAL_MAX_BITS >= 128) __uint128_t r = (__uint128_t)a * (__uint128_t)b; /* modern GCC could nicely optimize this */ *h = r >> 64; return r; #elif defined(mul_64x64_high) *h = mul_64x64_high(a, b); return a * b; #else /* performs 64x64 to 128 bit multiplication */ uint64_t ll = mul_32x32_64((uint32_t)a, (uint32_t)b); uint64_t lh = mul_32x32_64(a >> 32, (uint32_t)b); uint64_t hl = mul_32x32_64((uint32_t)a, b >> 32); *h = mul_32x32_64(a >> 32, b >> 32) + (lh >> 32) + (hl >> 32) + add_with_carry(&ll, lh << 32) + add_with_carry(&ll, hl << 32); return ll; #endif } #endif /* mul_64x64_128() */ #ifndef mul_64x64_high static __inline uint64_t mul_64x64_high(uint64_t a, uint64_t b) { uint64_t h; mul_64x64_128(a, b, &h); return h; } #endif /* mul_64x64_high */ static __inline bool is_power2(size_t x) { return (x & (x - 1)) == 0; } static __inline size_t roundup2(size_t value, size_t granularity) { assert(is_power2(granularity)); return (value + granularity - 1) & ~(granularity - 1); } //----------------------------------------------------------------------------- struct simple_checksum { uint64_t value; simple_checksum() : value(0) {} void push(uint32_t data) { value += data * UINT64_C(9386433910765580089) + 1; value ^= value >> 41; } void push(uint64_t data) { push((uint32_t)data); push((uint32_t)(data >> 32)); } void push(bool data) { push(data ? UINT32_C(0x780E) : UINT32_C(0xFA18E)); } void push(const void *ptr, size_t bytes) { const uint8_t *data = (const uint8_t *)ptr; for (size_t i = 0; i < bytes; ++i) push((uint32_t)data[i]); } void push(const double &data) { push(&data, sizeof(double)); } void push(const char *cstr) { push(cstr, strlen(cstr)); } void push(const std::string &str) { push(str.data(), str.size()); } #if defined(_WIN32) || defined(_WIN64) || defined(_WINDOWS) void push(const HANDLE &handle) { push(&handle, sizeof(handle)); } #endif /* _WINDOWS */ }; std::string data2hex(const void *ptr, size_t bytes, simple_checksum &checksum); bool hex2data(const char *hex_begin, const char *hex_end, void *ptr, size_t bytes, simple_checksum &checksum); std::string format(const char *fmt, ...);