/*- * See the file LICENSE for redistribution information. * * Copyright (c) 2004-2005 * Sleepycat Software. All rights reserved. * * $Id: seq_stat.c,v 12.6 2005/10/07 20:21:39 ubell Exp $ */ #include "db_config.h" #ifdef HAVE_SEQUENCE #ifndef NO_SYSTEM_INCLUDES #include #include #include #endif #include "db_int.h" #include "dbinc/db_page.h" #include "dbinc/db_am.h" #include "dbinc_auto/sequence_ext.h" #ifdef HAVE_STATISTICS static int __seq_print_all __P((DB_SEQUENCE *, u_int32_t)); static int __seq_print_stats __P((DB_SEQUENCE *, u_int32_t)); /* * __seq_stat -- * Get statistics from the sequence. * * PUBLIC: int __seq_stat __P((DB_SEQUENCE *, DB_SEQUENCE_STAT **, u_int32_t)); */ int __seq_stat(seq, spp, flags) DB_SEQUENCE *seq; DB_SEQUENCE_STAT **spp; u_int32_t flags; { DB *dbp; DB_ENV *dbenv; DB_THREAD_INFO *ip; DB_SEQ_RECORD record; DB_SEQUENCE_STAT *sp; DBT data; int handle_check, ret, t_ret; dbp = seq->seq_dbp; dbenv = dbp->dbenv; switch (flags) { case DB_STAT_CLEAR: case DB_STAT_ALL: case 0: break; default: return (__db_ferr(dbenv, "DB_SEQUENCE->stat", 0)); } ENV_ENTER(dbenv, ip); /* Check for replication block. */ handle_check = IS_ENV_REPLICATED(dbenv); if (handle_check && (ret = __db_rep_enter(dbp, 1, 0, 0)) != 0) { handle_check = 0; goto err; } /* Allocate and clear the structure. */ if ((ret = __os_umalloc(dbenv, sizeof(*sp), &sp)) != 0) goto err; memset(sp, 0, sizeof(*sp)); if (seq->mtx_seq != MUTEX_INVALID) { __mutex_set_wait_info( dbenv, seq->mtx_seq, &sp->st_wait, &sp->st_nowait); if (LF_ISSET(DB_STAT_CLEAR)) __mutex_clear(dbenv, seq->mtx_seq); } memset(&data, 0, sizeof(data)); data.data = &record; data.ulen = sizeof(record); data.flags = DB_DBT_USERMEM; retry: if ((ret = __db_get(dbp, NULL, &seq->seq_key, &data, 0)) != 0) { if (ret == DB_BUFFER_SMALL && data.size > sizeof(seq->seq_record)) { if ((ret = __os_malloc(dbenv, data.size, &data.data)) != 0) goto err; data.ulen = data.size; goto retry; } goto err; } if (data.data != &record) memcpy(&record, data.data, sizeof(record)); sp->st_current = record.seq_value; sp->st_value = seq->seq_record.seq_value; sp->st_last_value = seq->seq_last_value; sp->st_min = seq->seq_record.seq_min; sp->st_max = seq->seq_record.seq_max; sp->st_cache_size = seq->seq_cache_size; sp->st_flags = seq->seq_record.flags; *spp = sp; if (data.data != &record) __os_free(dbenv, data.data); /* Release replication block. */ err: if (handle_check && (t_ret = __env_db_rep_exit(dbenv)) != 0 && ret == 0) ret = t_ret; ENV_LEAVE(dbenv, ip); return (ret); } /* * __seq_stat_print -- * Print statistics from the sequence. * * PUBLIC: int __seq_stat_print __P((DB_SEQUENCE *, u_int32_t)); */ int __seq_stat_print(seq, flags) DB_SEQUENCE *seq; u_int32_t flags; { DB *dbp; DB_ENV *dbenv; DB_THREAD_INFO *ip; int handle_check, ret, t_ret; dbp = seq->seq_dbp; dbenv = dbp->dbenv; ENV_ENTER(dbenv, ip); /* Check for replication block. */ handle_check = IS_ENV_REPLICATED(dbenv); if (handle_check && (ret = __db_rep_enter(dbp, 1, 0, 0)) != 0) { handle_check = 0; goto err; } if ((ret = __seq_print_stats(seq, flags)) != 0) goto err; if (LF_ISSET(DB_STAT_ALL) && (ret = __seq_print_all(seq, flags)) != 0) goto err; /* Release replication block. */ err: if (handle_check && (t_ret = __env_db_rep_exit(dbenv)) != 0 && ret == 0) ret = t_ret; ENV_LEAVE(dbenv, ip); return (ret); } static const FN __db_seq_flags_fn[] = { { DB_SEQ_DEC, "decrement" }, { DB_SEQ_INC, "increment" }, { DB_SEQ_RANGE_SET, "range set (internal)" }, { DB_SEQ_WRAP, "wraparound at end" }, { 0, NULL } }; /* * __db_get_seq_flags_fn -- * Return the __db_seq_flags_fn array. * * PUBLIC: const FN * __db_get_seq_flags_fn __P((void)); */ const FN * __db_get_seq_flags_fn() { return (__db_seq_flags_fn); } /* * __seq_print_stats -- * Display sequence stat structure. */ static int __seq_print_stats(seq, flags) DB_SEQUENCE *seq; u_int32_t flags; { DB_ENV *dbenv; DB_SEQUENCE_STAT *sp; int ret; dbenv = seq->seq_dbp->dbenv; if ((ret = __seq_stat(seq, &sp, flags)) != 0) return (ret); __db_dl_pct(dbenv, "The number of sequence locks that required waiting", (u_long)sp->st_wait, DB_PCT(sp->st_wait, sp->st_wait + sp->st_nowait), NULL); STAT_FMT("The current sequence value", INT64_FMT, int64_t, sp->st_current); STAT_FMT("The cached sequence value", INT64_FMT, int64_t, sp->st_value); STAT_FMT("The last cached sequence value", INT64_FMT, int64_t, sp->st_last_value); STAT_FMT("The minimum sequence value", INT64_FMT, int64_t, sp->st_value); STAT_FMT("The maximum sequence value", INT64_FMT, int64_t, sp->st_value); STAT_ULONG("The cache size", sp->st_cache_size); __db_prflags(dbenv, NULL, sp->st_flags, __db_seq_flags_fn, NULL, "\tSequence flags"); __os_ufree(seq->seq_dbp->dbenv, sp); return (0); } /* * __seq_print_all -- * Display sequence debugging information - none for now. * (The name seems a bit strange, no?) */ static int __seq_print_all(seq, flags) DB_SEQUENCE *seq; u_int32_t flags; { COMPQUIET(seq, NULL); COMPQUIET(flags, 0); return (0); } #else /* !HAVE_STATISTICS */ int __seq_stat(seq, statp, flags) DB_SEQUENCE *seq; DB_SEQUENCE_STAT **statp; u_int32_t flags; { COMPQUIET(statp, NULL); COMPQUIET(flags, 0); return (__db_stat_not_built(seq->seq_dbp->dbenv)); } int __seq_stat_print(seq, flags) DB_SEQUENCE *seq; u_int32_t flags; { COMPQUIET(flags, 0); return (__db_stat_not_built(seq->seq_dbp->dbenv)); } /* * __db_get_seq_flags_fn -- * Return the __db_seq_flags_fn array. * * PUBLIC: const FN * __db_get_seq_flags_fn __P((void)); */ const FN * __db_get_seq_flags_fn() { static const FN __db_seq_flags_fn[] = { { 0, NULL } }; /* * !!! * The Tcl API uses this interface, stub it off. */ return (__db_seq_flags_fn); } #endif /* !HAVE_STATISTICS */ #endif /* HAVE_SEQUENCE */