mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-10 17:44:13 +08:00
b51d92d449
- add support for 'default' options values. - add min/max cases for option values. - add support for db-geometry params. - fix int-types for 32-bit builds (minor). - fix key/value generation for long-length cases. - fix update_flags for non-MDBX_DUPSORT. - 'none' for config-verbs. - check commandline length under Windows. - workaround for QueryFullProcessImageNameA() bug. - add setloglevel(). - workaroung for MSVC bug. - avoid extra 'jitter' testcase loops. - cleanup DUPSORT flags. - refine key/value min/max handling. - dump keygen params. - fix/refine keygen. - alter keygen defaults (rotate 3, offset 41). - default test-db size 4mb or 256mb. - fix/refine keygen for non-MDBX_DUPSORT. - seeding keygen with actor_id for better spreading.
575 lines
18 KiB
C++
575 lines
18 KiB
C++
/*
|
|
* Copyright 2017-2018 Leonid Yuriev <leo@yuriev.ru>
|
|
* 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
|
|
* <http://www.OpenLDAP.org/license.html>.
|
|
*/
|
|
|
|
#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<actor_params_pod>::value,
|
|
"actor_params_pod should by POD");
|
|
result.append(data2hex(static_cast<const actor_params_pod *>(¶ms),
|
|
sizeof(actor_params_pod), checksum));
|
|
result.append("|");
|
|
|
|
static_assert(std::is_pod<actor_config_pod>::value,
|
|
"actor_config_pod should by POD");
|
|
result.append(data2hex(static_cast<const actor_config_pod *>(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<actor_params_pod>::value,
|
|
"actor_params_pod should by POD");
|
|
if (!hex2data(str, slash, static_cast<actor_params_pod *>(&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<actor_config_pod>::value,
|
|
"actor_config_pod should by POD");
|
|
if (!hex2data(str, slash, static_cast<actor_config_pod *>(&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);
|
|
}
|