/* * Copyright 2017-2019 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 * . */ #include "test.h" #if defined(_MSC_VER) && !defined(strcasecmp) #define strcasecmp(str, len) _stricmp(str, len) #endif /* _MSC_VER && strcasecmp() */ namespace config { bool parse_option(int argc, char *const argv[], int &narg, const char *option, const char **value, const char *default_value) { assert(narg < argc); const char *current = argv[narg]; const size_t optlen = strlen(option); if (strncmp(current, "--", 2) || strncmp(current + 2, option, optlen)) return false; if (!value) { if (current[optlen + 2] == '=') failure("Option '--%s' doen't accept any value\n", option); return true; } *value = nullptr; if (current[optlen + 2] == '=') { *value = ¤t[optlen + 3]; return true; } if (narg + 1 < argc && strncmp("--", argv[narg + 1], 2) != 0) { *value = argv[narg + 1]; if (strcmp(*value, "default") == 0) { if (!default_value) failure("Option '--%s' doen't accept default value\n", option); *value = default_value; } ++narg; return true; } if (default_value) { *value = default_value; return true; } failure("No value given for '--%s' option\n", option); } bool parse_option(int argc, char *const argv[], int &narg, const char *option, std::string &value, bool allow_empty) { return parse_option(argc, argv, narg, option, value, allow_empty, allow_empty ? "" : nullptr); } bool parse_option(int argc, char *const argv[], int &narg, const char *option, std::string &value, bool allow_empty, const char *default_value) { const char *value_cstr; if (!parse_option(argc, argv, narg, option, &value_cstr, default_value)) return false; if (!allow_empty && strlen(value_cstr) == 0) failure("Value for option '--%s' could't be empty\n", option); value = value_cstr; return true; } bool parse_option(int argc, char *const argv[], int &narg, const char *option, unsigned &mask, const option_verb *verbs) { const char *list; if (!parse_option(argc, argv, narg, option, &list)) return false; unsigned clear = 0; while (*list) { if (*list == ',' || *list == ' ' || *list == '\t') { ++list; continue; } const char *const comma = strchr(list, ','); const bool strikethrough = *list == '-' || *list == '~'; if (strikethrough || *list == '+') ++list; else mask = clear; const size_t len = (comma) ? comma - list : strlen(list); const option_verb *scan = verbs; while (true) { if (!scan->verb) failure("Unknown verb '%.*s', for option '==%s'\n", (int)len, list, option); if (strlen(scan->verb) == len && strncmp(list, scan->verb, len) == 0) { mask = strikethrough ? mask & ~scan->mask : mask | scan->mask; clear = strikethrough ? clear & ~scan->mask : clear | scan->mask; list += len; break; } ++scan; } } return true; } bool parse_option(int argc, char *const argv[], int &narg, const char *option, uint64_t &value, const scale_mode scale, const uint64_t minval, const uint64_t maxval, const uint64_t default_value) { const char *value_cstr; if (!parse_option(argc, argv, narg, option, &value_cstr)) return false; if (default_value && strcmp(value_cstr, "default") == 0) { value = default_value; return true; } if (strcmp(value_cstr, "min") == 0 || strcmp(value_cstr, "minimal") == 0) { value = minval; return true; } if (strcmp(value_cstr, "max") == 0 || strcmp(value_cstr, "maximal") == 0) { value = maxval; return true; } char *suffix = nullptr; errno = 0; unsigned long long raw = strtoull(value_cstr, &suffix, 0); if ((suffix && *suffix) || errno) { suffix = nullptr; errno = 0; raw = strtoull(value_cstr, &suffix, 10); } if (errno) failure("Option '--%s' expects a numeric value (%s)\n", option, test_strerror(errno)); uint64_t multipler = 1; if (suffix && *suffix) { if (scale == no_scale) failure("Option '--%s' doen't accepts suffixes, so '%s' is unexpected\n", option, suffix); if (strcmp(suffix, "K") == 0 || strcasecmp(suffix, "Kilo") == 0) multipler = (scale == decimal) ? UINT64_C(1000) : UINT64_C(1024); else if (strcmp(suffix, "M") == 0 || strcasecmp(suffix, "Mega") == 0) multipler = (scale == decimal) ? UINT64_C(1000) * 1000 : UINT64_C(1024) * 1024; else if (strcmp(suffix, "G") == 0 || strcasecmp(suffix, "Giga") == 0) multipler = (scale == decimal) ? UINT64_C(1000) * 1000 * 1000 : UINT64_C(1024) * 1024 * 1024; else if (strcmp(suffix, "T") == 0 || strcasecmp(suffix, "Tera") == 0) multipler = (scale == decimal) ? UINT64_C(1000) * 1000 * 1000 * 1000 : UINT64_C(1024) * 1024 * 1024 * 1024; else if (scale == duration && (strcmp(suffix, "s") == 0 || strcasecmp(suffix, "Seconds") == 0)) multipler = 1; else if (scale == duration && (strcmp(suffix, "m") == 0 || strcasecmp(suffix, "Minutes") == 0)) multipler = 60; else if (scale == duration && (strcmp(suffix, "h") == 0 || strcasecmp(suffix, "Hours") == 0)) multipler = 3600; else if (scale == duration && (strcmp(suffix, "d") == 0 || strcasecmp(suffix, "Days") == 0)) multipler = 3600 * 24; else failure( "Option '--%s' expects a numeric value with Kilo/Mega/Giga/Tera %s" "suffixes, but '%s' is unexpected\n", option, (scale == duration) ? "or Seconds/Minutes/Hours/Days " : "", suffix); } if (raw >= UINT64_MAX / multipler) failure("The value for option '--%s' is too huge\n", option); value = raw * multipler; if (maxval && value > maxval) failure("The maximal value for option '--%s' is %" PRIu64 "\n", option, maxval); if (value < minval) failure("The minimal value for option '--%s' is %" PRIu64 "\n", option, minval); return true; } bool parse_option(int argc, char *const argv[], int &narg, const char *option, unsigned &value, const scale_mode scale, const unsigned minval, const unsigned maxval, const unsigned default_value) { uint64_t huge; if (!parse_option(argc, argv, narg, option, huge, scale, minval, maxval, default_value)) return false; value = (unsigned)huge; return true; } bool parse_option(int argc, char *const argv[], int &narg, const char *option, uint8_t &value, const uint8_t minval, const uint8_t maxval, const uint8_t default_value) { uint64_t huge; if (!parse_option(argc, argv, narg, option, huge, no_scale, minval, maxval, default_value)) return false; value = (uint8_t)huge; return true; } bool parse_option(int argc, char *const argv[], int &narg, const char *option, int64_t &value, const int64_t minval, const int64_t maxval, const int64_t default_value) { uint64_t proxy = (uint64_t)value; if (parse_option(argc, argv, narg, option, proxy, config::binary, (uint64_t)minval, (uint64_t)maxval, (uint64_t)default_value)) { value = (int64_t)proxy; return true; } return false; } bool parse_option(int argc, char *const argv[], int &narg, const char *option, int32_t &value, const int32_t minval, const int32_t maxval, const int32_t default_value) { uint64_t proxy = (uint64_t)value; if (parse_option(argc, argv, narg, option, proxy, config::binary, (uint64_t)minval, (uint64_t)maxval, (uint64_t)default_value)) { value = (int32_t)proxy; return true; } return false; } bool parse_option(int argc, char *const argv[], int &narg, const char *option, bool &value) { const char *value_cstr = nullptr; if (!parse_option(argc, argv, narg, option, &value_cstr, "yes")) { const char *current = argv[narg]; if (strncmp(current, "--no-", 5) == 0 && strcmp(current + 5, option) == 0) { value = false; return true; } if (strncmp(current, "--dont-", 7) == 0 && strcmp(current + 7, option) == 0) { value = false; return true; } return false; } if (!value_cstr) { value = true; return true; } if (strcasecmp(value_cstr, "yes") == 0 || strcasecmp(value_cstr, "1") == 0) { value = true; return true; } if (strcasecmp(value_cstr, "no") == 0 || strcasecmp(value_cstr, "0") == 0) { value = false; return true; } failure( "Option '--%s' expects a 'boolean' value Yes/No, so '%s' is unexpected\n", option, value_cstr); } //----------------------------------------------------------------------------- const struct option_verb mode_bits[] = { {"rdonly", MDBX_RDONLY}, {"mapasync", MDBX_MAPASYNC}, {"utterly", MDBX_UTTERLY_NOSYNC}, {"nosubdir", MDBX_NOSUBDIR}, {"nosync", MDBX_NOSYNC}, {"nometasync", MDBX_NOMETASYNC}, {"writemap", MDBX_WRITEMAP}, {"notls", MDBX_NOTLS}, {"nordahead", MDBX_NORDAHEAD}, {"nomeminit", MDBX_NOMEMINIT}, {"coalesce", MDBX_COALESCE}, {"lifo", MDBX_LIFORECLAIM}, {"perturb", MDBX_PAGEPERTURB}, {nullptr, 0}}; const struct option_verb table_bits[] = { {"key.reverse", MDBX_REVERSEKEY}, {"key.integer", MDBX_INTEGERKEY}, {"data.integer", MDBX_INTEGERDUP | MDBX_DUPFIXED | MDBX_DUPSORT}, {"data.fixed", MDBX_DUPFIXED | MDBX_DUPSORT}, {"data.reverse", MDBX_REVERSEDUP | MDBX_DUPSORT}, {"data.dups", MDBX_DUPSORT}, {nullptr, 0}}; static void dump_verbs(const char *caption, size_t bits, const struct option_verb *verbs) { log_info("%s: 0x%" PRIx64 " = ", caption, (uint64_t)bits); const char *comma = ""; while (verbs->mask && bits) { if ((bits & verbs->mask) == verbs->mask) { logging::feed("%s%s", comma, verbs->verb); bits -= verbs->mask; comma = ", "; } ++verbs; } logging::feed("%s\n", (*comma == '\0') ? "none" : ""); } static void dump_duration(const char *caption, unsigned duration) { log_info("%s: ", caption); if (duration) { if (duration > 24 * 3600) logging::feed("%u_", duration / (24 * 3600)); if (duration > 3600) logging::feed("%02u:", (duration % (24 * 3600)) / 3600); logging::feed("%02u:%02u", (duration % 3600) / 60, duration % 60); } else { logging::feed("INFINITE"); } logging::feed("\n"); } void dump(const char *title) { logging::local_suffix indent(title); for (auto i = global::actors.begin(); i != global::actors.end(); ++i) { log_info("#%u, testcase %s, space_id/table %u\n", i->actor_id, testcase2str(i->testcase), i->space_id); indent.push(); if (i->params.loglevel) { log_info("log: level %u, %s\n", i->params.loglevel, i->params.pathname_log.empty() ? "console" : i->params.pathname_log.c_str()); } log_info("database: %s, size %" PRIuPTR "[%" PRIiPTR "..%" PRIiPTR ", %i %i, %i]\n", i->params.pathname_db.c_str(), i->params.size_now, i->params.size_lower, i->params.size_upper, i->params.shrink_threshold, i->params.growth_step, i->params.pagesize); dump_verbs("mode", i->params.mode_flags, mode_bits); dump_verbs("table", i->params.table_flags, table_bits); if (i->params.test_nops) log_info("iterations/records %u\n", i->params.test_nops); else dump_duration("duration", i->params.test_duration); if (i->params.nrepeat) log_info("repeat %u\n", i->params.nrepeat); else log_info("repeat ETERNALLY\n"); log_info("threads %u\n", i->params.nthreads); log_info( "keygen.params: case %s, width %u, mesh %u, rotate %u, offset %" PRIu64 ", split %u/%u\n", keygencase2str(i->params.keygen.keycase), i->params.keygen.width, i->params.keygen.mesh, i->params.keygen.rotate, i->params.keygen.offset, i->params.keygen.split, i->params.keygen.width - i->params.keygen.split); log_info("keygen.seed: %u\n", i->params.keygen.seed); log_info("key: minlen %u, maxlen %u\n", i->params.keylen_min, i->params.keylen_max); log_info("data: minlen %u, maxlen %u\n", i->params.datalen_min, i->params.datalen_max); log_info("batch: read %u, write %u\n", i->params.batch_read, i->params.batch_write); if (i->params.waitfor_nops) log_info("wait: actor %u for %u ops\n", i->wait4id, i->params.waitfor_nops); else if (i->params.delaystart) dump_duration("delay", i->params.delaystart); else log_info("no-delay\n"); if (i->params.inject_writefaultn) log_info("inject-writefault on %u ops\n", i->params.inject_writefaultn); else log_info("no-inject-writefault\n"); log_info("limits: readers %u, tables %u\n", i->params.max_readers, i->params.max_tables); log_info("drop table: %s\n", i->params.drop_table ? "Yes" : "No"); indent.pop(); } dump_duration("timeout", global::config::timeout_duration_seconds); log_info("cleanup: before %s, after %s\n", global::config::cleanup_before ? "Yes" : "No", global::config::cleanup_after ? "Yes" : "No"); log_info("failfast: %s\n", global::config::failfast ? "Yes" : "No"); log_info("progress indicator: %s\n", global::config::progress_indicator ? "Yes" : "No"); } } /* namespace config */ //----------------------------------------------------------------------------- using namespace config; actor_config::actor_config(actor_testcase testcase, const actor_params ¶ms, unsigned space_id, unsigned wait4id) : params(params) { this->space_id = space_id; this->actor_id = 1 + (unsigned)global::actors.size(); this->testcase = testcase; this->wait4id = wait4id; signal_nops = 0; } const std::string actor_config::serialize(const char *prefix) const { simple_checksum checksum; std::string result; if (prefix) result.append(prefix); checksum.push(params.pathname_db); result.append(params.pathname_db); result.append("|"); checksum.push(params.pathname_log); result.append(params.pathname_log); result.append("|"); static_assert(std::is_pod::value, "actor_params_pod should by POD"); result.append(data2hex(static_cast(¶ms), sizeof(actor_params_pod), checksum)); result.append("|"); static_assert(std::is_pod::value, "actor_config_pod should by POD"); result.append(data2hex(static_cast(this), sizeof(actor_config_pod), checksum)); result.append("|"); result.append(osal_serialize(checksum)); result.append("|"); result.append(std::to_string(checksum.value)); return result; } bool actor_config::deserialize(const char *str, actor_config &config) { simple_checksum checksum; TRACE(">> actor_config::deserialize: %s\n", str); const char *slash = strchr(str, '|'); if (!slash) { TRACE("<< actor_config::deserialize: slash-1\n"); return false; } config.params.pathname_db.assign(str, slash - str); checksum.push(config.params.pathname_db); str = slash + 1; slash = strchr(str, '|'); if (!slash) { TRACE("<< actor_config::deserialize: slash-2\n"); return false; } config.params.pathname_log.assign(str, slash - str); checksum.push(config.params.pathname_log); str = slash + 1; slash = strchr(str, '|'); if (!slash) { TRACE("<< actor_config::deserialize: slash-3\n"); return false; } static_assert(std::is_pod::value, "actor_params_pod should by POD"); if (!hex2data(str, slash, static_cast(&config.params), sizeof(actor_params_pod), checksum)) { TRACE("<< actor_config::deserialize: actor_params_pod(%.*s)\n", (int)(slash - str), str); return false; } str = slash + 1; slash = strchr(str, '|'); if (!slash) { TRACE("<< actor_config::deserialize: slash-4\n"); return false; } static_assert(std::is_pod::value, "actor_config_pod should by POD"); if (!hex2data(str, slash, static_cast(&config), sizeof(actor_config_pod), checksum)) { TRACE("<< actor_config::deserialize: actor_config_pod(%.*s)\n", (int)(slash - str), str); return false; } str = slash + 1; slash = strchr(str, '|'); if (!slash) { TRACE("<< actor_config::deserialize: slash-5\n"); return false; } if (!config.osal_deserialize(str, slash, checksum)) { TRACE("<< actor_config::deserialize: osal\n"); return false; } str = slash + 1; uint64_t verify = std::stoull(std::string(str)); if (checksum.value != verify) { TRACE("<< actor_config::deserialize: checksum mismatch\n"); return false; } TRACE("<< actor_config::deserialize: OK\n"); return true; } unsigned actor_params::mdbx_keylen_min() const { return (table_flags & MDBX_INTEGERKEY) ? 4 : 0; } unsigned actor_params::mdbx_keylen_max() const { return (table_flags & MDBX_INTEGERKEY) ? 8 : std::min((unsigned)mdbx_get_maxkeysize(pagesize), (unsigned)UINT16_MAX); } unsigned actor_params::mdbx_datalen_min() const { return (table_flags & MDBX_INTEGERDUP) ? 4 : 0; } unsigned actor_params::mdbx_datalen_max() const { return (table_flags & MDBX_INTEGERDUP) ? 8 : std::min((table_flags & MDBX_DUPSORT) ? (unsigned)mdbx_get_maxkeysize(pagesize) : (unsigned)MDBX_MAXDATASIZE, (unsigned)UINT16_MAX); }