mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-23 01:48:21 +08:00
mdbx-tools: поддержка не-печатных имен subDb в mdbx_chk
.
This commit is contained in:
parent
7011743262
commit
44493c6448
239
src/mdbx_chk.c
239
src/mdbx_chk.c
@ -23,6 +23,8 @@
|
|||||||
#define xMDBX_TOOLS /* Avoid using internal eASSERT() */
|
#define xMDBX_TOOLS /* Avoid using internal eASSERT() */
|
||||||
#include "internals.h"
|
#include "internals.h"
|
||||||
|
|
||||||
|
#include <ctype.h>
|
||||||
|
|
||||||
typedef struct flagbit {
|
typedef struct flagbit {
|
||||||
int bit;
|
int bit;
|
||||||
const char *name;
|
const char *name;
|
||||||
@ -71,7 +73,7 @@ static void signal_handler(int sig) {
|
|||||||
#define EXIT_FAILURE_CHECK_MINOR EXIT_FAILURE
|
#define EXIT_FAILURE_CHECK_MINOR EXIT_FAILURE
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
const char *name;
|
MDBX_val name;
|
||||||
struct {
|
struct {
|
||||||
uint64_t branch, large_count, large_volume, leaf;
|
uint64_t branch, large_count, large_volume, leaf;
|
||||||
uint64_t subleaf_dupsort, leaf_dupfixed, subleaf_dupfixed;
|
uint64_t subleaf_dupsort, leaf_dupfixed, subleaf_dupfixed;
|
||||||
@ -102,7 +104,7 @@ uint64_t total_unused_bytes, reclaimable_pages, gc_pages, alloc_pages,
|
|||||||
unused_pages, backed_pages;
|
unused_pages, backed_pages;
|
||||||
unsigned verbose;
|
unsigned verbose;
|
||||||
bool ignore_wrong_order, quiet, dont_traversal;
|
bool ignore_wrong_order, quiet, dont_traversal;
|
||||||
const char *only_subdb;
|
MDBX_val only_subdb;
|
||||||
int stuck_meta = -1;
|
int stuck_meta = -1;
|
||||||
|
|
||||||
struct problem {
|
struct problem {
|
||||||
@ -125,6 +127,93 @@ static void MDBX_PRINTF_ARGS(1, 2) print(const char *msg, ...) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static MDBX_val printable_buf;
|
||||||
|
static void free_printable_buf(void) { osal_free(printable_buf.iov_base); }
|
||||||
|
|
||||||
|
static const char *sdb_name(const MDBX_val *val) {
|
||||||
|
if (val == MDBX_PGWALK_MAIN)
|
||||||
|
return "@MAIN";
|
||||||
|
if (val == MDBX_PGWALK_GC)
|
||||||
|
return "@GC";
|
||||||
|
if (val == MDBX_PGWALK_META)
|
||||||
|
return "@META";
|
||||||
|
|
||||||
|
const unsigned char *const data = val->iov_base;
|
||||||
|
const size_t len = val->iov_len;
|
||||||
|
if (data == MDBX_PGWALK_MAIN)
|
||||||
|
return "@MAIN";
|
||||||
|
if (data == MDBX_PGWALK_GC)
|
||||||
|
return "@GC";
|
||||||
|
if (data == MDBX_PGWALK_META)
|
||||||
|
return "@META";
|
||||||
|
|
||||||
|
if (!len)
|
||||||
|
return "<zero-length>";
|
||||||
|
if (!data)
|
||||||
|
return "<nullptr>";
|
||||||
|
if (len > 65536) {
|
||||||
|
static char buf[64];
|
||||||
|
snprintf(buf, sizeof(buf), "<too-long-%zu>", len);
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool printable = true;
|
||||||
|
bool quoting = false;
|
||||||
|
size_t xchars = 0;
|
||||||
|
for (size_t i = 0; i < val->iov_len && printable; ++i) {
|
||||||
|
quoting |= data[i] != '_' && isalnum(data[i]) == 0;
|
||||||
|
printable = isprint(data[i]) != 0 ||
|
||||||
|
(data[i] < ' ' && ++xchars < 4 && len > xchars * 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t need = len + 1;
|
||||||
|
if (quoting || !printable)
|
||||||
|
need += len + /* quotes */ 2 + 2 * /* max xchars */ 4;
|
||||||
|
if (need > printable_buf.iov_len) {
|
||||||
|
void *ptr = osal_realloc(printable_buf.iov_base, need);
|
||||||
|
if (!ptr)
|
||||||
|
return "<out-of-memory>";
|
||||||
|
if (!printable_buf.iov_base)
|
||||||
|
atexit(free_printable_buf);
|
||||||
|
printable_buf.iov_base = ptr;
|
||||||
|
printable_buf.iov_len = need;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *out = printable_buf.iov_base;
|
||||||
|
if (!quoting) {
|
||||||
|
memcpy(out, data, len);
|
||||||
|
out += len;
|
||||||
|
} else if (printable) {
|
||||||
|
*out++ = '\'';
|
||||||
|
for (size_t i = 0; i < len; ++i) {
|
||||||
|
if (data[i] < ' ') {
|
||||||
|
assert((char *)printable_buf.iov_base + printable_buf.iov_len >
|
||||||
|
out + 4);
|
||||||
|
static const char hex[] = "0123456789abcdef";
|
||||||
|
out[0] = '\\';
|
||||||
|
out[1] = 'x';
|
||||||
|
out[2] = hex[data[i] >> 4];
|
||||||
|
out[3] = hex[data[i] & 15];
|
||||||
|
out += 4;
|
||||||
|
} else if (strchr("\"'`\\", data[i])) {
|
||||||
|
assert((char *)printable_buf.iov_base + printable_buf.iov_len >
|
||||||
|
out + 2);
|
||||||
|
out[0] = '\\';
|
||||||
|
out[1] = data[i];
|
||||||
|
out += 2;
|
||||||
|
} else {
|
||||||
|
assert((char *)printable_buf.iov_base + printable_buf.iov_len >
|
||||||
|
out + 1);
|
||||||
|
*out++ = data[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*out++ = '\'';
|
||||||
|
}
|
||||||
|
assert((char *)printable_buf.iov_base + printable_buf.iov_len > out);
|
||||||
|
*out = 0;
|
||||||
|
return printable_buf.iov_base;
|
||||||
|
}
|
||||||
|
|
||||||
static void va_log(MDBX_log_level_t level, const char *function, int line,
|
static void va_log(MDBX_log_level_t level, const char *function, int line,
|
||||||
const char *msg, va_list args) {
|
const char *msg, va_list args) {
|
||||||
static const char *const prefixes[] = {
|
static const char *const prefixes[] = {
|
||||||
@ -190,19 +279,17 @@ static int check_user_break(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void pagemap_cleanup(void) {
|
static void pagemap_cleanup(void) {
|
||||||
for (size_t i = CORE_DBS + /* account pseudo-entry for meta */ 1;
|
|
||||||
i < ARRAY_LENGTH(walk.dbi); ++i) {
|
|
||||||
if (walk.dbi[i].name) {
|
|
||||||
osal_free((void *)walk.dbi[i].name);
|
|
||||||
walk.dbi[i].name = nullptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
osal_free(walk.pagemap);
|
osal_free(walk.pagemap);
|
||||||
walk.pagemap = nullptr;
|
walk.pagemap = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static walk_dbi_t *pagemap_lookup_dbi(const char *dbi_name, bool silent) {
|
static bool eq(const MDBX_val a, const MDBX_val b) {
|
||||||
|
return a.iov_len == b.iov_len &&
|
||||||
|
(a.iov_base == b.iov_base || a.iov_len == 0 ||
|
||||||
|
!memcmp(a.iov_base, b.iov_base, a.iov_len));
|
||||||
|
}
|
||||||
|
|
||||||
|
static walk_dbi_t *pagemap_lookup_dbi(const MDBX_val *dbi_name, bool silent) {
|
||||||
static walk_dbi_t *last;
|
static walk_dbi_t *last;
|
||||||
|
|
||||||
if (dbi_name == MDBX_PGWALK_MAIN)
|
if (dbi_name == MDBX_PGWALK_MAIN)
|
||||||
@ -212,24 +299,24 @@ static walk_dbi_t *pagemap_lookup_dbi(const char *dbi_name, bool silent) {
|
|||||||
if (dbi_name == MDBX_PGWALK_META)
|
if (dbi_name == MDBX_PGWALK_META)
|
||||||
return &dbi_meta;
|
return &dbi_meta;
|
||||||
|
|
||||||
if (last && strcmp(last->name, dbi_name) == 0)
|
if (last && eq(last->name, *dbi_name))
|
||||||
return last;
|
return last;
|
||||||
|
|
||||||
walk_dbi_t *dbi = walk.dbi + CORE_DBS + /* account pseudo-entry for meta */ 1;
|
walk_dbi_t *dbi = walk.dbi + CORE_DBS + /* account pseudo-entry for meta */ 1;
|
||||||
for (; dbi < ARRAY_END(walk.dbi) && dbi->name; ++dbi) {
|
for (; dbi < ARRAY_END(walk.dbi) && dbi->name.iov_base; ++dbi) {
|
||||||
if (strcmp(dbi->name, dbi_name) == 0)
|
if (eq(dbi->name, *dbi_name))
|
||||||
return last = dbi;
|
return last = dbi;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (verbose > 0 && !silent) {
|
if (verbose > 0 && !silent) {
|
||||||
print(" - found `%s` area\n", dbi_name);
|
print(" - found %s area\n", sdb_name(dbi_name));
|
||||||
fflush(nullptr);
|
fflush(nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dbi == ARRAY_END(walk.dbi))
|
if (dbi == ARRAY_END(walk.dbi))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
dbi->name = osal_strdup(dbi_name);
|
dbi->name = *dbi_name;
|
||||||
return last = dbi;
|
return last = dbi;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -304,13 +391,13 @@ static size_t problems_pop(struct problem *list) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int pgvisitor(const uint64_t pgno, const unsigned pgnumber,
|
static int pgvisitor(const uint64_t pgno, const unsigned pgnumber,
|
||||||
void *const ctx, const int deep,
|
void *const ctx, const int deep, const MDBX_val *dbi_name,
|
||||||
const char *const dbi_name_or_tag, const size_t page_size,
|
const size_t page_size, const MDBX_page_type_t pagetype,
|
||||||
const MDBX_page_type_t pagetype, const MDBX_error_t err,
|
const MDBX_error_t err, const size_t nentries,
|
||||||
const size_t nentries, const size_t payload_bytes,
|
const size_t payload_bytes, const size_t header_bytes,
|
||||||
const size_t header_bytes, const size_t unused_bytes) {
|
const size_t unused_bytes) {
|
||||||
(void)ctx;
|
(void)ctx;
|
||||||
const bool is_gc_tree = dbi_name_or_tag == MDBX_PGWALK_GC;
|
const bool is_gc_tree = dbi_name == MDBX_PGWALK_GC;
|
||||||
if (deep > 42) {
|
if (deep > 42) {
|
||||||
problem_add("deep", deep, "too large", nullptr);
|
problem_add("deep", deep, "too large", nullptr);
|
||||||
data_tree_problems += !is_gc_tree;
|
data_tree_problems += !is_gc_tree;
|
||||||
@ -318,7 +405,7 @@ static int pgvisitor(const uint64_t pgno, const unsigned pgnumber,
|
|||||||
return MDBX_CORRUPTED /* avoid infinite loop/recursion */;
|
return MDBX_CORRUPTED /* avoid infinite loop/recursion */;
|
||||||
}
|
}
|
||||||
|
|
||||||
walk_dbi_t *dbi = pagemap_lookup_dbi(dbi_name_or_tag, false);
|
walk_dbi_t *dbi = pagemap_lookup_dbi(dbi_name, false);
|
||||||
if (!dbi) {
|
if (!dbi) {
|
||||||
data_tree_problems += !is_gc_tree;
|
data_tree_problems += !is_gc_tree;
|
||||||
gc_tree_problems += is_gc_tree;
|
gc_tree_problems += is_gc_tree;
|
||||||
@ -383,14 +470,14 @@ static int pgvisitor(const uint64_t pgno, const unsigned pgnumber,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (pgnumber) {
|
if (pgnumber) {
|
||||||
if (verbose > 3 && (!only_subdb || strcmp(only_subdb, dbi->name) == 0)) {
|
if (verbose > 3 && (!only_subdb.iov_base || eq(only_subdb, dbi->name))) {
|
||||||
if (pgnumber == 1)
|
if (pgnumber == 1)
|
||||||
print(" %s-page %" PRIu64, pagetype_caption, pgno);
|
print(" %s-page %" PRIu64, pagetype_caption, pgno);
|
||||||
else
|
else
|
||||||
print(" %s-span %" PRIu64 "[%u]", pagetype_caption, pgno, pgnumber);
|
print(" %s-span %" PRIu64 "[%u]", pagetype_caption, pgno, pgnumber);
|
||||||
print(" of %s: header %" PRIiPTR ", %s %" PRIiPTR ", payload %" PRIiPTR
|
print(" of %s: header %" PRIiPTR ", %s %" PRIiPTR ", payload %" PRIiPTR
|
||||||
", unused %" PRIiPTR ", deep %i\n",
|
", unused %" PRIiPTR ", deep %i\n",
|
||||||
dbi->name, header_bytes,
|
sdb_name(&dbi->name), header_bytes,
|
||||||
(pagetype == MDBX_page_branch) ? "keys" : "entries", nentries,
|
(pagetype == MDBX_page_branch) ? "keys" : "entries", nentries,
|
||||||
payload_bytes, unused_bytes, deep);
|
payload_bytes, unused_bytes, deep);
|
||||||
}
|
}
|
||||||
@ -408,8 +495,8 @@ static int pgvisitor(const uint64_t pgno, const unsigned pgnumber,
|
|||||||
walk_dbi_t *coll_dbi = &walk.dbi[walk.pagemap[spanpgno] - 1];
|
walk_dbi_t *coll_dbi = &walk.dbi[walk.pagemap[spanpgno] - 1];
|
||||||
problem_add("page", spanpgno,
|
problem_add("page", spanpgno,
|
||||||
(branch && coll_dbi == dbi) ? "loop" : "already used",
|
(branch && coll_dbi == dbi) ? "loop" : "already used",
|
||||||
"%s-page: by %s, deep %i", pagetype_caption, coll_dbi->name,
|
"%s-page: by %s, deep %i", pagetype_caption,
|
||||||
deep);
|
sdb_name(&coll_dbi->name), deep);
|
||||||
already_used = true;
|
already_used = true;
|
||||||
data_tree_problems += !is_gc_tree;
|
data_tree_problems += !is_gc_tree;
|
||||||
gc_tree_problems += is_gc_tree;
|
gc_tree_problems += is_gc_tree;
|
||||||
@ -491,7 +578,8 @@ static int pgvisitor(const uint64_t pgno, const unsigned pgnumber,
|
|||||||
|
|
||||||
typedef int(visitor)(const uint64_t record_number, const MDBX_val *key,
|
typedef int(visitor)(const uint64_t record_number, const MDBX_val *key,
|
||||||
const MDBX_val *data);
|
const MDBX_val *data);
|
||||||
static int process_db(MDBX_dbi dbi_handle, char *dbi_name, visitor *handler);
|
static int process_db(MDBX_dbi dbi_handle, const MDBX_val *dbi_name,
|
||||||
|
visitor *handler);
|
||||||
|
|
||||||
static int handle_userdb(const uint64_t record_number, const MDBX_val *key,
|
static int handle_userdb(const uint64_t record_number, const MDBX_val *key,
|
||||||
const MDBX_val *data) {
|
const MDBX_val *data) {
|
||||||
@ -569,7 +657,7 @@ static int handle_freedb(const uint64_t record_number, const MDBX_val *key,
|
|||||||
walk.pagemap[pgno] = -1;
|
walk.pagemap[pgno] = -1;
|
||||||
else if (idx > 0)
|
else if (idx > 0)
|
||||||
problem_add("page", pgno, "already used", "by %s",
|
problem_add("page", pgno, "already used", "by %s",
|
||||||
walk.dbi[idx - 1].name);
|
sdb_name(&walk.dbi[idx - 1].name));
|
||||||
else
|
else
|
||||||
problem_add("page", pgno, "already listed in GC", nullptr);
|
problem_add("page", pgno, "already listed in GC", nullptr);
|
||||||
}
|
}
|
||||||
@ -580,7 +668,7 @@ static int handle_freedb(const uint64_t record_number, const MDBX_val *key,
|
|||||||
: pgno_sub(pgno, span)))
|
: pgno_sub(pgno, span)))
|
||||||
++span;
|
++span;
|
||||||
}
|
}
|
||||||
if (verbose > 3 && !only_subdb) {
|
if (verbose > 3 && !only_subdb.iov_base) {
|
||||||
print(" transaction %" PRIaTXN ", %" PRIuPTR
|
print(" transaction %" PRIaTXN ", %" PRIuPTR
|
||||||
" pages, maxspan %" PRIaPGNO "%s\n",
|
" pages, maxspan %" PRIaPGNO "%s\n",
|
||||||
txnid, number, span, bad);
|
txnid, number, span, bad);
|
||||||
@ -607,39 +695,18 @@ static int handle_freedb(const uint64_t record_number, const MDBX_val *key,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int equal_or_greater(const MDBX_val *a, const MDBX_val *b) {
|
static int equal_or_greater(const MDBX_val *a, const MDBX_val *b) {
|
||||||
return (a->iov_len == b->iov_len &&
|
return eq(*a, *b) ? 0 : 1;
|
||||||
memcmp(a->iov_base, b->iov_base, a->iov_len) == 0)
|
|
||||||
? 0
|
|
||||||
: 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int handle_maindb(const uint64_t record_number, const MDBX_val *key,
|
static int handle_maindb(const uint64_t record_number, const MDBX_val *key,
|
||||||
const MDBX_val *data) {
|
const MDBX_val *data) {
|
||||||
char *name;
|
if (data->iov_len == sizeof(MDBX_db)) {
|
||||||
int rc;
|
int rc = process_db(~0u, key, handle_userdb);
|
||||||
size_t i;
|
|
||||||
|
|
||||||
if (data->iov_len != sizeof(MDBX_db))
|
|
||||||
return handle_userdb(record_number, key, data);
|
|
||||||
name = key->iov_base;
|
|
||||||
for (i = 0; i < key->iov_len; ++i) {
|
|
||||||
if (name[i] < ' ')
|
|
||||||
return handle_userdb(record_number, key, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
name = osal_malloc(key->iov_len + 1);
|
|
||||||
if (unlikely(!name))
|
|
||||||
return MDBX_ENOMEM;
|
|
||||||
memcpy(name, key->iov_base, key->iov_len);
|
|
||||||
name[key->iov_len] = '\0';
|
|
||||||
|
|
||||||
rc = process_db(~0u, name, handle_userdb);
|
|
||||||
osal_free(name);
|
|
||||||
if (rc != MDBX_INCOMPATIBLE) {
|
if (rc != MDBX_INCOMPATIBLE) {
|
||||||
userdb_count++;
|
userdb_count++;
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return handle_userdb(record_number, key, data);
|
return handle_userdb(record_number, key, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -693,7 +760,8 @@ static const char *db_flags2valuemode(unsigned flags) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int process_db(MDBX_dbi dbi_handle, char *dbi_name, visitor *handler) {
|
static int process_db(MDBX_dbi dbi_handle, const MDBX_val *dbi_name,
|
||||||
|
visitor *handler) {
|
||||||
MDBX_cursor *mc;
|
MDBX_cursor *mc;
|
||||||
MDBX_stat ms;
|
MDBX_stat ms;
|
||||||
MDBX_val key, data;
|
MDBX_val key, data;
|
||||||
@ -708,13 +776,13 @@ static int process_db(MDBX_dbi dbi_handle, char *dbi_name, visitor *handler) {
|
|||||||
uint64_t key_bytes = 0, data_bytes = 0;
|
uint64_t key_bytes = 0, data_bytes = 0;
|
||||||
|
|
||||||
if ((MDBX_TXN_FINISHED | MDBX_TXN_ERROR) & mdbx_txn_flags(txn)) {
|
if ((MDBX_TXN_FINISHED | MDBX_TXN_ERROR) & mdbx_txn_flags(txn)) {
|
||||||
print(" ! abort processing `%s` due to a previous error\n",
|
print(" ! abort processing %s due to a previous error\n",
|
||||||
dbi_name ? dbi_name : "@MAIN");
|
sdb_name(dbi_name));
|
||||||
return MDBX_BAD_TXN;
|
return MDBX_BAD_TXN;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dbi_handle == ~0u) {
|
if (dbi_handle == ~0u) {
|
||||||
rc = mdbx_dbi_open_ex(
|
rc = mdbx_dbi_open_ex2(
|
||||||
txn, dbi_name, MDBX_DB_ACCEDE, &dbi_handle,
|
txn, dbi_name, MDBX_DB_ACCEDE, &dbi_handle,
|
||||||
(dbi_name && ignore_wrong_order) ? equal_or_greater : nullptr,
|
(dbi_name && ignore_wrong_order) ? equal_or_greater : nullptr,
|
||||||
(dbi_name && ignore_wrong_order) ? equal_or_greater : nullptr);
|
(dbi_name && ignore_wrong_order) ? equal_or_greater : nullptr);
|
||||||
@ -722,17 +790,17 @@ static int process_db(MDBX_dbi dbi_handle, char *dbi_name, visitor *handler) {
|
|||||||
if (!dbi_name ||
|
if (!dbi_name ||
|
||||||
rc !=
|
rc !=
|
||||||
MDBX_INCOMPATIBLE) /* LY: mainDB's record is not a user's DB. */ {
|
MDBX_INCOMPATIBLE) /* LY: mainDB's record is not a user's DB. */ {
|
||||||
error("mdbx_dbi_open(`%s`) failed, error %d %s\n",
|
error("mdbx_dbi_open(%s) failed, error %d %s\n", sdb_name(dbi_name), rc,
|
||||||
dbi_name ? dbi_name : "@MAIN", rc, mdbx_strerror(rc));
|
mdbx_strerror(rc));
|
||||||
}
|
}
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dbi_handle >= CORE_DBS && dbi_name && only_subdb &&
|
if (dbi_handle >= CORE_DBS && dbi_name && only_subdb.iov_base &&
|
||||||
strcmp(only_subdb, dbi_name) != 0) {
|
!eq(only_subdb, *dbi_name)) {
|
||||||
if (verbose) {
|
if (verbose) {
|
||||||
print("Skip processing %s...\n", dbi_name);
|
print("Skip processing %s...\n", sdb_name(dbi_name));
|
||||||
fflush(nullptr);
|
fflush(nullptr);
|
||||||
}
|
}
|
||||||
skipped_subdb++;
|
skipped_subdb++;
|
||||||
@ -740,7 +808,7 @@ static int process_db(MDBX_dbi dbi_handle, char *dbi_name, visitor *handler) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!second_pass && verbose)
|
if (!second_pass && verbose)
|
||||||
print("Processing %s...\n", dbi_name ? dbi_name : "@MAIN");
|
print("Processing %s...\n", sdb_name(dbi_name));
|
||||||
fflush(nullptr);
|
fflush(nullptr);
|
||||||
|
|
||||||
rc = mdbx_dbi_flags(txn, dbi_handle, &flags);
|
rc = mdbx_dbi_flags(txn, dbi_handle, &flags);
|
||||||
@ -1110,9 +1178,9 @@ int main(int argc, char *argv[]) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
dbi_meta.name = "@META";
|
dbi_meta.name.iov_base = MDBX_PGWALK_META;
|
||||||
dbi_free.name = "@GC";
|
dbi_free.name.iov_base = MDBX_PGWALK_GC;
|
||||||
dbi_main.name = "@MAIN";
|
dbi_main.name.iov_base = MDBX_PGWALK_MAIN;
|
||||||
atexit(pagemap_cleanup);
|
atexit(pagemap_cleanup);
|
||||||
|
|
||||||
if (argc < 2)
|
if (argc < 2)
|
||||||
@ -1190,9 +1258,10 @@ int main(int argc, char *argv[]) {
|
|||||||
dont_traversal = true;
|
dont_traversal = true;
|
||||||
break;
|
break;
|
||||||
case 's':
|
case 's':
|
||||||
if (only_subdb && strcmp(only_subdb, optarg))
|
if (only_subdb.iov_base && strcmp(only_subdb.iov_base, optarg))
|
||||||
usage(prog);
|
usage(prog);
|
||||||
only_subdb = optarg;
|
only_subdb.iov_base = optarg;
|
||||||
|
only_subdb.iov_len = strlen(optarg);
|
||||||
break;
|
break;
|
||||||
case 'i':
|
case 'i':
|
||||||
ignore_wrong_order = true;
|
ignore_wrong_order = true;
|
||||||
@ -1230,7 +1299,7 @@ int main(int argc, char *argv[]) {
|
|||||||
error("write-mode must be enabled to turn to the specified meta-page.\n");
|
error("write-mode must be enabled to turn to the specified meta-page.\n");
|
||||||
rc = EXIT_INTERRUPTED;
|
rc = EXIT_INTERRUPTED;
|
||||||
}
|
}
|
||||||
if (only_subdb || dont_traversal) {
|
if (only_subdb.iov_base || dont_traversal) {
|
||||||
error(
|
error(
|
||||||
"whole database checking with b-tree traversal are required to turn "
|
"whole database checking with b-tree traversal are required to turn "
|
||||||
"to the specified meta-page.\n");
|
"to the specified meta-page.\n");
|
||||||
@ -1572,8 +1641,8 @@ int main(int argc, char *argv[]) {
|
|||||||
unused_pages += 1;
|
unused_pages += 1;
|
||||||
|
|
||||||
empty_pages = lost_bytes = 0;
|
empty_pages = lost_bytes = 0;
|
||||||
for (walk_dbi_t *dbi = &dbi_main; dbi < ARRAY_END(walk.dbi) && dbi->name;
|
for (walk_dbi_t *dbi = &dbi_main;
|
||||||
++dbi) {
|
dbi < ARRAY_END(walk.dbi) && dbi->name.iov_base; ++dbi) {
|
||||||
empty_pages += dbi->pages.empty;
|
empty_pages += dbi->pages.empty;
|
||||||
lost_bytes += dbi->lost_bytes;
|
lost_bytes += dbi->lost_bytes;
|
||||||
}
|
}
|
||||||
@ -1583,9 +1652,10 @@ int main(int argc, char *argv[]) {
|
|||||||
print(" - pages: walked %" PRIu64 ", left/unused %" PRIu64 "\n",
|
print(" - pages: walked %" PRIu64 ", left/unused %" PRIu64 "\n",
|
||||||
walk.pgcount, unused_pages);
|
walk.pgcount, unused_pages);
|
||||||
if (verbose > 1) {
|
if (verbose > 1) {
|
||||||
for (walk_dbi_t *dbi = walk.dbi; dbi < ARRAY_END(walk.dbi) && dbi->name;
|
for (walk_dbi_t *dbi = walk.dbi;
|
||||||
++dbi) {
|
dbi < ARRAY_END(walk.dbi) && dbi->name.iov_base; ++dbi) {
|
||||||
print(" %s: subtotal %" PRIu64, dbi->name, dbi->pages.total);
|
print(" %s: subtotal %" PRIu64, sdb_name(&dbi->name),
|
||||||
|
dbi->pages.total);
|
||||||
if (dbi->pages.other && dbi->pages.other != dbi->pages.total)
|
if (dbi->pages.other && dbi->pages.other != dbi->pages.total)
|
||||||
print(", other %" PRIu64, dbi->pages.other);
|
print(", other %" PRIu64, dbi->pages.other);
|
||||||
if (dbi->pages.branch)
|
if (dbi->pages.branch)
|
||||||
@ -1617,14 +1687,15 @@ int main(int argc, char *argv[]) {
|
|||||||
(total_page_bytes - walk.total_payload_bytes) * 100.0 /
|
(total_page_bytes - walk.total_payload_bytes) * 100.0 /
|
||||||
total_page_bytes);
|
total_page_bytes);
|
||||||
if (verbose > 2) {
|
if (verbose > 2) {
|
||||||
for (walk_dbi_t *dbi = walk.dbi; dbi < ARRAY_END(walk.dbi) && dbi->name;
|
for (walk_dbi_t *dbi = walk.dbi;
|
||||||
++dbi)
|
dbi < ARRAY_END(walk.dbi) && dbi->name.iov_base; ++dbi)
|
||||||
if (dbi->pages.total) {
|
if (dbi->pages.total) {
|
||||||
uint64_t dbi_bytes = dbi->pages.total * envinfo.mi_dxb_pagesize;
|
uint64_t dbi_bytes = dbi->pages.total * envinfo.mi_dxb_pagesize;
|
||||||
print(" %s: subtotal %" PRIu64 " bytes (%.1f%%),"
|
print(" %s: subtotal %" PRIu64 " bytes (%.1f%%),"
|
||||||
" payload %" PRIu64 " (%.1f%%), unused %" PRIu64 " (%.1f%%)",
|
" payload %" PRIu64 " (%.1f%%), unused %" PRIu64 " (%.1f%%)",
|
||||||
dbi->name, dbi_bytes, dbi_bytes * 100.0 / total_page_bytes,
|
sdb_name(&dbi->name), dbi_bytes,
|
||||||
dbi->payload_bytes, dbi->payload_bytes * 100.0 / dbi_bytes,
|
dbi_bytes * 100.0 / total_page_bytes, dbi->payload_bytes,
|
||||||
|
dbi->payload_bytes * 100.0 / dbi_bytes,
|
||||||
dbi_bytes - dbi->payload_bytes,
|
dbi_bytes - dbi->payload_bytes,
|
||||||
(dbi_bytes - dbi->payload_bytes) * 100.0 / dbi_bytes);
|
(dbi_bytes - dbi->payload_bytes) * 100.0 / dbi_bytes);
|
||||||
if (dbi->pages.empty)
|
if (dbi->pages.empty)
|
||||||
@ -1633,7 +1704,7 @@ int main(int argc, char *argv[]) {
|
|||||||
print(", %" PRIu64 " bytes lost", dbi->lost_bytes);
|
print(", %" PRIu64 " bytes lost", dbi->lost_bytes);
|
||||||
print("\n");
|
print("\n");
|
||||||
} else
|
} else
|
||||||
print(" %s: empty\n", dbi->name);
|
print(" %s: empty\n", sdb_name(&dbi->name));
|
||||||
}
|
}
|
||||||
print(" - summary: average fill %.1f%%",
|
print(" - summary: average fill %.1f%%",
|
||||||
walk.total_payload_bytes * 100.0 / total_page_bytes);
|
walk.total_payload_bytes * 100.0 / total_page_bytes);
|
||||||
@ -1653,7 +1724,7 @@ int main(int argc, char *argv[]) {
|
|||||||
"b-tree", gc_tree_problems);
|
"b-tree", gc_tree_problems);
|
||||||
problems_freedb = gc_tree_problems;
|
problems_freedb = gc_tree_problems;
|
||||||
} else
|
} else
|
||||||
problems_freedb = process_db(FREE_DBI, "@GC", handle_freedb);
|
problems_freedb = process_db(FREE_DBI, MDBX_PGWALK_GC, handle_freedb);
|
||||||
|
|
||||||
if (verbose) {
|
if (verbose) {
|
||||||
uint64_t value = envinfo.mi_mapsize / envinfo.mi_dxb_pagesize;
|
uint64_t value = envinfo.mi_mapsize / envinfo.mi_dxb_pagesize;
|
||||||
@ -1719,7 +1790,7 @@ int main(int argc, char *argv[]) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (rc == 0 && total_problems == 1 && problems_meta == 1 && !dont_traversal &&
|
if (rc == 0 && total_problems == 1 && problems_meta == 1 && !dont_traversal &&
|
||||||
(envflags & MDBX_RDONLY) == 0 && !only_subdb && stuck_meta < 0 &&
|
(envflags & MDBX_RDONLY) == 0 && !only_subdb.iov_base && stuck_meta < 0 &&
|
||||||
get_meta_txnid(meta_recent(true)) < envinfo.mi_recent_txnid) {
|
get_meta_txnid(meta_recent(true)) < envinfo.mi_recent_txnid) {
|
||||||
print("Perform sync-to-disk for make steady checkpoint at txn-id #%" PRIi64
|
print("Perform sync-to-disk for make steady checkpoint at txn-id #%" PRIi64
|
||||||
"\n",
|
"\n",
|
||||||
@ -1738,7 +1809,7 @@ int main(int argc, char *argv[]) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (turn_meta && stuck_meta >= 0 && !dont_traversal && !only_subdb &&
|
if (turn_meta && stuck_meta >= 0 && !dont_traversal && !only_subdb.iov_base &&
|
||||||
(envflags & (MDBX_RDONLY | MDBX_EXCLUSIVE)) == MDBX_EXCLUSIVE) {
|
(envflags & (MDBX_RDONLY | MDBX_EXCLUSIVE)) == MDBX_EXCLUSIVE) {
|
||||||
const bool successful_check = (rc | total_problems | problems_meta) == 0;
|
const bool successful_check = (rc | total_problems | problems_meta) == 0;
|
||||||
if (successful_check || force_turn_meta) {
|
if (successful_check || force_turn_meta) {
|
||||||
|
@ -66,7 +66,7 @@ static const char hexc[] = "0123456789abcdef";
|
|||||||
|
|
||||||
static void dumpbyte(unsigned char c) {
|
static void dumpbyte(unsigned char c) {
|
||||||
putchar(hexc[c >> 4]);
|
putchar(hexc[c >> 4]);
|
||||||
putchar(hexc[c & 0xf]);
|
putchar(hexc[c & 15]);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void text(MDBX_val *v) {
|
static void text(MDBX_val *v) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user