summaryrefslogtreecommitdiff
path: root/db-4.8.30/txn/txn_stat.c
diff options
context:
space:
mode:
authorJesse Morgan <jesse@jesterpm.net>2016-12-17 21:28:53 -0800
committerJesse Morgan <jesse@jesterpm.net>2016-12-17 21:28:53 -0800
commit54df2afaa61c6a03cbb4a33c9b90fa572b6d07b8 (patch)
tree18147b92b969d25ffbe61935fb63035cac820dd0 /db-4.8.30/txn/txn_stat.c
Berkeley DB 4.8 with rust build script for linux.
Diffstat (limited to 'db-4.8.30/txn/txn_stat.c')
-rw-r--r--db-4.8.30/txn/txn_stat.c437
1 files changed, 437 insertions, 0 deletions
diff --git a/db-4.8.30/txn/txn_stat.c b/db-4.8.30/txn/txn_stat.c
new file mode 100644
index 0000000..7793e09
--- /dev/null
+++ b/db-4.8.30/txn/txn_stat.c
@@ -0,0 +1,437 @@
+/*-
+ * See the file LICENSE for redistribution information.
+ *
+ * Copyright (c) 1996-2009 Oracle. All rights reserved.
+ *
+ * $Id$
+ */
+
+#include "db_config.h"
+
+#include "db_int.h"
+#include "dbinc/db_page.h"
+#include "dbinc/db_am.h"
+#include "dbinc/log.h"
+#include "dbinc/txn.h"
+
+#ifdef HAVE_STATISTICS
+static int __txn_compare __P((const void *, const void *));
+static int __txn_print_all __P((ENV *, u_int32_t));
+static int __txn_print_stats __P((ENV *, u_int32_t));
+static int __txn_stat __P((ENV *, DB_TXN_STAT **, u_int32_t));
+static char *__txn_status __P((DB_TXN_ACTIVE *));
+static void __txn_gid __P((ENV *, DB_MSGBUF *, DB_TXN_ACTIVE *));
+
+/*
+ * __txn_stat_pp --
+ * DB_ENV->txn_stat pre/post processing.
+ *
+ * PUBLIC: int __txn_stat_pp __P((DB_ENV *, DB_TXN_STAT **, u_int32_t));
+ */
+int
+__txn_stat_pp(dbenv, statp, flags)
+ DB_ENV *dbenv;
+ DB_TXN_STAT **statp;
+ u_int32_t flags;
+{
+ DB_THREAD_INFO *ip;
+ ENV *env;
+ int ret;
+
+ env = dbenv->env;
+
+ ENV_REQUIRES_CONFIG(env,
+ env->tx_handle, "DB_ENV->txn_stat", DB_INIT_TXN);
+
+ if ((ret = __db_fchk(env,
+ "DB_ENV->txn_stat", flags, DB_STAT_CLEAR)) != 0)
+ return (ret);
+
+ ENV_ENTER(env, ip);
+ REPLICATION_WRAP(env, (__txn_stat(env, statp, flags)), 0, ret);
+ ENV_LEAVE(env, ip);
+ return (ret);
+}
+
+/*
+ * __txn_stat --
+ * ENV->txn_stat.
+ */
+static int
+__txn_stat(env, statp, flags)
+ ENV *env;
+ DB_TXN_STAT **statp;
+ u_int32_t flags;
+{
+ DB_TXNMGR *mgr;
+ DB_TXNREGION *region;
+ DB_TXN_STAT *stats;
+ TXN_DETAIL *td;
+ size_t nbytes;
+ u_int32_t maxtxn, ndx;
+ int ret;
+
+ *statp = NULL;
+ mgr = env->tx_handle;
+ region = mgr->reginfo.primary;
+
+ /*
+ * Allocate for the maximum active transactions -- the DB_TXN_ACTIVE
+ * struct is small and the maximum number of active transactions is
+ * not going to be that large. Don't have to lock anything to look
+ * at the region's maximum active transactions value, it's read-only
+ * and never changes after the region is created.
+ *
+ * The maximum active transactions isn't a hard limit, so allocate
+ * some extra room, and don't walk off the end.
+ */
+ maxtxn = region->maxtxns + (region->maxtxns / 10) + 10;
+ nbytes = sizeof(DB_TXN_STAT) + sizeof(DB_TXN_ACTIVE) * maxtxn;
+ if ((ret = __os_umalloc(env, nbytes, &stats)) != 0)
+ return (ret);
+
+ TXN_SYSTEM_LOCK(env);
+ memcpy(stats, &region->stat, sizeof(*stats));
+ stats->st_last_txnid = region->last_txnid;
+ stats->st_last_ckp = region->last_ckp;
+ stats->st_time_ckp = region->time_ckp;
+ stats->st_txnarray = (DB_TXN_ACTIVE *)&stats[1];
+
+ for (ndx = 0,
+ td = SH_TAILQ_FIRST(&region->active_txn, __txn_detail);
+ td != NULL && ndx < maxtxn;
+ td = SH_TAILQ_NEXT(td, links, __txn_detail), ++ndx) {
+ stats->st_txnarray[ndx].txnid = td->txnid;
+ if (td->parent == INVALID_ROFF)
+ stats->st_txnarray[ndx].parentid = TXN_INVALID;
+ else
+ stats->st_txnarray[ndx].parentid =
+ ((TXN_DETAIL *)R_ADDR(&mgr->reginfo,
+ td->parent))->txnid;
+ stats->st_txnarray[ndx].pid = td->pid;
+ stats->st_txnarray[ndx].tid = td->tid;
+ stats->st_txnarray[ndx].lsn = td->begin_lsn;
+ stats->st_txnarray[ndx].read_lsn = td->read_lsn;
+ stats->st_txnarray[ndx].mvcc_ref = td->mvcc_ref;
+ stats->st_txnarray[ndx].status = td->status;
+ if (td->status == TXN_PREPARED)
+ memcpy(stats->st_txnarray[ndx].gid,
+ td->gid, sizeof(td->gid));
+ if (td->name != INVALID_ROFF) {
+ (void)strncpy(stats->st_txnarray[ndx].name,
+ R_ADDR(&mgr->reginfo, td->name),
+ sizeof(stats->st_txnarray[ndx].name) - 1);
+ stats->st_txnarray[ndx].name[
+ sizeof(stats->st_txnarray[ndx].name) - 1] = '\0';
+ } else
+ stats->st_txnarray[ndx].name[0] = '\0';
+ }
+
+ __mutex_set_wait_info(env, region->mtx_region,
+ &stats->st_region_wait, &stats->st_region_nowait);
+ stats->st_regsize = mgr->reginfo.rp->size;
+ if (LF_ISSET(DB_STAT_CLEAR)) {
+ if (!LF_ISSET(DB_STAT_SUBSYSTEM))
+ __mutex_clear(env, region->mtx_region);
+ memset(&region->stat, 0, sizeof(region->stat));
+ region->stat.st_maxtxns = region->maxtxns;
+ region->stat.st_maxnactive =
+ region->stat.st_nactive = stats->st_nactive;
+ region->stat.st_maxnsnapshot =
+ region->stat.st_nsnapshot = stats->st_nsnapshot;
+ }
+
+ TXN_SYSTEM_UNLOCK(env);
+
+ *statp = stats;
+ return (0);
+}
+
+/*
+ * __txn_stat_print_pp --
+ * DB_ENV->txn_stat_print pre/post processing.
+ *
+ * PUBLIC: int __txn_stat_print_pp __P((DB_ENV *, u_int32_t));
+ */
+int
+__txn_stat_print_pp(dbenv, flags)
+ DB_ENV *dbenv;
+ u_int32_t flags;
+{
+ DB_THREAD_INFO *ip;
+ ENV *env;
+ int ret;
+
+ env = dbenv->env;
+
+ ENV_REQUIRES_CONFIG(env,
+ env->tx_handle, "DB_ENV->txn_stat_print", DB_INIT_TXN);
+
+ if ((ret = __db_fchk(env, "DB_ENV->txn_stat_print",
+ flags, DB_STAT_ALL | DB_STAT_CLEAR)) != 0)
+ return (ret);
+
+ ENV_ENTER(env, ip);
+ REPLICATION_WRAP(env, (__txn_stat_print(env, flags)), 0, ret);
+ ENV_LEAVE(env, ip);
+ return (ret);
+}
+
+/*
+ * __txn_stat_print
+ * ENV->txn_stat_print method.
+ *
+ * PUBLIC: int __txn_stat_print __P((ENV *, u_int32_t));
+ */
+int
+__txn_stat_print(env, flags)
+ ENV *env;
+ u_int32_t flags;
+{
+ u_int32_t orig_flags;
+ int ret;
+
+ orig_flags = flags;
+ LF_CLR(DB_STAT_CLEAR | DB_STAT_SUBSYSTEM);
+ if (flags == 0 || LF_ISSET(DB_STAT_ALL)) {
+ ret = __txn_print_stats(env, orig_flags);
+ if (flags == 0 || ret != 0)
+ return (ret);
+ }
+
+ if (LF_ISSET(DB_STAT_ALL) &&
+ (ret = __txn_print_all(env, orig_flags)) != 0)
+ return (ret);
+
+ return (0);
+}
+
+/*
+ * __txn_print_stats --
+ * Display default transaction region statistics.
+ */
+static int
+__txn_print_stats(env, flags)
+ ENV *env;
+ u_int32_t flags;
+{
+ DB_ENV *dbenv;
+ DB_MSGBUF mb;
+ DB_TXN_ACTIVE *txn;
+ DB_TXN_STAT *sp;
+ u_int32_t i;
+ int ret;
+ char buf[DB_THREADID_STRLEN], time_buf[CTIME_BUFLEN];
+
+ dbenv = env->dbenv;
+
+ if ((ret = __txn_stat(env, &sp, flags)) != 0)
+ return (ret);
+
+ if (LF_ISSET(DB_STAT_ALL))
+ __db_msg(env, "Default transaction region information:");
+ __db_msg(env, "%lu/%lu\t%s",
+ (u_long)sp->st_last_ckp.file, (u_long)sp->st_last_ckp.offset,
+ sp->st_last_ckp.file == 0 ?
+ "No checkpoint LSN" : "File/offset for last checkpoint LSN");
+ if (sp->st_time_ckp == 0)
+ __db_msg(env, "0\tNo checkpoint timestamp");
+ else
+ __db_msg(env, "%.24s\tCheckpoint timestamp",
+ __os_ctime(&sp->st_time_ckp, time_buf));
+ __db_msg(env, "%#lx\tLast transaction ID allocated",
+ (u_long)sp->st_last_txnid);
+ __db_dl(env, "Maximum number of active transactions configured",
+ (u_long)sp->st_maxtxns);
+ __db_dl(env, "Active transactions", (u_long)sp->st_nactive);
+ __db_dl(env,
+ "Maximum active transactions", (u_long)sp->st_maxnactive);
+ __db_dl(env,
+ "Number of transactions begun", (u_long)sp->st_nbegins);
+ __db_dl(env,
+ "Number of transactions aborted", (u_long)sp->st_naborts);
+ __db_dl(env,
+ "Number of transactions committed", (u_long)sp->st_ncommits);
+ __db_dl(env, "Snapshot transactions", (u_long)sp->st_nsnapshot);
+ __db_dl(env, "Maximum snapshot transactions",
+ (u_long)sp->st_maxnsnapshot);
+ __db_dl(env,
+ "Number of transactions restored", (u_long)sp->st_nrestores);
+
+ __db_dlbytes(env, "Transaction region size",
+ (u_long)0, (u_long)0, (u_long)sp->st_regsize);
+ __db_dl_pct(env,
+ "The number of region locks that required waiting",
+ (u_long)sp->st_region_wait, DB_PCT(sp->st_region_wait,
+ sp->st_region_wait + sp->st_region_nowait), NULL);
+
+ qsort(sp->st_txnarray,
+ sp->st_nactive, sizeof(sp->st_txnarray[0]), __txn_compare);
+ __db_msg(env, "Active transactions:");
+ DB_MSGBUF_INIT(&mb);
+ for (i = 0; i < sp->st_nactive; ++i) {
+ txn = &sp->st_txnarray[i];
+ __db_msgadd(env, &mb,
+ "\t%lx: %s; pid/thread %s; begin LSN: file/offset %lu/%lu",
+ (u_long)txn->txnid, __txn_status(txn),
+ dbenv->thread_id_string(dbenv, txn->pid, txn->tid, buf),
+ (u_long)txn->lsn.file, (u_long)txn->lsn.offset);
+ if (txn->parentid != 0)
+ __db_msgadd(env, &mb,
+ "; parent: %lx", (u_long)txn->parentid);
+ if (!IS_MAX_LSN(txn->read_lsn))
+ __db_msgadd(env, &mb, "; read LSN: %lu/%lu",
+ (u_long)txn->read_lsn.file,
+ (u_long)txn->read_lsn.offset);
+ if (txn->mvcc_ref != 0)
+ __db_msgadd(env, &mb,
+ "; mvcc refcount: %lu", (u_long)txn->mvcc_ref);
+ if (txn->name[0] != '\0')
+ __db_msgadd(env, &mb, "; \"%s\"", txn->name);
+ if (txn->status == TXN_PREPARE)
+ __txn_gid(env, &mb, txn);
+ DB_MSGBUF_FLUSH(env, &mb);
+ }
+
+ __os_ufree(env, sp);
+
+ return (0);
+}
+
+/*
+ * __txn_print_all --
+ * Display debugging transaction region statistics.
+ */
+static int
+__txn_print_all(env, flags)
+ ENV *env;
+ u_int32_t flags;
+{
+ static const FN fn[] = {
+ { TXN_IN_RECOVERY, "TXN_IN_RECOVERY" },
+ { 0, NULL }
+ };
+ DB_TXNMGR *mgr;
+ DB_TXNREGION *region;
+ char time_buf[CTIME_BUFLEN];
+
+ mgr = env->tx_handle;
+ region = mgr->reginfo.primary;
+
+ TXN_SYSTEM_LOCK(env);
+
+ __db_print_reginfo(env, &mgr->reginfo, "Transaction", flags);
+
+ __db_msg(env, "%s", DB_GLOBAL(db_line));
+ __db_msg(env, "DB_TXNMGR handle information:");
+ __mutex_print_debug_single(env, "DB_TXNMGR mutex", mgr->mutex, flags);
+ __db_dl(env,
+ "Number of transactions discarded", (u_long)mgr->n_discards);
+
+ __db_msg(env, "%s", DB_GLOBAL(db_line));
+ __db_msg(env, "DB_TXNREGION handle information:");
+ __mutex_print_debug_single(
+ env, "DB_TXNREGION region mutex", region->mtx_region, flags);
+ STAT_ULONG("Maximum number of active txns", region->maxtxns);
+ STAT_HEX("Last transaction ID allocated", region->last_txnid);
+ STAT_HEX("Current maximum unused ID", region->cur_maxid);
+
+ __mutex_print_debug_single(
+ env, "checkpoint mutex", region->mtx_ckp, flags);
+ STAT_LSN("Last checkpoint LSN", &region->last_ckp);
+ __db_msg(env,
+ "%.24s\tLast checkpoint timestamp",
+ region->time_ckp == 0 ? "0" :
+ __os_ctime(&region->time_ckp, time_buf));
+
+ __db_prflags(env, NULL, region->flags, fn, NULL, "\tFlags");
+
+ __db_msg(env, "%s", DB_GLOBAL(db_line));
+ TXN_SYSTEM_UNLOCK(env);
+
+ return (0);
+}
+
+static char *
+__txn_status(txn)
+ DB_TXN_ACTIVE *txn;
+{
+ switch (txn->status) {
+ case TXN_ABORTED:
+ return ("aborted");
+ case TXN_COMMITTED:
+ return ("committed");
+ case TXN_PREPARED:
+ return ("prepared");
+ case TXN_RUNNING:
+ return ("running");
+ default:
+ break;
+ }
+ return ("unknown state");
+}
+
+static void
+__txn_gid(env, mbp, txn)
+ ENV *env;
+ DB_MSGBUF *mbp;
+ DB_TXN_ACTIVE *txn;
+{
+ u_int32_t v, *xp;
+ u_int i;
+ int cnt;
+
+ __db_msgadd(env, mbp, "\n\tGID:");
+ for (cnt = 0, xp = (u_int32_t *)txn->gid, i = 0;;) {
+ memcpy(&v, xp++, sizeof(u_int32_t));
+ __db_msgadd(env, mbp, "%#lx ", (u_long)v);
+ if ((i += sizeof(u_int32_t)) >= DB_GID_SIZE)
+ break;
+ if (++cnt == 4) {
+ DB_MSGBUF_FLUSH(env, mbp);
+ __db_msgadd(env, mbp, "\t\t");
+ cnt = 0;
+ }
+ }
+}
+
+static int
+__txn_compare(a1, b1)
+ const void *a1, *b1;
+{
+ const DB_TXN_ACTIVE *a, *b;
+
+ a = a1;
+ b = b1;
+
+ if (a->txnid > b->txnid)
+ return (1);
+ if (a->txnid < b->txnid)
+ return (-1);
+ return (0);
+}
+
+#else /* !HAVE_STATISTICS */
+
+int
+__txn_stat_pp(dbenv, statp, flags)
+ DB_ENV *dbenv;
+ DB_TXN_STAT **statp;
+ u_int32_t flags;
+{
+ COMPQUIET(statp, NULL);
+ COMPQUIET(flags, 0);
+
+ return (__db_stat_not_built(dbenv->env));
+}
+
+int
+__txn_stat_print_pp(dbenv, flags)
+ DB_ENV *dbenv;
+ u_int32_t flags;
+{
+ COMPQUIET(flags, 0);
+
+ return (__db_stat_not_built(dbenv->env));
+}
+#endif