mdbx-test: rework/refine key-value generation.

Change-Id: I0da7c708cc18785f804112483bb86921fefdb8eb
This commit is contained in:
Leonid Yuriev 2020-10-25 10:51:46 +03:00
parent 3c55a27230
commit 1b21703c7b
3 changed files with 97 additions and 46 deletions

View File

@ -25,7 +25,7 @@ static inline MDBX_PURE_FUNCTION serial_t mask(unsigned bits) {
serial_t injective(const serial_t serial, serial_t injective(const serial_t serial,
const unsigned bits /* at least serial_minwith (8) */, const unsigned bits /* at least serial_minwith (8) */,
const serial_t salt) { const serial_t salt) {
assert(bits > serial_minwith && bits <= serial_maxwith); assert(bits >= serial_minwith && bits <= serial_maxwith);
/* LY: All these "magic" prime numbers were found /* LY: All these "magic" prime numbers were found
* and verified with a bit of brute force. */ * and verified with a bit of brute force. */
@ -124,7 +124,8 @@ void __hot maker::pair(serial_t serial, const buffer &key, buffer &value,
* Поэтому key_serial не трогаем, а в value_serial нелинейно вмешиваем * Поэтому key_serial не трогаем, а в value_serial нелинейно вмешиваем
* запрошенное количество бит из serial */ * запрошенное количество бит из serial */
value_serial += value_serial +=
(serial ^ (serial >> mapping.split)) & mask(mapping.split); (serial ^ (serial >> mapping.split) * UINT64_C(57035339200100753)) &
mask(mapping.split);
} }
value_serial |= value_age << mapping.split; value_serial |= value_age << mapping.split;
@ -207,9 +208,9 @@ void maker::setup(const config::actor_params_pod &actor, unsigned actor_id,
key_essentials.minlen = (uint16_t)actor.keylen_min; key_essentials.minlen = (uint16_t)actor.keylen_min;
assert(actor.keylen_max <= UINT32_MAX); assert(actor.keylen_max <= UINT32_MAX);
key_essentials.maxlen = key_essentials.maxlen =
std::min((uint32_t)actor.keylen_max, std::min(uint32_t(actor.keylen_max),
(uint32_t)mdbx_limits_keysize_max( uint32_t(mdbx_limits_keysize_max(
actor.pagesize, MDBX_db_flags_t(key_essentials.flags))); actor.pagesize, MDBX_db_flags_t(key_essentials.flags))));
value_essentials.flags = value_essentials.flags =
actor.table_flags & uint16_t(MDBX_INTEGERDUP | MDBX_REVERSEDUP); actor.table_flags & uint16_t(MDBX_INTEGERDUP | MDBX_REVERSEDUP);
@ -217,9 +218,9 @@ void maker::setup(const config::actor_params_pod &actor, unsigned actor_id,
value_essentials.minlen = (uint16_t)actor.datalen_min; value_essentials.minlen = (uint16_t)actor.datalen_min;
assert(actor.datalen_max <= UINT32_MAX); assert(actor.datalen_max <= UINT32_MAX);
value_essentials.maxlen = value_essentials.maxlen =
std::min((uint32_t)actor.datalen_max, std::min(uint32_t(actor.datalen_max),
(uint32_t)mdbx_limits_valsize_max( uint32_t(mdbx_limits_valsize_max(
actor.pagesize, MDBX_db_flags_t(key_essentials.flags))); actor.pagesize, MDBX_db_flags_t(key_essentials.flags))));
if (!actor.keygen.zero_fill) { if (!actor.keygen.zero_fill) {
key_essentials.flags |= essentials::prng_fill_flag; key_essentials.flags |= essentials::prng_fill_flag;
@ -234,13 +235,45 @@ void maker::setup(const config::actor_params_pod &actor, unsigned actor_id,
base = 0; base = 0;
} }
void maker::make_ordered() { void maker::make_linear() {
mapping.mesh = 0; mapping.mesh = (key_essentials.flags & MDBX_DUPSORT) ? 0 : mapping.split;
mapping.rotate = 0; mapping.rotate = 0;
mapping.offset = 0;
const auto max_serial = mask(mapping.width) + base;
const auto max_key_serial =
(mapping.split && (key_essentials.flags & MDBX_DUPSORT))
? max_serial >> mapping.split
: max_serial;
const auto max_value_serial =
(mapping.split && (key_essentials.flags & MDBX_DUPSORT))
? mask(mapping.split)
: 0;
while (key_essentials.minlen < 8 &&
(key_essentials.minlen == 0 ||
mask(key_essentials.minlen * 8) < max_key_serial)) {
key_essentials.minlen +=
(key_essentials.flags & (MDBX_INTEGERKEY | MDBX_INTEGERDUP)) ? 4 : 1;
if (key_essentials.maxlen < key_essentials.minlen)
key_essentials.maxlen = key_essentials.minlen;
}
if ((key_essentials.flags | value_essentials.flags) & MDBX_DUPSORT)
while (value_essentials.minlen < 8 &&
(value_essentials.minlen == 0 ||
mask(value_essentials.minlen * 8) < max_value_serial)) {
value_essentials.minlen +=
(value_essentials.flags & (MDBX_INTEGERKEY | MDBX_INTEGERDUP)) ? 4
: 1;
if (value_essentials.maxlen < value_essentials.minlen)
value_essentials.maxlen = value_essentials.minlen;
}
} }
bool maker::is_unordered() const { bool maker::is_unordered() const {
return (mapping.mesh >= serial_minwith || mapping.rotate) != 0; return mapping.rotate ||
mapping.mesh >
((key_essentials.flags & MDBX_DUPSORT) ? 0 : mapping.split);
} }
bool maker::increment(serial_t &serial, int delta) const { bool maker::increment(serial_t &serial, int delta) const {
@ -266,8 +299,9 @@ bool maker::increment(serial_t &serial, int delta) const {
//----------------------------------------------------------------------------- //-----------------------------------------------------------------------------
static size_t length(serial_t serial) { MDBX_NOTHROW_PURE_FUNCTION static inline unsigned length(serial_t serial) {
size_t n = 0; #if defined(__clang__) && __clang__ > 8
unsigned n = 0;
if (serial > UINT32_MAX) { if (serial > UINT32_MAX) {
n = 4; n = 4;
serial >>= 32; serial >>= 32;
@ -280,16 +314,26 @@ static size_t length(serial_t serial) {
n += 1; n += 1;
serial >>= 8; serial >>= 8;
} }
return (serial > 0) ? n + 1 : n; #else
unsigned n = (serial > UINT32_MAX) ? 4 : 0;
serial = (serial > UINT32_MAX) ? serial >> 32 : serial;
n += (serial > UINT16_MAX) ? 2 : 0;
serial = (serial > UINT16_MAX) ? serial >> 16 : serial;
n += (serial > UINT8_MAX);
serial = (serial > UINT8_MAX) ? serial >> 8 : serial;
#endif
return n + (serial > 0);
} }
buffer alloc(size_t limit) { buffer alloc(size_t limit) {
result *ptr = (result *)malloc(sizeof(result) + limit); result *ptr = (result *)malloc(sizeof(result) + limit + 8);
if (unlikely(ptr == nullptr)) if (unlikely(ptr == nullptr))
failure_perror("malloc(keyvalue_buffer)", errno); failure_perror("malloc(keyvalue_buffer)", errno);
ptr->value.iov_base = ptr->bytes; ptr->value.iov_base = ptr->bytes;
ptr->value.iov_len = 0; ptr->value.iov_len = 0;
ptr->limit = limit; ptr->limit = limit + 8;
return buffer(ptr); return buffer(ptr);
} }
@ -299,14 +343,20 @@ void __hot maker::mk_begin(const serial_t serial, const essentials &params,
assert(params.maxlen >= params.minlen); assert(params.maxlen >= params.minlen);
assert(params.maxlen >= length(serial)); assert(params.maxlen >= length(serial));
out.value.iov_len = out.value.iov_len = std::max(unsigned(params.minlen), length(serial));
(params.maxlen > params.minlen) const auto variation = params.maxlen - params.minlen;
? params.minlen + serial % (params.maxlen - params.minlen) if (variation) {
: params.minlen; if (serial % (variation + 1)) {
auto refix = serial * UINT64_C(48835288005252737);
refix ^= refix >> 32;
out.value.iov_len = std::max(
out.value.iov_len, params.minlen + 1 + size_t(refix) % variation);
}
}
if ((params.flags & (MDBX_INTEGERKEY | MDBX_INTEGERDUP)) == 0 && assert(length(serial) <= out.value.iov_len);
out.value.iov_len < 8) assert(out.value.iov_len >= params.minlen);
out.value.iov_len = std::max(length(serial), out.value.iov_len); assert(out.value.iov_len <= params.maxlen);
} }
void __hot maker::mk_continue(const serial_t serial, const essentials &params, void __hot maker::mk_continue(const serial_t serial, const essentials &params,
@ -322,36 +372,30 @@ void __hot maker::mk_continue(const serial_t serial, const essentials &params,
unsigned(MDBX_DUPSORT | MDBX_DUPFIXED | MDBX_INTEGERKEY | unsigned(MDBX_DUPSORT | MDBX_DUPFIXED | MDBX_INTEGERKEY |
MDBX_INTEGERDUP | MDBX_REVERSEKEY | MDBX_REVERSEDUP)) == 0); MDBX_INTEGERDUP | MDBX_REVERSEKEY | MDBX_REVERSEDUP)) == 0);
#endif #endif
assert(length(serial) <= out.value.iov_len);
out.value.iov_base = out.bytes; out.value.iov_base = out.bytes;
if (params.flags & (MDBX_INTEGERKEY | MDBX_INTEGERDUP)) { if (params.flags & (MDBX_INTEGERKEY | MDBX_INTEGERDUP)) {
assert(params.maxlen == params.minlen); assert(params.maxlen == params.minlen);
assert(params.minlen == 4 || params.minlen == 8); if (params.flags & (MDBX_INTEGERKEY | MDBX_INTEGERDUP))
if (is_byteorder_le() || params.minlen == 8) assert(params.minlen == 4 || params.minlen == 8);
out.u64 = serial; out.u64 = serial;
else if (!is_byteorder_le() && out.value.iov_len != 8)
out.u32 = (uint32_t)serial; out.u32 = uint32_t(serial);
} else if (params.flags & unsigned(MDBX_REVERSEKEY | MDBX_REVERSEDUP)) {
if (out.value.iov_len > 8) {
if (params.flags & essentials::prng_fill_flag) {
uint64_t state = serial ^ UINT64_C(0x41803711c9b75f19);
prng_fill(state, out.bytes, out.value.iov_len - 8);
} else
memset(out.bytes, '\0', out.value.iov_len - 8);
unaligned::store(out.bytes + out.value.iov_len - 8, htobe64(serial));
} else {
out.u64 = htobe64(serial);
if (out.value.iov_len < 8)
out.value.iov_base = out.bytes + 8 - out.value.iov_len;
}
} else { } else {
out.u64 = htole64(serial); const auto prefix =
if (out.value.iov_len > 8) { std::max(std::min(unsigned(params.minlen), 8u), length(serial));
out.u64 = htobe64(serial);
out.value.iov_base = out.bytes + 8 - prefix;
if (out.value.iov_len > prefix) {
if (params.flags & essentials::prng_fill_flag) { if (params.flags & essentials::prng_fill_flag) {
uint64_t state = serial ^ UINT64_C(0x923ab47b7ee6f6e4); uint64_t state = serial ^ UINT64_C(0x923ab47b7ee6f6e4);
prng_fill(state, out.bytes + 8, out.value.iov_len - 8); prng_fill(state, out.bytes + 8, out.value.iov_len - prefix);
} else } else
memset(out.bytes + 8, '\0', out.value.iov_len - 8); memset(out.bytes + 8, '\0', out.value.iov_len - prefix);
} }
if (unlikely(params.flags & (MDBX_REVERSEKEY | MDBX_REVERSEDUP)))
std::reverse((char *)out.value.iov_base,
(char *)out.value.iov_base + out.value.iov_len);
} }
assert(out.value.iov_len >= params.minlen); assert(out.value.iov_len >= params.minlen);

View File

@ -127,10 +127,16 @@ public:
serial_t value_age, const bool keylen_changeable); serial_t value_age, const bool keylen_changeable);
void setup(const config::actor_params_pod &actor, unsigned actor_id, void setup(const config::actor_params_pod &actor, unsigned actor_id,
unsigned thread_number); unsigned thread_number);
void make_ordered(); void make_linear();
bool is_unordered() const; bool is_unordered() const;
bool increment(serial_t &serial, int delta) const; bool increment(serial_t &serial, int delta) const;
bool increment_key_part(serial_t &serial, int delta,
bool reset_value_part = true) const {
if (reset_value_part)
serial &= ~((serial_t(1) << mapping.split) - 1);
return increment(serial, delta << mapping.split);
}
}; };
void log_pair(logging::loglevel level, const char *prefix, const buffer &key, void log_pair(logging::loglevel level, const char *prefix, const buffer &key,

View File

@ -632,6 +632,7 @@ bool test_execute(const actor_config &config_const) {
else else
log_verbose("test successfully (iteration %zi)", iter); log_verbose("test successfully (iteration %zi)", iter);
config.params.keygen.seed += INT32_C(0xA4F4D37B); config.params.keygen.seed += INT32_C(0xA4F4D37B);
log_verbose("turn keygen to %u", config.params.keygen.seed);
} }
} while (config.params.nrepeat == 0 || iter < config.params.nrepeat); } while (config.params.nrepeat == 0 || iter < config.params.nrepeat);