libdb/lang/tcl/tcl_env.c

3252 lines
79 KiB
C

/*-
* See the file LICENSE for redistribution information.
*
* Copyright (c) 1999, 2011 Oracle and/or its affiliates. All rights reserved.
*
* $Id$
*/
#include "db_config.h"
#include "db_int.h"
#ifdef HAVE_SYSTEM_INCLUDE_FILES
#include <tcl.h>
#endif
#include "dbinc/lock.h"
#include "dbinc/txn.h"
#include "dbinc/tcl_db.h"
/*
* Prototypes for procedures defined later in this file:
*/
static void _EnvInfoDelete __P((Tcl_Interp *, DBTCL_INFO *));
static int env_DbRemove __P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
static int env_DbRename __P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
static int env_EventInfo __P((Tcl_Interp *,
int, Tcl_Obj * CONST*, DB_ENV *, DBTCL_INFO *));
static int env_GetFlags __P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
static int env_GetOpenFlag
__P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
static int env_GetLockDetect
__P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
static int env_GetTimeout __P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
static int env_GetVerbose __P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
/*
* PUBLIC: int env_Cmd __P((ClientData, Tcl_Interp *, int, Tcl_Obj * CONST*));
*
* env_Cmd --
* Implements the "env" command.
*/
int
env_Cmd(clientData, interp, objc, objv)
ClientData clientData; /* Env handle */
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
{
static const char *envcmds[] = {
#ifdef CONFIG_TEST
"attributes",
"errfile",
"errpfx",
"event_info",
"failchk",
"id_reset",
"lock_detect",
"lock_id",
"lock_id_free",
"lock_id_set",
"lock_get",
"lock_set_priority",
"lock_stat",
"lock_stat_print",
"lock_timeout",
"lock_vec",
"log_archive",
"log_compare",
"log_config",
"log_cursor",
"log_file",
"log_flush",
"log_get",
"log_get_config",
"log_put",
"log_stat",
"log_stat_print",
"lsn_reset",
"mpool",
"mpool_stat",
"mpool_stat_print",
"mpool_sync",
"mpool_trickle",
"msgfile",
"mutex",
"mutex_free",
"mutex_get_align",
"mutex_get_incr",
"mutex_get_max",
"mutex_get_tas_spins",
"mutex_lock",
"mutex_set_tas_spins",
"mutex_stat",
"mutex_stat_print",
"mutex_unlock",
"rep_config",
"rep_elect",
"rep_flush",
"rep_get_clockskew",
"rep_get_config",
"rep_get_limit",
"rep_get_nsites",
"rep_get_priority",
"rep_get_request",
"rep_get_timeout",
"rep_lease",
"rep_limit",
"rep_nsites",
"rep_process_message",
"rep_request",
"rep_start",
"rep_stat",
"rep_stat_print",
"rep_sync",
"rep_transport",
"repmgr",
"repmgr_get_ack_policy",
"repmgr_get_local_site",
"repmgr_site_list",
"repmgr_stat",
"repmgr_stat_print",
"set_flags",
"stat_print",
"test",
"txn_applied",
"txn_id_set",
"txn_recover",
"txn_stat",
"txn_stat_print",
"txn_timeout",
"verbose",
#endif
"cdsgroup",
"close",
"dbremove",
"dbrename",
"get_cachesize",
"get_cache_max",
"get_create_dir",
"get_data_dirs",
"get_encrypt_flags",
"get_errpfx",
"get_flags",
"get_home",
"get_lg_bsize",
"get_lg_dir",
"get_lg_filemode",
"get_lg_max",
"get_lg_regionmax",
"get_lk_detect",
"get_lk_init_lockers",
"get_lk_init_locks",
"get_lk_init_objects",
"get_lk_max_lockers",
"get_lk_max_locks",
"get_lk_max_objects",
"get_lk_partitions",
"get_mp_max_openfd",
"get_mp_max_write",
"get_mp_mmapsize",
"get_mp_mtxcount",
"get_mp_pagesize",
"get_open_flags",
"get_shm_key",
"get_tas_spins",
"get_timeout",
"get_tmp_dir",
"get_tx_init",
"get_tx_max",
"get_tx_timestamp",
"get_verbose",
"resize_cache",
"set_data_dir",
"set_maxcache",
"txn",
"txn_checkpoint",
NULL
};
enum envcmds {
#ifdef CONFIG_TEST
ENVATTR,
ENVERRFILE,
ENVERRPFX,
ENVEVENTINFO,
ENVFAILCHK,
ENVIDRESET,
ENVLKDETECT,
ENVLKID,
ENVLKFREEID,
ENVLKSETID,
ENVLKGET,
ENVLKPRI,
ENVLKSTAT,
ENVLKSTATPRT,
ENVLKTIMEOUT,
ENVLKVEC,
ENVLOGARCH,
ENVLOGCMP,
ENVLOGCONFIG,
ENVLOGCURSOR,
ENVLOGFILE,
ENVLOGFLUSH,
ENVLOGGET,
ENVLOGGETCONFIG,
ENVLOGPUT,
ENVLOGSTAT,
ENVLOGSTATPRT,
ENVLSNRESET,
ENVMP,
ENVMPSTAT,
ENVMPSTATPRT,
ENVMPSYNC,
ENVTRICKLE,
ENVMSGFILE,
ENVMUTEX,
ENVMUTFREE,
ENVMUTGETALIGN,
ENVMUTGETINCR,
ENVMUTGETMAX,
ENVMUTGETTASSPINS,
ENVMUTLOCK,
ENVMUTSETTASSPINS,
ENVMUTSTAT,
ENVMUTSTATPRT,
ENVMUTUNLOCK,
ENVREPCONFIG,
ENVREPELECT,
ENVREPFLUSH,
ENVREPGETCLOCKSKEW,
ENVREPGETCONFIG,
ENVREPGETLIMIT,
ENVREPGETNSITES,
ENVREPGETPRIORITY,
ENVREPGETREQUEST,
ENVREPGETTIMEOUT,
ENVREPLEASE,
ENVREPLIMIT,
ENVREPNSITES,
ENVREPPROCMESS,
ENVREPREQUEST,
ENVREPSTART,
ENVREPSTAT,
ENVREPSTATPRT,
ENVREPSYNC,
ENVREPTRANSPORT,
ENVREPMGR,
ENVREPMGRGETACK,
ENVREPMGRGETLOCAL,
ENVREPMGRSITELIST,
ENVREPMGRSTAT,
ENVREPMGRSTATPRT,
ENVSETFLAGS,
ENVSTATPRT,
ENVTEST,
ENVTXNAPPLIED,
ENVTXNSETID,
ENVTXNRECOVER,
ENVTXNSTAT,
ENVTXNSTATPRT,
ENVTXNTIMEOUT,
ENVVERB,
#endif
ENVCDSGROUP,
ENVCLOSE,
ENVDBREMOVE,
ENVDBRENAME,
ENVGETCACHESIZE,
ENVGETCACHEMAX,
ENVGETCREATEDIR,
ENVGETDATADIRS,
ENVGETENCRYPTFLAGS,
ENVGETERRPFX,
ENVGETFLAGS,
ENVGETHOME,
ENVGETLGBSIZE,
ENVGETLGDIR,
ENVGETLGFILEMODE,
ENVGETLGMAX,
ENVGETLGREGIONMAX,
ENVGETLKDETECT,
ENVGETLKINITLOCKERS,
ENVGETLKINITLOCKS,
ENVGETLKINITOBJECTS,
ENVGETLKMAXLOCKERS,
ENVGETLKMAXLOCKS,
ENVGETLKMAXOBJECTS,
ENVGETLKPARTITIONS,
ENVGETMPMAXOPENFD,
ENVGETMPMAXWRITE,
ENVGETMPMMAPSIZE,
ENVGETMPMTXCOUNT,
ENVGETMPPAGESIZE,
ENVGETOPENFLAG,
ENVGETSHMKEY,
ENVGETTASSPINS,
ENVGETTIMEOUT,
ENVGETTMPDIR,
ENVGETTXINIT,
ENVGETTXMAX,
ENVGETTXTIMESTAMP,
ENVGETVERBOSE,
ENVRESIZECACHE,
ENVSETDATADIR,
ENVSETMAXCACHE,
ENVTXN,
ENVTXNCKP
};
DBTCL_INFO *envip;
DB_ENV *dbenv;
Tcl_Obj **listobjv, *myobjv[3], *res;
db_timeout_t timeout;
size_t size;
time_t timeval;
u_int32_t bytes, gbytes, value;
long shm_key;
int cmdindex, i, intvalue, listobjc, ncache, result, ret, t_ret;
const char *strval, **dirs;
char *strarg, newname[MSG_SIZE];
#ifdef CONFIG_TEST
DBTCL_INFO *logcip;
DB_LOGC *logc;
u_int32_t lockid;
long newval, otherval;
#endif
Tcl_ResetResult(interp);
dbenv = (DB_ENV *)clientData;
envip = _PtrToInfo((void *)dbenv);
result = TCL_OK;
memset(newname, 0, MSG_SIZE);
if (objc <= 1) {
Tcl_WrongNumArgs(interp, 1, objv, "command cmdargs");
return (TCL_ERROR);
}
if (dbenv == NULL) {
Tcl_SetResult(interp, "NULL env pointer", TCL_STATIC);
return (TCL_ERROR);
}
if (envip == NULL) {
Tcl_SetResult(interp, "NULL env info pointer", TCL_STATIC);
return (TCL_ERROR);
}
/*
* Get the command name index from the object based on the berkdbcmds
* defined above.
*/
if (Tcl_GetIndexFromObj(interp, objv[1], envcmds, "command",
TCL_EXACT, &cmdindex) != TCL_OK)
return (IS_HELP(objv[1]));
res = NULL;
switch ((enum envcmds)cmdindex) {
#ifdef CONFIG_TEST
case ENVEVENTINFO:
result = env_EventInfo(interp, objc, objv, dbenv, envip);
break;
case ENVFAILCHK:
/*
* No args for this. Error if there are some.
*/
if (objc > 2) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
_debug_check();
ret = dbenv->failchk(dbenv, 0);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"failchk");
break;
case ENVIDRESET:
result = tcl_EnvIdReset(interp, objc, objv, dbenv);
break;
case ENVLSNRESET:
result = tcl_EnvLsnReset(interp, objc, objv, dbenv);
break;
case ENVLKDETECT:
result = tcl_LockDetect(interp, objc, objv, dbenv);
break;
case ENVLKSTAT:
result = tcl_LockStat(interp, objc, objv, dbenv);
break;
case ENVLKSTATPRT:
result = tcl_LockStatPrint(interp, objc, objv, dbenv);
break;
case ENVLKTIMEOUT:
result = tcl_LockTimeout(interp, objc, objv, dbenv);
break;
case ENVLKID:
/*
* No args for this. Error if there are some.
*/
if (objc > 2) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
_debug_check();
ret = dbenv->lock_id(dbenv, &lockid);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"lock_id");
if (result == TCL_OK)
res = Tcl_NewWideIntObj((Tcl_WideInt)lockid);
break;
case ENVLKFREEID:
if (objc != 3) {
Tcl_WrongNumArgs(interp, 3, objv, NULL);
return (TCL_ERROR);
}
result = Tcl_GetLongFromObj(interp, objv[2], &newval);
if (result != TCL_OK)
return (result);
ret = dbenv->lock_id_free(dbenv, (u_int32_t)newval);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"lock id_free");
break;
case ENVLKSETID:
if (objc != 4) {
Tcl_WrongNumArgs(interp, 4, objv, "current max");
return (TCL_ERROR);
}
result = Tcl_GetLongFromObj(interp, objv[2], &newval);
if (result != TCL_OK)
return (result);
result = Tcl_GetLongFromObj(interp, objv[3], &otherval);
if (result != TCL_OK)
return (result);
ret = __lock_id_set(dbenv->env,
(u_int32_t)newval, (u_int32_t)otherval);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"lock id_free");
break;
case ENVLKPRI:
if (objc != 4) {
Tcl_WrongNumArgs(interp, 4, objv, NULL);
return (TCL_ERROR);
}
result = _GetUInt32(interp, objv[2], &lockid);
if (result != TCL_OK)
return (result);
result = _GetUInt32(interp, objv[3], &value);
if (result != TCL_OK)
return (result);
if (dbenv->env->lk_handle == NULL) {
Tcl_SetResult(interp, "env not configured for locking", NULL);
return (TCL_ERROR);
}
ret = dbenv->set_lk_priority(dbenv, lockid, value);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"lock set priority");
break;
case ENVLKGET:
result = tcl_LockGet(interp, objc, objv, dbenv);
break;
case ENVLKVEC:
result = tcl_LockVec(interp, objc, objv, dbenv);
break;
case ENVLOGARCH:
result = tcl_LogArchive(interp, objc, objv, dbenv);
break;
case ENVLOGCMP:
result = tcl_LogCompare(interp, objc, objv);
break;
case ENVLOGCONFIG:
/*
* Two args for this. Error if different.
*/
if (objc != 4) {
Tcl_WrongNumArgs(interp, 2, objv, "opt on|off");
return (TCL_ERROR);
}
result = tcl_LogConfig(interp, dbenv, objv[2], objv[3]);
break;
case ENVLOGCURSOR:
snprintf(newname, sizeof(newname),
"%s.logc%d", envip->i_name, envip->i_envlogcid);
logcip = _NewInfo(interp, NULL, newname, I_LOGC);
if (logcip != NULL) {
ret = dbenv->log_cursor(dbenv, &logc, 0);
if (ret == 0) {
result = TCL_OK;
envip->i_envlogcid++;
/*
* We do NOT want to set i_parent to
* envip here because log cursors are
* not "tied" to the env. That is, they
* are NOT closed if the env is closed.
*/
(void)Tcl_CreateObjCommand(interp, newname,
(Tcl_ObjCmdProc *)logc_Cmd,
(ClientData)logc, NULL);
res = NewStringObj(newname, strlen(newname));
_SetInfoData(logcip, logc);
} else {
_DeleteInfo(logcip);
result = _ErrorSetup(interp, ret, "log cursor");
}
} else {
Tcl_SetResult(interp,
"Could not set up info", TCL_STATIC);
result = TCL_ERROR;
}
break;
case ENVLOGFILE:
result = tcl_LogFile(interp, objc, objv, dbenv);
break;
case ENVLOGFLUSH:
result = tcl_LogFlush(interp, objc, objv, dbenv);
break;
case ENVLOGGET:
result = tcl_LogGet(interp, objc, objv, dbenv);
break;
case ENVLOGGETCONFIG:
/*
* Two args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = tcl_LogGetConfig(interp, dbenv, objv[2]);
break;
case ENVLOGPUT:
result = tcl_LogPut(interp, objc, objv, dbenv);
break;
case ENVLOGSTAT:
result = tcl_LogStat(interp, objc, objv, dbenv);
break;
case ENVLOGSTATPRT:
result = tcl_LogStatPrint(interp, objc, objv, dbenv);
break;
case ENVMPSTAT:
result = tcl_MpStat(interp, objc, objv, dbenv);
break;
case ENVMPSTATPRT:
result = tcl_MpStatPrint(interp, objc, objv, dbenv);
break;
case ENVMPSYNC:
result = tcl_MpSync(interp, objc, objv, dbenv);
break;
case ENVTRICKLE:
result = tcl_MpTrickle(interp, objc, objv, dbenv);
break;
case ENVMP:
result = tcl_Mp(interp, objc, objv, dbenv, envip);
break;
case ENVMUTEX:
result = tcl_Mutex(interp, objc, objv, dbenv);
break;
case ENVMUTFREE:
result = tcl_MutFree(interp, objc, objv, dbenv);
break;
case ENVMUTGETALIGN:
result = tcl_MutGet(interp, dbenv, DBTCL_MUT_ALIGN);
break;
case ENVMUTGETINCR:
result = tcl_MutGet(interp, dbenv, DBTCL_MUT_INCR);
break;
case ENVMUTGETMAX:
result = tcl_MutGet(interp, dbenv, DBTCL_MUT_MAX);
break;
case ENVMUTGETTASSPINS:
result = tcl_MutGet(interp, dbenv, DBTCL_MUT_TAS);
break;
case ENVMUTLOCK:
result = tcl_MutLock(interp, objc, objv, dbenv);
break;
case ENVMUTSETTASSPINS:
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = tcl_MutSet(interp, objv[2], dbenv, DBTCL_MUT_TAS);
break;
case ENVMUTSTAT:
result = tcl_MutStat(interp, objc, objv, dbenv);
break;
case ENVMUTSTATPRT:
result = tcl_MutStatPrint(interp, objc, objv, dbenv);
break;
case ENVMUTUNLOCK:
result = tcl_MutUnlock(interp, objc, objv, dbenv);
break;
case ENVREPCONFIG:
/*
* Two args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = tcl_RepConfig(interp, dbenv, objv[2]);
break;
case ENVREPELECT:
result = tcl_RepElect(interp, objc, objv, dbenv);
break;
case ENVREPFLUSH:
result = tcl_RepFlush(interp, objc, objv, dbenv);
break;
case ENVREPGETCLOCKSKEW:
result = tcl_RepGetTwo(interp, dbenv, DBTCL_GETCLOCK);
break;
case ENVREPGETCONFIG:
/*
* Two args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = tcl_RepGetConfig(interp, dbenv, objv[2]);
break;
case ENVREPGETLIMIT:
result = tcl_RepGetTwo(interp, dbenv, DBTCL_GETLIMIT);
break;
case ENVREPGETNSITES:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->rep_get_nsites(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env rep_get_nsites")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVREPGETPRIORITY:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->rep_get_priority(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env rep_get_priority")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVREPGETREQUEST:
result = tcl_RepGetTwo(interp, dbenv, DBTCL_GETREQ);
break;
case ENVREPGETTIMEOUT:
/*
* Two args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = tcl_RepGetTimeout(interp, dbenv, objv[2]);
break;
case ENVREPLEASE:
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = Tcl_ListObjGetElements(interp, objv[2],
&listobjc, &listobjv);
if (result == TCL_OK)
result = tcl_RepLease(interp,
listobjc, listobjv, dbenv);
break;
case ENVREPLIMIT:
result = tcl_RepLimit(interp, objc, objv, dbenv);
break;
case ENVREPNSITES:
result = tcl_RepNSites(interp, objc, objv, dbenv);
break;
case ENVREPPROCMESS:
result = tcl_RepProcessMessage(interp, objc, objv, dbenv);
break;
case ENVREPREQUEST:
result = tcl_RepRequest(interp, objc, objv, dbenv);
break;
case ENVREPSTART:
result = tcl_RepStart(interp, objc, objv, dbenv);
break;
case ENVREPSTAT:
result = tcl_RepStat(interp, objc, objv, dbenv);
break;
case ENVREPSTATPRT:
result = tcl_RepStatPrint(interp, objc, objv, dbenv);
break;
case ENVREPSYNC:
result = tcl_RepSync(interp, objc, objv, dbenv);
break;
case ENVREPTRANSPORT:
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = Tcl_ListObjGetElements(interp, objv[2],
&listobjc, &listobjv);
if (result == TCL_OK)
result = tcl_RepTransport(interp,
listobjc, listobjv, dbenv, envip);
break;
case ENVREPMGR:
result = tcl_RepMgr(interp, objc, objv, dbenv);
break;
case ENVREPMGRGETACK:
result = tcl_RepGetAckPolicy(interp, objc, objv, dbenv);
break;
case ENVREPMGRGETLOCAL:
result = tcl_RepGetLocalSite(interp, objc, objv, dbenv);
break;
case ENVREPMGRSITELIST:
result = tcl_RepMgrSiteList(interp, objc, objv, dbenv);
break;
case ENVREPMGRSTAT:
result = tcl_RepMgrStat(interp, objc, objv, dbenv);
break;
case ENVREPMGRSTATPRT:
result = tcl_RepMgrStatPrint(interp, objc, objv, dbenv);
break;
case ENVTXNAPPLIED:
result = tcl_RepApplied(interp, objc, objv, dbenv);
break;
case ENVTXNSETID:
if (objc != 4) {
Tcl_WrongNumArgs(interp, 4, objv, "current max");
return (TCL_ERROR);
}
result = Tcl_GetLongFromObj(interp, objv[2], &newval);
if (result != TCL_OK)
return (result);
result = Tcl_GetLongFromObj(interp, objv[3], &otherval);
if (result != TCL_OK)
return (result);
ret = __txn_id_set(dbenv->env,
(u_int32_t)newval, (u_int32_t)otherval);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"txn setid");
break;
case ENVTXNRECOVER:
result = tcl_TxnRecover(interp, objc, objv, dbenv, envip);
break;
case ENVTXNSTAT:
result = tcl_TxnStat(interp, objc, objv, dbenv);
break;
case ENVTXNSTATPRT:
result = tcl_TxnStatPrint(interp, objc, objv, dbenv);
break;
case ENVTXNTIMEOUT:
result = tcl_TxnTimeout(interp, objc, objv, dbenv);
break;
case ENVATTR:
result = tcl_EnvAttr(interp, objc, objv, dbenv);
break;
case ENVERRFILE:
/*
* One args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, "errfile");
return (TCL_ERROR);
}
strarg = Tcl_GetStringFromObj(objv[2], NULL);
tcl_EnvSetErrfile(interp, dbenv, envip, strarg);
result = TCL_OK;
break;
case ENVERRPFX:
/*
* One args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, "pfx");
return (TCL_ERROR);
}
strarg = Tcl_GetStringFromObj(objv[2], NULL);
result = tcl_EnvSetErrpfx(interp, dbenv, envip, strarg);
break;
case ENVMSGFILE:
/*
* One args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, "msgfile");
return (TCL_ERROR);
}
strarg = Tcl_GetStringFromObj(objv[2], NULL);
tcl_EnvSetMsgfile(interp, dbenv, envip, strarg);
result = TCL_OK;
break;
case ENVSETFLAGS:
/*
* Two args for this. Error if different.
*/
if (objc != 4) {
Tcl_WrongNumArgs(interp, 2, objv, "which on|off");
return (TCL_ERROR);
}
result = tcl_EnvSetFlags(interp, dbenv, objv[2], objv[3]);
break;
case ENVSTATPRT:
result = tcl_EnvStatPrint(interp, objc, objv, dbenv);
break;
case ENVTEST:
result = tcl_EnvTest(interp, objc, objv, dbenv);
break;
case ENVVERB:
/*
* Two args for this. Error if different.
*/
if (objc != 4) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
result = tcl_EnvVerbose(interp, dbenv, objv[2], objv[3]);
break;
#endif
case ENVCDSGROUP:
result = tcl_CDSGroup(interp, objc, objv, dbenv, envip);
break;
case ENVCLOSE:
/*
* No args for this. Error if there are some.
*/
if (objc > 2) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
/*
* Any transactions will be aborted, and an mpools
* closed automatically. We must delete any txn
* and mp widgets we have here too for this env.
* NOTE: envip is freed when we come back from
* this function. Set it to NULL to make sure no
* one tries to use it later.
*/
ret = __mutex_free(dbenv->env, &envip->i_mutex);
_debug_check();
if ((t_ret = dbenv->close(dbenv, 0)) != 0 && ret == 0)
ret = t_ret;
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env close");
_EnvInfoDelete(interp, envip);
envip = NULL;
break;
case ENVDBREMOVE:
result = env_DbRemove(interp, objc, objv, dbenv);
break;
case ENVDBRENAME:
result = env_DbRename(interp, objc, objv, dbenv);
break;
case ENVGETCACHESIZE:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_cachesize(dbenv, &gbytes, &bytes, &ncache);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_cachesize")) == TCL_OK) {
myobjv[0] = Tcl_NewLongObj((long)gbytes);
myobjv[1] = Tcl_NewLongObj((long)bytes);
myobjv[2] = Tcl_NewLongObj((long)ncache);
res = Tcl_NewListObj(3, myobjv);
}
break;
case ENVGETCACHEMAX:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_cache_max(dbenv, &gbytes, &bytes);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_cache_max")) == TCL_OK) {
myobjv[0] = Tcl_NewLongObj((long)gbytes);
myobjv[1] = Tcl_NewLongObj((long)bytes);
res = Tcl_NewListObj(2, myobjv);
}
break;
case ENVGETCREATEDIR:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_create_dir(dbenv, &strval);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_create_dir")) == TCL_OK)
res = NewStringObj(strval, strlen(strval));
break;
case ENVGETDATADIRS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_data_dirs(dbenv, &dirs);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_data_dirs")) == TCL_OK) {
res = Tcl_NewListObj(0, NULL);
for (i = 0; result == TCL_OK && dirs[i] != NULL; i++)
result = Tcl_ListObjAppendElement(interp, res,
NewStringObj(dirs[i], strlen(dirs[i])));
}
break;
case ENVGETENCRYPTFLAGS:
result = tcl_EnvGetEncryptFlags(interp, objc, objv, dbenv);
break;
case ENVGETERRPFX:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
dbenv->get_errpfx(dbenv, &strval);
res = NewStringObj(strval, strlen(strval));
break;
case ENVGETFLAGS:
result = env_GetFlags(interp, objc, objv, dbenv);
break;
case ENVGETHOME:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_home(dbenv, &strval);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_home")) == TCL_OK)
res = NewStringObj(strval, strlen(strval));
break;
case ENVGETLGBSIZE:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lg_bsize(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lg_bsize")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLGDIR:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lg_dir(dbenv, &strval);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lg_dir")) == TCL_OK)
res = NewStringObj(strval, strlen(strval));
break;
case ENVGETLGFILEMODE:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lg_filemode(dbenv, &intvalue);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lg_filemode")) == TCL_OK)
res = Tcl_NewLongObj((long)intvalue);
break;
case ENVGETLGMAX:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lg_max(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lg_max")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLGREGIONMAX:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lg_regionmax(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lg_regionmax")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLKDETECT:
result = env_GetLockDetect(interp, objc, objv, dbenv);
break;
case ENVGETLKINITLOCKERS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_memory_init(dbenv, DB_MEM_LOCKER, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_init_lockers")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLKINITLOCKS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_memory_init(dbenv, DB_MEM_LOCK, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_init_locks")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLKINITOBJECTS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_memory_init(dbenv, DB_MEM_LOCKOBJECT, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_init_objects")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLKMAXLOCKERS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lk_max_lockers(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_max_lockers")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLKMAXLOCKS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lk_max_locks(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_max_locks")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLKMAXOBJECTS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lk_max_objects(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_max_objects")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETLKPARTITIONS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lk_partitions(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_partitions")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETMPMAXOPENFD:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_mp_max_openfd(dbenv, &intvalue);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_mp_max_openfd")) == TCL_OK)
res = Tcl_NewIntObj(intvalue);
break;
case ENVGETMPMAXWRITE:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_mp_max_write(dbenv, &intvalue, &timeout);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_mp_max_write")) == TCL_OK) {
myobjv[0] = Tcl_NewIntObj(intvalue);
myobjv[1] = Tcl_NewIntObj((int)timeout);
res = Tcl_NewListObj(2, myobjv);
}
break;
case ENVGETMPMMAPSIZE:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_mp_mmapsize(dbenv, &size);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_mp_mmapsize")) == TCL_OK)
res = Tcl_NewLongObj((long)size);
break;
case ENVGETMPMTXCOUNT:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_mp_mtxcount(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_mp_mtxcount")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETMPPAGESIZE:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_mp_pagesize(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_mp_pagesize")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETOPENFLAG:
result = env_GetOpenFlag(interp, objc, objv, dbenv);
break;
case ENVGETSHMKEY:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_shm_key(dbenv, &shm_key);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env shm_key")) == TCL_OK)
res = Tcl_NewLongObj(shm_key);
break;
case ENVGETTASSPINS:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->mutex_get_tas_spins(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_tas_spins")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETTIMEOUT:
result = env_GetTimeout(interp, objc, objv, dbenv);
break;
case ENVGETTMPDIR:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_tmp_dir(dbenv, &strval);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_tmp_dir")) == TCL_OK)
res = NewStringObj(strval, strlen(strval));
break;
case ENVGETTXINIT:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_memory_init(dbenv, DB_MEM_TRANSACTION, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_tx_init")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETTXMAX:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_tx_max(dbenv, &value);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_tx_max")) == TCL_OK)
res = Tcl_NewLongObj((long)value);
break;
case ENVGETTXTIMESTAMP:
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_tx_timestamp(dbenv, &timeval);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_tx_timestamp")) == TCL_OK)
res = Tcl_NewLongObj((long)timeval);
break;
case ENVGETVERBOSE:
result = env_GetVerbose(interp, objc, objv, dbenv);
break;
case ENVRESIZECACHE:
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv,
"?-resize_cache {gbytes bytes}?");
result = TCL_ERROR;
break;
}
if ((result = Tcl_ListObjGetElements(
interp, objv[2], &listobjc, &listobjv)) != TCL_OK)
break;
if (listobjc != 2) {
Tcl_WrongNumArgs(interp, 2, objv,
"?-resize_cache {gbytes bytes}?");
result = TCL_ERROR;
break;
}
result = _GetUInt32(interp, listobjv[0], &gbytes);
if (result != TCL_OK)
break;
result = _GetUInt32(interp, listobjv[1], &bytes);
if (result != TCL_OK)
break;
ret = dbenv->set_cachesize(dbenv, gbytes, bytes, 0);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"resize_cache");
break;
case ENVSETDATADIR:
/*
* One args for this. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, "pfx");
return (TCL_ERROR);
}
strarg = Tcl_GetStringFromObj(objv[2], NULL);
ret = dbenv->set_data_dir(dbenv, strarg);
return (_ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env set data dir"));
case ENVSETMAXCACHE:
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv,
"?-set_max_cache {gbytes bytes}?");
result = TCL_ERROR;
break;
}
if ((result = Tcl_ListObjGetElements(
interp, objv[2], &listobjc, &listobjv)) != TCL_OK)
break;
if (listobjc != 2) {
Tcl_WrongNumArgs(interp, 2, objv,
"?-set_max_cache {gbytes bytes}?");
result = TCL_ERROR;
break;
}
result = _GetUInt32(interp, listobjv[0], &gbytes);
if (result != TCL_OK)
break;
result = _GetUInt32(interp, listobjv[1], &bytes);
if (result != TCL_OK)
break;
ret = dbenv->set_cache_max(dbenv, gbytes, bytes);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"set_max_cache");
break;
case ENVTXN:
result = tcl_Txn(interp, objc, objv, dbenv, envip);
break;
case ENVTXNCKP:
result = tcl_TxnCheckpoint(interp, objc, objv, dbenv);
break;
}
/*
* Only set result if we have a res. Otherwise, lower
* functions have already done so.
*/
if (result == TCL_OK && res)
Tcl_SetObjResult(interp, res);
return (result);
}
/*
* PUBLIC: int tcl_EnvRemove __P((Tcl_Interp *, int, Tcl_Obj * CONST*));
*
* tcl_EnvRemove --
*/
int
tcl_EnvRemove(interp, objc, objv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
{
static const char *envremopts[] = {
#ifdef CONFIG_TEST
"-overwrite",
#endif
"-data_dir",
"-encryptaes",
"-encryptany",
"-force",
"-home",
"-log_dir",
"-tmp_dir",
"-use_environ",
"-use_environ_root",
NULL
};
enum envremopts {
#ifdef CONFIG_TEST
ENVREM_OVERWRITE,
#endif
ENVREM_DATADIR,
ENVREM_ENCRYPT_AES,
ENVREM_ENCRYPT_ANY,
ENVREM_FORCE,
ENVREM_HOME,
ENVREM_LOGDIR,
ENVREM_TMPDIR,
ENVREM_USE_ENVIRON,
ENVREM_USE_ENVIRON_ROOT
};
DB_ENV *dbenv;
u_int32_t cflag, enc_flag, flag, forceflag, sflag;
int i, optindex, result, ret;
char *datadir, *home, *logdir, *passwd, *tmpdir;
result = TCL_OK;
cflag = flag = forceflag = sflag = 0;
home = NULL;
passwd = NULL;
datadir = logdir = tmpdir = NULL;
enc_flag = 0;
if (objc < 2) {
Tcl_WrongNumArgs(interp, 2, objv, "?args?");
return (TCL_ERROR);
}
i = 2;
while (i < objc) {
if (Tcl_GetIndexFromObj(interp, objv[i], envremopts, "option",
TCL_EXACT, &optindex) != TCL_OK) {
result = IS_HELP(objv[i]);
goto error;
}
i++;
switch ((enum envremopts)optindex) {
case ENVREM_ENCRYPT_AES:
/* Make sure we have an arg to check against! */
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv,
"?-encryptaes passwd?");
result = TCL_ERROR;
break;
}
passwd = Tcl_GetStringFromObj(objv[i++], NULL);
enc_flag = DB_ENCRYPT_AES;
break;
case ENVREM_ENCRYPT_ANY:
/* Make sure we have an arg to check against! */
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv,
"?-encryptany passwd?");
result = TCL_ERROR;
break;
}
passwd = Tcl_GetStringFromObj(objv[i++], NULL);
enc_flag = 0;
break;
case ENVREM_FORCE:
forceflag |= DB_FORCE;
break;
case ENVREM_HOME:
/* Make sure we have an arg to check against! */
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv,
"?-home dir?");
result = TCL_ERROR;
break;
}
home = Tcl_GetStringFromObj(objv[i++], NULL);
break;
#ifdef CONFIG_TEST
case ENVREM_OVERWRITE:
sflag |= DB_OVERWRITE;
break;
#endif
case ENVREM_USE_ENVIRON:
flag |= DB_USE_ENVIRON;
break;
case ENVREM_USE_ENVIRON_ROOT:
flag |= DB_USE_ENVIRON_ROOT;
break;
case ENVREM_DATADIR:
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv,
"-data_dir dir");
result = TCL_ERROR;
break;
}
datadir = Tcl_GetStringFromObj(objv[i++], NULL);
break;
case ENVREM_LOGDIR:
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv,
"-log_dir dir");
result = TCL_ERROR;
break;
}
logdir = Tcl_GetStringFromObj(objv[i++], NULL);
break;
case ENVREM_TMPDIR:
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv,
"-tmp_dir dir");
result = TCL_ERROR;
break;
}
tmpdir = Tcl_GetStringFromObj(objv[i++], NULL);
break;
}
/*
* If, at any time, parsing the args we get an error,
* bail out and return.
*/
if (result != TCL_OK)
goto error;
}
if ((ret = db_env_create(&dbenv, cflag)) != 0) {
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"db_env_create");
goto error;
}
if (datadir != NULL) {
_debug_check();
ret = dbenv->set_data_dir(dbenv, datadir);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"set_data_dir");
if (result != TCL_OK)
goto error;
}
if (logdir != NULL) {
_debug_check();
ret = dbenv->set_lg_dir(dbenv, logdir);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"set_log_dir");
if (result != TCL_OK)
goto error;
}
if (tmpdir != NULL) {
_debug_check();
ret = dbenv->set_tmp_dir(dbenv, tmpdir);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"set_tmp_dir");
if (result != TCL_OK)
goto error;
}
if (passwd != NULL) {
ret = dbenv->set_encrypt(dbenv, passwd, enc_flag);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"set_encrypt");
}
if (sflag != 0 &&
(ret = dbenv->set_flags(dbenv, sflag, 1)) != 0) {
_debug_check();
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"set_flags");
if (result != TCL_OK)
goto error;
}
dbenv->set_errpfx(dbenv, "EnvRemove");
dbenv->set_errcall(dbenv, _ErrorFunc);
flag |= forceflag;
/*
* When we get here we have parsed all the args. Now remove
* the environment.
*/
_debug_check();
ret = dbenv->remove(dbenv, home, flag);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret), "env remove");
error:
return (result);
}
static void
_EnvInfoDelete(interp, envip)
Tcl_Interp *interp; /* Tcl Interpreter */
DBTCL_INFO *envip; /* Info for env */
{
DBTCL_INFO *nextp, *p;
/*
* Before we can delete the environment info, we must close
* any open subsystems in this env. We will:
* 1. Abort any transactions (which aborts any nested txns).
* 2. Close any mpools (which will put any pages itself).
* 3. Put any locks and close log cursors.
* 4. Close the error file.
*/
for (p = LIST_FIRST(&__db_infohead); p != NULL; p = nextp) {
/*
* Check if this info structure "belongs" to this
* env. If so, remove its commands and info structure.
* We do not close/abort/whatever here, because we
* don't want to replicate DB behavior.
*
* NOTE: Only those types that can nest need to be
* itemized in the switch below. That is txns and mps.
* Other types like log cursors and locks will just
* get cleaned up here.
*/
if (p->i_parent == envip) {
switch (p->i_type) {
case I_TXN:
_TxnInfoDelete(interp, p);
break;
case I_MP:
_MpInfoDelete(interp, p);
break;
case I_AUX:
case I_DB:
case I_DBC:
case I_ENV:
case I_LOCK:
case I_LOGC:
case I_NDBM:
case I_PG:
case I_SEQ:
Tcl_SetResult(interp,
"_EnvInfoDelete: bad info type",
TCL_STATIC);
break;
}
nextp = LIST_NEXT(p, entries);
(void)Tcl_DeleteCommand(interp, p->i_name);
_DeleteInfo(p);
} else
nextp = LIST_NEXT(p, entries);
}
(void)Tcl_DeleteCommand(interp, envip->i_name);
_DeleteInfo(envip);
}
#ifdef CONFIG_TEST
/*
* PUBLIC: int tcl_EnvIdReset __P((Tcl_Interp *, int, Tcl_Obj * CONST*,
* PUBLIC: DB_ENV *));
*
* tcl_EnvIdReset --
* Implements the ENV->fileid_reset command.
*/
int
tcl_EnvIdReset(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* arg count */
Tcl_Obj * CONST* objv; /* args */
DB_ENV *dbenv; /* Database pointer */
{
static const char *idwhich[] = {
"-encrypt",
NULL
};
enum idwhich {
IDENCRYPT
};
int enc, i, result, ret;
u_int32_t flags;
char *file;
result = TCL_OK;
flags = 0;
i = 2;
Tcl_SetResult(interp, "0", TCL_STATIC);
if (objc < 3) {
Tcl_WrongNumArgs(interp, 2, objv, "?-encrypt? filename");
return (TCL_ERROR);
} else if (objc > 3) {
/*
* If there is an arg, make sure it is the right one.
*/
if (Tcl_GetIndexFromObj(interp, objv[2], idwhich, "option",
TCL_EXACT, &enc) != TCL_OK)
return (IS_HELP(objv[2]));
switch ((enum idwhich)enc) {
case IDENCRYPT:
flags |= DB_ENCRYPT;
break;
}
i = 3;
}
file = Tcl_GetStringFromObj(objv[i], NULL);
ret = dbenv->fileid_reset(dbenv, file, flags);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret), "fileid reset");
return (result);
}
/*
* PUBLIC: int tcl_EnvLsnReset __P((Tcl_Interp *, int, Tcl_Obj * CONST*,
* PUBLIC: DB_ENV *));
*
* tcl_EnvLsnReset --
* Implements the ENV->lsn_reset command.
*/
int
tcl_EnvLsnReset(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* arg count */
Tcl_Obj * CONST* objv; /* args */
DB_ENV *dbenv; /* Database pointer */
{
static const char *lsnwhich[] = {
"-encrypt",
NULL
};
enum lsnwhich {
IDENCRYPT
};
int enc, i, result, ret;
u_int32_t flags;
char *file;
result = TCL_OK;
flags = 0;
i = 2;
Tcl_SetResult(interp, "0", TCL_STATIC);
if (objc < 3) {
Tcl_WrongNumArgs(interp, 2, objv, "?-encrypt? filename");
return (TCL_ERROR);
} else if (objc > 3) {
/*
* If there is an arg, make sure it is the right one.
*/
if (Tcl_GetIndexFromObj(interp, objv[2], lsnwhich, "option",
TCL_EXACT, &enc) != TCL_OK)
return (IS_HELP(objv[2]));
switch ((enum lsnwhich)enc) {
case IDENCRYPT:
flags |= DB_ENCRYPT;
break;
}
i = 3;
}
file = Tcl_GetStringFromObj(objv[i], NULL);
ret = dbenv->lsn_reset(dbenv, file, flags);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret), "lsn reset");
return (result);
}
/*
* PUBLIC: int tcl_EnvVerbose __P((Tcl_Interp *, DB_ENV *, Tcl_Obj *,
* PUBLIC: Tcl_Obj *));
*
* tcl_EnvVerbose --
*/
int
tcl_EnvVerbose(interp, dbenv, which, onoff)
Tcl_Interp *interp; /* Interpreter */
DB_ENV *dbenv; /* Env pointer */
Tcl_Obj *which; /* Which subsystem */
Tcl_Obj *onoff; /* On or off */
{
static const char *verbwhich[] = {
"deadlock",
"fileops",
"fileops_all",
"recovery",
"register",
"rep",
"rep_elect",
"rep_lease",
"rep_misc",
"rep_msgs",
"rep_sync",
"rep_system",
"rep_test",
"repmgr_connfail",
"repmgr_misc",
"wait",
NULL
};
enum verbwhich {
ENVVERB_DEADLOCK,
ENVVERB_FILEOPS,
ENVVERB_FILEOPS_ALL,
ENVVERB_RECOVERY,
ENVVERB_REGISTER,
ENVVERB_REPLICATION,
ENVVERB_REP_ELECT,
ENVVERB_REP_LEASE,
ENVVERB_REP_MISC,
ENVVERB_REP_MSGS,
ENVVERB_REP_SYNC,
ENVVERB_REP_SYSTEM,
ENVVERB_REP_TEST,
ENVVERB_REPMGR_CONNFAIL,
ENVVERB_REPMGR_MISC,
ENVVERB_WAITSFOR
};
static const char *verbonoff[] = {
"off",
"on",
NULL
};
enum verbonoff {
ENVVERB_OFF,
ENVVERB_ON
};
int on, optindex, ret;
u_int32_t wh;
if (Tcl_GetIndexFromObj(interp, which, verbwhich, "option",
TCL_EXACT, &optindex) != TCL_OK)
return (IS_HELP(which));
switch ((enum verbwhich)optindex) {
case ENVVERB_DEADLOCK:
wh = DB_VERB_DEADLOCK;
break;
case ENVVERB_FILEOPS:
wh = DB_VERB_FILEOPS;
break;
case ENVVERB_FILEOPS_ALL:
wh = DB_VERB_FILEOPS_ALL;
break;
case ENVVERB_RECOVERY:
wh = DB_VERB_RECOVERY;
break;
case ENVVERB_REGISTER:
wh = DB_VERB_REGISTER;
break;
case ENVVERB_REPLICATION:
wh = DB_VERB_REPLICATION;
break;
case ENVVERB_REP_ELECT:
wh = DB_VERB_REP_ELECT;
break;
case ENVVERB_REP_LEASE:
wh = DB_VERB_REP_LEASE;
break;
case ENVVERB_REP_MISC:
wh = DB_VERB_REP_MISC;
break;
case ENVVERB_REP_MSGS:
wh = DB_VERB_REP_MSGS;
break;
case ENVVERB_REP_SYNC:
wh = DB_VERB_REP_SYNC;
break;
case ENVVERB_REP_SYSTEM:
wh = DB_VERB_REP_SYSTEM;
break;
case ENVVERB_REP_TEST:
wh = DB_VERB_REP_TEST;
break;
case ENVVERB_REPMGR_CONNFAIL:
wh = DB_VERB_REPMGR_CONNFAIL;
break;
case ENVVERB_REPMGR_MISC:
wh = DB_VERB_REPMGR_MISC;
break;
case ENVVERB_WAITSFOR:
wh = DB_VERB_WAITSFOR;
break;
default:
return (TCL_ERROR);
}
if (Tcl_GetIndexFromObj(interp, onoff, verbonoff, "option",
TCL_EXACT, &optindex) != TCL_OK)
return (IS_HELP(onoff));
switch ((enum verbonoff)optindex) {
case ENVVERB_OFF:
on = 0;
break;
case ENVVERB_ON:
on = 1;
break;
default:
return (TCL_ERROR);
}
ret = dbenv->set_verbose(dbenv, wh, on);
return (_ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env set verbose"));
}
#endif
#ifdef CONFIG_TEST
/*
* PUBLIC: int tcl_EnvAttr __P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
*
* tcl_EnvAttr --
* Return a list of the env's attributes
*/
int
tcl_EnvAttr(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv; /* Env pointer */
{
ENV *env;
Tcl_Obj *myobj, *retlist;
int result;
env = dbenv->env;
result = TCL_OK;
if (objc > 2) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
retlist = Tcl_NewListObj(0, NULL);
/*
* XXX
* We peek at the ENV to determine what subsystems we have available
* in this environment.
*/
myobj = NewStringObj("-home", strlen("-home"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
myobj = NewStringObj(env->db_home, strlen(env->db_home));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
if (CDB_LOCKING(env)) {
myobj = NewStringObj("-cdb", strlen("-cdb"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
}
if (CRYPTO_ON(env)) {
myobj = NewStringObj("-crypto", strlen("-crypto"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
}
if (LOCKING_ON(env)) {
myobj = NewStringObj("-lock", strlen("-lock"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
}
if (LOGGING_ON(env)) {
myobj = NewStringObj("-log", strlen("-log"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
}
if (MPOOL_ON(env)) {
myobj = NewStringObj("-mpool", strlen("-mpool"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
}
if (REP_ON(env)) {
myobj = NewStringObj("-rep", strlen("-rep"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
}
if (TXN_ON(env)) {
myobj = NewStringObj("-txn", strlen("-txn"));
if ((result = Tcl_ListObjAppendElement(interp,
retlist, myobj)) != TCL_OK)
goto err;
}
Tcl_SetObjResult(interp, retlist);
err:
return (result);
}
/*
* env_EventInfo --
* Implements the ENV->event_info command.
*/
static int
env_EventInfo(interp, objc, objv, dbenv, ip)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
DBTCL_INFO *ip;
{
static const char *envinfo_option_names[] = {
"-clear",
NULL
};
enum envinfo_options {
ENVINFCLEAR
};
DB_LSN *lsn;
Tcl_Obj *lsnlist[2], *myobjv[2], *one_event, *res;
int clear, enc, i, ret, t_ret;
u_int32_t bit_flag;
static const struct {
u_int32_t flag;
char *name;
} event_names[] = {
{ DB_EVENT_PANIC, "panic" },
{ DB_EVENT_REG_ALIVE, "reg_alive" },
{ DB_EVENT_REG_PANIC, "reg_panic" },
{ DB_EVENT_REP_CLIENT, "client" },
{ DB_EVENT_REP_CONNECT_BROKEN, "connection_broken" },
{ DB_EVENT_REP_CONNECT_ESTD, "connection_established" },
{ DB_EVENT_REP_CONNECT_TRY_FAILED, "connection_retry_failed" },
{ DB_EVENT_REP_DUPMASTER, "dupmaster" },
{ DB_EVENT_REP_ELECTED, "elected" },
{ DB_EVENT_REP_ELECTION_FAILED, "election_failed" },
{ DB_EVENT_REP_JOIN_FAILURE, "join_failure" },
{ DB_EVENT_REP_LOCAL_SITE_REMOVED, "local_site_removed" },
{ DB_EVENT_REP_MASTER, "master" },
{ DB_EVENT_REP_MASTER_FAILURE, "master_failure" },
{ DB_EVENT_REP_NEWMASTER, "newmaster" },
{ DB_EVENT_REP_PERM_FAILED, "perm_failed" },
{ DB_EVENT_REP_SITE_ADDED, "site_added" },
{ DB_EVENT_REP_SITE_REMOVED, "site_removed" },
{ DB_EVENT_REP_STARTUPDONE, "startupdone" },
{ DB_EVENT_REP_WOULD_ROLLBACK, "would_rollback" },
{ DB_EVENT_WRITE_FAILED, "write_failed" },
{ DB_EVENT_NO_SUCH_EVENT, NULL }
};
/*
* Note that when this list grows to more than 32 event types, the code
* below (the shift operation) will be broken.
*/
if (objc > 3) {
Tcl_WrongNumArgs(interp, 2, objv, "?-clear?");
return (TCL_ERROR);
}
clear = 0;
if (objc == 3) {
if (Tcl_GetIndexFromObj(interp, objv[2], envinfo_option_names,
"option", TCL_EXACT, &enc) != TCL_OK)
return (IS_HELP(objv[2]));
switch ((enum envinfo_options)enc) {
case ENVINFCLEAR:
clear = 1;
break;
}
}
if(ip->i_event_info == NULL) {
/* Script needs "-event" in "berkdb env" cmd. */
Tcl_SetResult(interp,
"event collection not enabled on this env", TCL_STATIC);
return (TCL_ERROR);
}
res = Tcl_NewListObj(0, NULL);
if ((ret = tcl_LockMutex(dbenv, ip->i_mutex)) != 0)
return (_ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"mutex lock"));
ret = TCL_OK;
for (i = 0; event_names[i].flag != DB_EVENT_NO_SUCH_EVENT; i++) {
bit_flag = 1 << event_names[i].flag;
if (FLD_ISSET(ip->i_event_info->events, bit_flag)) {
/*
* Build the 2-element list "one_event", consisting of
* the event name and the event info. The event info is
* sometimes more than a scalar value: in those cases,
* reuse the "myobjv" array to build the sub-list first.
* For this reason, set up the event info first
* (myobjv[1]), and only when that is done set up the
* event name (myobjv[0])
*/
switch (event_names[i].flag) {
case DB_EVENT_PANIC:
myobjv[1] = Tcl_NewIntObj(ip->
i_event_info->panic_error);
break;
case DB_EVENT_REG_ALIVE:
myobjv[1] = Tcl_NewLongObj((long)ip->
i_event_info->attached_process);
break;
case DB_EVENT_REP_CONNECT_BROKEN:
myobjv[0] = Tcl_NewIntObj(ip->
i_event_info->conn_broken_info.eid);
myobjv[1] = Tcl_NewIntObj(ip->
i_event_info->conn_broken_info.error);
myobjv[1] = Tcl_NewListObj(2, myobjv);
break;
case DB_EVENT_REP_CONNECT_ESTD:
myobjv[1] = Tcl_NewIntObj(ip->
i_event_info->connected_eid);
break;
case DB_EVENT_REP_CONNECT_TRY_FAILED:
myobjv[0] = Tcl_NewIntObj(ip->
i_event_info->conn_failed_try_info.eid);
myobjv[1] = Tcl_NewIntObj(ip->
i_event_info->conn_failed_try_info.error);
myobjv[1] = Tcl_NewListObj(2, myobjv);
break;
case DB_EVENT_REP_NEWMASTER:
myobjv[1] = Tcl_NewIntObj(ip->
i_event_info->newmaster_eid);
break;
case DB_EVENT_REP_SITE_ADDED:
myobjv[1] = Tcl_NewIntObj(ip->
i_event_info->added_eid);
break;
case DB_EVENT_REP_SITE_REMOVED:
myobjv[1] = Tcl_NewIntObj(ip->
i_event_info->removed_eid);
break;
case DB_EVENT_REP_WOULD_ROLLBACK:
lsn = &ip->i_event_info->sync_point;
lsnlist[0] = Tcl_NewLongObj((long)lsn->file);
lsnlist[1] = Tcl_NewLongObj((long)lsn->offset);
myobjv[1] = Tcl_NewListObj(2, lsnlist);
break;
default:
myobjv[1] = NewStringObj("", 0);
break;
}
myobjv[0] = NewStringObj(event_names[i].name,
strlen(event_names[i].name));
one_event = Tcl_NewListObj(2, myobjv);
if ((ret = Tcl_ListObjAppendElement(interp,
res, one_event)) != TCL_OK)
break;
}
}
/*
* Here, either everything is OK, or Tcl_ListObjAppendElement failed,
* above. Regardless, we need to make sure we unlock the mutex. Then,
* if either operation generated an error, return it, giving precedence
* to the earlier-occurring one.
*/
t_ret = tcl_UnlockMutex(dbenv, ip->i_mutex);
if (ret != TCL_OK)
return (ret);
if (t_ret != 0)
return (_ReturnSetup(interp, t_ret, DB_RETOK_STD(t_ret),
"mutex unlock"));
Tcl_SetObjResult(interp, res);
if (clear)
ip->i_event_info->events = 0;
return (TCL_OK);
}
/*
* PUBLIC: int tcl_EnvSetFlags __P((Tcl_Interp *, DB_ENV *, Tcl_Obj *,
* PUBLIC: Tcl_Obj *));
*
* tcl_EnvSetFlags --
* Set flags in an env.
*/
int
tcl_EnvSetFlags(interp, dbenv, which, onoff)
Tcl_Interp *interp; /* Interpreter */
DB_ENV *dbenv; /* Env pointer */
Tcl_Obj *which; /* Which subsystem */
Tcl_Obj *onoff; /* On or off */
{
static const char *sfwhich[] = {
"-auto_commit",
"-direct_db",
"-dsync_db",
"-multiversion",
"-nolock",
"-nommap",
"-nopanic",
"-nosync",
"-overwrite",
"-panic",
"-snapshot",
"-time_notgranted",
"-wrnosync",
"-hotbackup_in_progress",
NULL
};
enum sfwhich {
ENVSF_AUTOCOMMIT,
ENVSF_DIRECTDB,
ENVSF_DSYNCDB,
ENVSF_MULTIVERSION,
ENVSF_NOLOCK,
ENVSF_NOMMAP,
ENVSF_NOPANIC,
ENVSF_NOSYNC,
ENVSF_OVERWRITE,
ENVSF_PANIC,
ENVSF_SNAPSHOT,
ENVSF_TIME_NOTGRANTED,
ENVSF_WRNOSYNC,
ENVSF_HOTBACKUP_IN_PROGRESS
};
static const char *sfonoff[] = {
"off",
"on",
NULL
};
enum sfonoff {
ENVSF_OFF,
ENVSF_ON
};
int on, optindex, ret;
u_int32_t wh;
if (Tcl_GetIndexFromObj(interp, which, sfwhich, "option",
TCL_EXACT, &optindex) != TCL_OK)
return (IS_HELP(which));
switch ((enum sfwhich)optindex) {
case ENVSF_AUTOCOMMIT:
wh = DB_AUTO_COMMIT;
break;
case ENVSF_DIRECTDB:
wh = DB_DIRECT_DB;
break;
case ENVSF_DSYNCDB:
wh = DB_DSYNC_DB;
break;
case ENVSF_MULTIVERSION:
wh = DB_MULTIVERSION;
break;
case ENVSF_NOLOCK:
wh = DB_NOLOCKING;
break;
case ENVSF_NOMMAP:
wh = DB_NOMMAP;
break;
case ENVSF_NOSYNC:
wh = DB_TXN_NOSYNC;
break;
case ENVSF_NOPANIC:
wh = DB_NOPANIC;
break;
case ENVSF_PANIC:
wh = DB_PANIC_ENVIRONMENT;
break;
case ENVSF_OVERWRITE:
wh = DB_OVERWRITE;
break;
case ENVSF_SNAPSHOT:
wh = DB_TXN_SNAPSHOT;
break;
case ENVSF_TIME_NOTGRANTED:
wh = DB_TIME_NOTGRANTED;
break;
case ENVSF_WRNOSYNC:
wh = DB_TXN_WRITE_NOSYNC;
break;
case ENVSF_HOTBACKUP_IN_PROGRESS:
wh = DB_HOTBACKUP_IN_PROGRESS;
break;
default:
return (TCL_ERROR);
}
if (Tcl_GetIndexFromObj(interp, onoff, sfonoff, "option",
TCL_EXACT, &optindex) != TCL_OK)
return (IS_HELP(onoff));
switch ((enum sfonoff)optindex) {
case ENVSF_OFF:
on = 0;
break;
case ENVSF_ON:
on = 1;
break;
default:
return (TCL_ERROR);
}
ret = dbenv->set_flags(dbenv, wh, on);
return (_ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env set flags"));
}
/*
* tcl_EnvTest --
* The "$env test ..." command is a sort of catch-all for any sort of
* desired test hook manipulation. The "abort", "check" and "copy" subcommands
* all set one or another certain location in the DB_ENV handle to a specific
* value. (In the case of "check", the value is an integer passed in with the
* command itself. For the other two, the "value" is a predefined enum
* constant, specified by name.)
* The "$env test force ..." subcommand invokes other, more arbitrary
* manipulations.
* Although these functions may not all seem closely related, putting them
* all under the name "test" has the aesthetic appeal of keeping the rest of the
* API clean.
*
* PUBLIC: int tcl_EnvTest __P((Tcl_Interp *, int, Tcl_Obj * CONST*, DB_ENV *));
*/
int
tcl_EnvTest(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv; /* Env pointer */
{
static const char *envtestcmd[] = {
"abort",
"check",
"copy",
"force",
NULL
};
enum envtestcmd {
ENVTEST_ABORT,
ENVTEST_CHECK,
ENVTEST_COPY,
ENVTEST_FORCE
};
static const char *envtestat[] = {
"electinit",
"electvote1",
"no_pages",
"none",
"predestroy",
"preopen",
"postdestroy",
"postlog",
"postlogmeta",
"postopen",
"postsync",
"repmgr_perm",
"subdb_lock",
NULL
};
enum envtestat {
ENVTEST_ELECTINIT,
ENVTEST_ELECTVOTE1,
ENVTEST_NO_PAGES,
ENVTEST_NONE,
ENVTEST_PREDESTROY,
ENVTEST_PREOPEN,
ENVTEST_POSTDESTROY,
ENVTEST_POSTLOG,
ENVTEST_POSTLOGMETA,
ENVTEST_POSTOPEN,
ENVTEST_POSTSYNC,
ENVTEST_REPMGR_PERM,
ENVTEST_SUBDB_LOCKS
};
static const char *envtestforce[] = {
"noarchive_timeout",
NULL
};
enum envtestforce {
ENVTEST_NOARCHIVE_TIMEOUT
};
ENV *env;
int *loc, optindex, result, testval;
env = dbenv->env;
result = TCL_OK;
loc = NULL;
if (objc != 4) {
Tcl_WrongNumArgs(interp,
2, objv, "abort|check|copy|force <args>");
return (TCL_ERROR);
}
/*
* This must be the "check", "copy" or "abort" portion of the command.
*/
if (Tcl_GetIndexFromObj(interp, objv[2], envtestcmd, "command",
TCL_EXACT, &optindex) != TCL_OK) {
result = IS_HELP(objv[2]);
return (result);
}
switch ((enum envtestcmd)optindex) {
case ENVTEST_ABORT:
loc = &env->test_abort;
break;
case ENVTEST_CHECK:
loc = &env->test_check;
if (Tcl_GetIntFromObj(interp, objv[3], &testval) != TCL_OK) {
result = IS_HELP(objv[3]);
return (result);
}
goto done;
case ENVTEST_COPY:
loc = &env->test_copy;
break;
case ENVTEST_FORCE:
if (Tcl_GetIndexFromObj(interp, objv[3], envtestforce, "arg",
TCL_EXACT, &optindex) != TCL_OK) {
result = IS_HELP(objv[3]);
return (result);
}
/*
* In the future we might add more, and then we'd use a switch
* statement.
*/
DB_ASSERT(env,
(enum envtestforce)optindex == ENVTEST_NOARCHIVE_TIMEOUT);
return (tcl_RepNoarchiveTimeout(interp, dbenv));
default:
Tcl_SetResult(interp, "Illegal store location", TCL_STATIC);
return (TCL_ERROR);
}
/*
* This must be the location portion of the command.
*/
if (Tcl_GetIndexFromObj(interp, objv[3], envtestat, "location",
TCL_EXACT, &optindex) != TCL_OK) {
result = IS_HELP(objv[3]);
return (result);
}
switch ((enum envtestat)optindex) {
case ENVTEST_ELECTINIT:
DB_ASSERT(env, loc == &env->test_abort);
testval = DB_TEST_ELECTINIT;
break;
case ENVTEST_ELECTVOTE1:
DB_ASSERT(env, loc == &env->test_abort);
testval = DB_TEST_ELECTVOTE1;
break;
case ENVTEST_NO_PAGES:
DB_ASSERT(env, loc == &env->test_abort);
testval = DB_TEST_NO_PAGES;
break;
case ENVTEST_NONE:
testval = 0;
break;
case ENVTEST_PREOPEN:
testval = DB_TEST_PREOPEN;
break;
case ENVTEST_PREDESTROY:
testval = DB_TEST_PREDESTROY;
break;
case ENVTEST_POSTLOG:
testval = DB_TEST_POSTLOG;
break;
case ENVTEST_POSTLOGMETA:
testval = DB_TEST_POSTLOGMETA;
break;
case ENVTEST_POSTOPEN:
testval = DB_TEST_POSTOPEN;
break;
case ENVTEST_POSTDESTROY:
testval = DB_TEST_POSTDESTROY;
break;
case ENVTEST_POSTSYNC:
testval = DB_TEST_POSTSYNC;
break;
case ENVTEST_REPMGR_PERM:
DB_ASSERT(env, loc == &env->test_abort);
testval = DB_TEST_REPMGR_PERM;
break;
case ENVTEST_SUBDB_LOCKS:
DB_ASSERT(env, loc == &env->test_abort);
testval = DB_TEST_SUBDB_LOCKS;
break;
default:
Tcl_SetResult(interp, "Illegal test location", TCL_STATIC);
return (TCL_ERROR);
}
done:
*loc = testval;
Tcl_SetResult(interp, "0", TCL_STATIC);
return (result);
}
#endif
/*
* env_DbRemove --
* Implements the ENV->dbremove command.
*/
static int
env_DbRemove(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
{
static const char *envdbrem[] = {
"-auto_commit",
"-notdurable",
"-txn",
"--",
NULL
};
enum envdbrem {
TCL_EDBREM_COMMIT,
TCL_EDBREM_NOTDURABLE,
TCL_EDBREM_TXN,
TCL_EDBREM_ENDARG
};
DB_TXN *txn;
u_int32_t flag;
int endarg, i, optindex, result, ret, subdblen;
u_char *subdbtmp;
char *arg, *db, *dbr, *subdb, *subdbr, msg[MSG_SIZE];
size_t nlen;
txn = NULL;
result = TCL_OK;
subdbtmp = NULL;
db = dbr = subdb = subdbr = NULL;
endarg = nlen = subdblen = 0;
flag = 0;
if (objc < 2) {
Tcl_WrongNumArgs(interp, 2, objv, "?args? filename ?database?");
return (TCL_ERROR);
}
/*
* We must first parse for the environment flag, since that
* is needed for db_create. Then create the db handle.
*/
i = 2;
while (i < objc) {
if (Tcl_GetIndexFromObj(interp, objv[i], envdbrem,
"option", TCL_EXACT, &optindex) != TCL_OK) {
arg = Tcl_GetStringFromObj(objv[i], NULL);
if (arg[0] == '-') {
result = IS_HELP(objv[i]);
goto error;
} else
Tcl_ResetResult(interp);
break;
}
i++;
switch ((enum envdbrem)optindex) {
case TCL_EDBREM_COMMIT:
flag |= DB_AUTO_COMMIT;
break;
case TCL_EDBREM_TXN:
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv, "?-txn id?");
result = TCL_ERROR;
break;
}
arg = Tcl_GetStringFromObj(objv[i++], NULL);
txn = NAME_TO_TXN(arg);
if (txn == NULL) {
snprintf(msg, MSG_SIZE,
"env dbremove: Invalid txn %s\n", arg);
Tcl_SetResult(interp, msg, TCL_VOLATILE);
return (TCL_ERROR);
}
break;
case TCL_EDBREM_ENDARG:
endarg = 1;
break;
case TCL_EDBREM_NOTDURABLE:
flag |= DB_TXN_NOT_DURABLE;
break;
}
/*
* If, at any time, parsing the args we get an error,
* bail out and return.
*/
if (result != TCL_OK)
goto error;
if (endarg)
break;
}
if (result != TCL_OK)
goto error;
/*
* Any args we have left, (better be 1 or 2 left) are
* file names. If there is 1, a db name, if 2 a db and subdb name.
*/
if ((i != (objc - 1)) || (i != (objc - 2))) {
/*
* Dbs must be NULL terminated file names, but subdbs can
* be anything. Use Strings for the db name and byte
* arrays for the subdb.
*/
db = Tcl_GetStringFromObj(objv[i++], NULL);
if (strcmp(db, "") == 0)
db = NULL;
if (i != objc) {
subdbtmp =
Tcl_GetByteArrayFromObj(objv[i++], &subdblen);
if ((ret = __os_malloc(
dbenv->env, (size_t)subdblen + 1, &subdb)) != 0) {
Tcl_SetResult(interp,
db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(subdb, subdbtmp, (size_t)subdblen);
subdb[subdblen] = '\0';
}
} else {
Tcl_WrongNumArgs(interp, 2, objv, "?args? filename ?database?");
result = TCL_ERROR;
goto error;
}
ret = dbenv->dbremove(dbenv, txn, db, subdb, flag);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env dbremove");
/*
* If we are heap, we have auxiliary dbs. However in order to tell if
* we are heap we have to open the db to get its type. So we will just
* try the dbremove and ignore ENOENT.
*/
if ((db != NULL || subdb != NULL) && ret == 0) {
/* set up file name for associated recno db for heap*/
if (db != NULL) {
nlen = strlen(db);
if ((ret = __os_malloc(
dbenv->env, nlen + 2, &dbr)) != 0) {
Tcl_SetResult(
interp, db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(dbr, db, nlen);
dbr[nlen] = '1';
dbr[nlen+1] = '\0';
}
if (subdb != NULL) {
if ((ret = __os_malloc(
dbenv->env, (size_t)subdblen + 2, &subdbr)) != 0) {
Tcl_SetResult(
interp, db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(subdbr, subdb, (size_t)subdblen);
subdbr[subdblen] = '1';
subdbr[subdblen+1] = '\0';
}
ret = dbenv->dbremove(dbenv, txn, dbr, subdbr, flag);
if (ret == ENOENT)
ret = 0;
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"db remove associated recno");
if (ret)
goto error;
/* Remove the btree which is doing the RID to recno mapping*/
if (dbr != NULL)
dbr[nlen] = '2';
if (subdbr != NULL)
subdbr[subdblen] = '2';
ret = dbenv->dbremove(dbenv, txn, dbr, subdbr, flag);
if (ret == ENOENT)
ret = 0;
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"db remove associated btree");
}
error:
if (subdb)
__os_free(dbenv->env, subdb);
if (subdbr)
__os_free(dbenv->env, subdbr);
if (dbr)
__os_free(dbenv->env, dbr);
return (result);
}
/*
* env_DbRename --
* Implements the ENV->dbrename command.
*/
static int
env_DbRename(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
{
static const char *envdbmv[] = {
"-auto_commit",
"-txn",
"--",
NULL
};
enum envdbmv {
TCL_EDBMV_COMMIT,
TCL_EDBMV_TXN,
TCL_EDBMV_ENDARG
};
DB_TXN *txn;
u_int32_t flag;
int endarg, i, newlen, optindex, result, ret, subdblen;
u_char *subdbtmp;
char *arg, *db, *dbr, *newname, *newnamer, *subdb, *subdbr;
char msg[MSG_SIZE];
size_t nlen;
txn = NULL;
result = TCL_OK;
subdbtmp = NULL;
db = dbr = newname = newnamer = subdb = subdbr = NULL;
endarg = nlen = subdblen = 0;
flag = 0;
if (objc < 2) {
Tcl_WrongNumArgs(interp, 3, objv,
"?args? filename ?database? ?newname?");
return (TCL_ERROR);
}
/*
* We must first parse for the environment flag, since that
* is needed for db_create. Then create the db handle.
*/
i = 2;
while (i < objc) {
if (Tcl_GetIndexFromObj(interp, objv[i], envdbmv,
"option", TCL_EXACT, &optindex) != TCL_OK) {
arg = Tcl_GetStringFromObj(objv[i], NULL);
if (arg[0] == '-') {
result = IS_HELP(objv[i]);
goto error;
} else
Tcl_ResetResult(interp);
break;
}
i++;
switch ((enum envdbmv)optindex) {
case TCL_EDBMV_COMMIT:
flag |= DB_AUTO_COMMIT;
break;
case TCL_EDBMV_TXN:
if (i >= objc) {
Tcl_WrongNumArgs(interp, 2, objv, "?-txn id?");
result = TCL_ERROR;
break;
}
arg = Tcl_GetStringFromObj(objv[i++], NULL);
txn = NAME_TO_TXN(arg);
if (txn == NULL) {
snprintf(msg, MSG_SIZE,
"env dbrename: Invalid txn %s\n", arg);
Tcl_SetResult(interp, msg, TCL_VOLATILE);
return (TCL_ERROR);
}
break;
case TCL_EDBMV_ENDARG:
endarg = 1;
break;
}
/*
* If, at any time, parsing the args we get an error,
* bail out and return.
*/
if (result != TCL_OK)
goto error;
if (endarg)
break;
}
if (result != TCL_OK)
goto error;
/*
* Any args we have left, (better be 2 or 3 left) are
* file names. If there is 2, a db name, if 3 a db and subdb name.
*/
if ((i != (objc - 2)) || (i != (objc - 3))) {
/*
* Dbs must be NULL terminated file names, but subdbs can
* be anything. Use Strings for the db name and byte
* arrays for the subdb.
*/
db = Tcl_GetStringFromObj(objv[i++], NULL);
if (strcmp(db, "") == 0)
db = NULL;
if (i == objc - 2) {
subdbtmp =
Tcl_GetByteArrayFromObj(objv[i++], &subdblen);
if ((ret = __os_malloc(
dbenv->env, (size_t)subdblen + 1, &subdb)) != 0) {
Tcl_SetResult(interp,
db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(subdb, subdbtmp, (size_t)subdblen);
subdb[subdblen] = '\0';
}
subdbtmp = Tcl_GetByteArrayFromObj(objv[i++], &newlen);
if ((ret = __os_malloc(
dbenv->env, (size_t)newlen + 1, &newname)) != 0) {
Tcl_SetResult(interp,
db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(newname, subdbtmp, (size_t)newlen);
newname[newlen] = '\0';
} else {
Tcl_WrongNumArgs(interp, 3, objv,
"?args? filename ?database? ?newname?");
result = TCL_ERROR;
goto error;
}
ret = dbenv->dbrename(dbenv, txn, db, subdb, newname, flag);
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env dbrename");
/*
* If we are heap, we have auxiliary dbs. However in order to tell if
* we are heap we have to open the db to get its type. So we will just
* try the dbrename and ignore ENOENT.
*/
if ((db != NULL || subdb != NULL) && ret == 0) {
/* set up file name for associated recno db for heap*/
if (db != NULL) {
nlen = strlen(db);
if ((ret = __os_malloc(
dbenv->env, nlen + 2, &dbr)) != 0) {
Tcl_SetResult(
interp, db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(dbr, db, nlen);
dbr[nlen] = '1';
dbr[nlen+1] = '\0';
}
if (subdb != NULL) {
if ((ret = __os_malloc(
dbenv->env, (size_t)subdblen + 2, &subdbr)) != 0) {
Tcl_SetResult(
interp, db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(subdbr, subdb, (size_t)subdblen);
subdbr[subdblen] = '1';
subdbr[subdblen+1] = '\0';
}
if ((ret = __os_malloc(dbenv->env,
(size_t)newlen + 2, &newnamer)) != 0) {
Tcl_SetResult(interp, db_strerror(ret), TCL_STATIC);
return (0);
}
memcpy(newnamer, newname, (size_t)newlen);
newnamer[newlen] = '1';
newnamer[newlen+1] = '\0';
ret = dbenv->dbrename(dbenv, txn, dbr, subdbr, newnamer, flag);
if (ret == ENOENT)
ret = 0;
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"db rename associated recno");
if (ret != 0)
goto error;
/* Rename the btree which is doing the RID to recno mapping*/
if (dbr != NULL)
dbr[nlen] = '2';
if (subdbr != NULL)
subdbr[subdblen] = '2';
newnamer[newlen] = '2';
ret = dbenv->dbrename(dbenv, txn, dbr, subdbr, newnamer, flag);
if (ret == ENOENT)
ret = 0;
result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"db rename associated btree");
}
error:
if (subdb)
__os_free(dbenv->env, subdb);
if (newname)
__os_free(dbenv->env, newname);
if (dbr)
__os_free(dbenv->env, dbr);
if (subdbr)
__os_free(dbenv->env, subdbr);
if (newnamer)
__os_free(dbenv->env, newnamer);
return (result);
}
/*
* env_GetFlags --
* Implements the ENV->get_flags command.
*/
static int
env_GetFlags(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
{
int i, ret, result;
u_int32_t flags;
char buf[512];
Tcl_Obj *res;
static const struct {
u_int32_t flag;
char *arg;
} open_flags[] = {
{ DB_AUTO_COMMIT, "-auto_commit" },
{ DB_CDB_ALLDB, "-cdb_alldb" },
{ DB_DIRECT_DB, "-direct_db" },
{ DB_DSYNC_DB, "-dsync_db" },
{ DB_MULTIVERSION, "-multiversion" },
{ DB_NOLOCKING, "-nolock" },
{ DB_NOMMAP, "-nommap" },
{ DB_NOPANIC, "-nopanic" },
{ DB_OVERWRITE, "-overwrite" },
{ DB_PANIC_ENVIRONMENT, "-panic" },
{ DB_REGION_INIT, "-region_init" },
{ DB_TIME_NOTGRANTED, "-time_notgranted" },
{ DB_TXN_NOSYNC, "-nosync" },
{ DB_TXN_NOWAIT, "-nowait" },
{ DB_TXN_SNAPSHOT, "-snapshot" },
{ DB_TXN_WRITE_NOSYNC, "-wrnosync" },
{ DB_YIELDCPU, "-yield" },
{ DB_HOTBACKUP_IN_PROGRESS, "-hotbackup_in_progress" },
{ 0, NULL }
};
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_flags(dbenv, &flags);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_flags")) == TCL_OK) {
buf[0] = '\0';
for (i = 0; open_flags[i].flag != 0; i++)
if (LF_ISSET(open_flags[i].flag)) {
if (strlen(buf) > 0)
(void)strncat(buf, " ", sizeof(buf));
(void)strncat(
buf, open_flags[i].arg, sizeof(buf));
}
res = NewStringObj(buf, strlen(buf));
Tcl_SetObjResult(interp, res);
}
return (result);
}
/*
* env_GetOpenFlag --
* Implements the ENV->get_open_flags command.
*/
static int
env_GetOpenFlag(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
{
int i, ret, result;
u_int32_t flags;
char buf[512];
Tcl_Obj *res;
static const struct {
u_int32_t flag;
char *arg;
} open_flags[] = {
{ DB_CREATE, "-create" },
{ DB_FAILCHK, "-failchk" },
{ DB_INIT_CDB, "-cdb" },
{ DB_INIT_LOCK, "-lock" },
{ DB_INIT_LOG, "-log" },
{ DB_INIT_MPOOL, "-mpool" },
{ DB_INIT_REP, "-rep" },
{ DB_INIT_TXN, "-txn" },
{ DB_LOCKDOWN, "-lockdown" },
{ DB_PRIVATE, "-private" },
{ DB_RECOVER, "-recover" },
{ DB_RECOVER_FATAL, "-recover_fatal" },
{ DB_REGISTER, "-register" },
{ DB_FAILCHK, "-failchk" },
{ DB_SYSTEM_MEM, "-system_mem" },
{ DB_THREAD, "-thread" },
{ DB_USE_ENVIRON, "-use_environ" },
{ DB_USE_ENVIRON_ROOT, "-use_environ_root" },
{ 0, NULL }
};
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_open_flags(dbenv, &flags);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_open_flags")) == TCL_OK) {
buf[0] = '\0';
for (i = 0; open_flags[i].flag != 0; i++)
if (LF_ISSET(open_flags[i].flag)) {
if (strlen(buf) > 0)
(void)strncat(buf, " ", sizeof(buf));
(void)strncat(
buf, open_flags[i].arg, sizeof(buf));
}
res = NewStringObj(buf, strlen(buf));
Tcl_SetObjResult(interp, res);
}
return (result);
}
/*
* PUBLIC: int tcl_EnvGetEncryptFlags __P((Tcl_Interp *, int, Tcl_Obj * CONST*,
* PUBLIC: DB_ENV *));
*
* tcl_EnvGetEncryptFlags --
* Implements the ENV->get_encrypt_flags command.
*/
int
tcl_EnvGetEncryptFlags(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv; /* Database pointer */
{
int i, ret, result;
u_int32_t flags;
char buf[512];
Tcl_Obj *res;
static const struct {
u_int32_t flag;
char *arg;
} encrypt_flags[] = {
{ DB_ENCRYPT_AES, "-encryptaes" },
{ 0, NULL }
};
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_encrypt_flags(dbenv, &flags);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_encrypt_flags")) == TCL_OK) {
buf[0] = '\0';
for (i = 0; encrypt_flags[i].flag != 0; i++)
if (LF_ISSET(encrypt_flags[i].flag)) {
if (strlen(buf) > 0)
(void)strncat(buf, " ", sizeof(buf));
(void)strncat(
buf, encrypt_flags[i].arg, sizeof(buf));
}
res = NewStringObj(buf, strlen(buf));
Tcl_SetObjResult(interp, res);
}
return (result);
}
/*
* env_GetLockDetect --
* Implements the ENV->get_lk_detect command.
*/
static int
env_GetLockDetect(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
{
int i, ret, result;
u_int32_t lk_detect;
const char *answer;
Tcl_Obj *res;
static const struct {
u_int32_t flag;
char *name;
} lk_detect_returns[] = {
{ DB_LOCK_DEFAULT, "default" },
{ DB_LOCK_EXPIRE, "expire" },
{ DB_LOCK_MAXLOCKS, "maxlocks" },
{ DB_LOCK_MAXWRITE, "maxwrite" },
{ DB_LOCK_MINLOCKS, "minlocks" },
{ DB_LOCK_MINWRITE, "minwrite" },
{ DB_LOCK_OLDEST, "oldest" },
{ DB_LOCK_RANDOM, "random" },
{ DB_LOCK_YOUNGEST, "youngest" },
{ 0, NULL }
};
if (objc != 2) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
ret = dbenv->get_lk_detect(dbenv, &lk_detect);
if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_lk_detect")) == TCL_OK) {
answer = "unknown";
for (i = 0; lk_detect_returns[i].flag != 0; i++)
if (lk_detect == lk_detect_returns[i].flag)
answer = lk_detect_returns[i].name;
res = NewStringObj(answer, strlen(answer));
Tcl_SetObjResult(interp, res);
}
return (result);
}
/*
* env_GetTimeout --
* Implements the ENV->get_timeout command.
*/
static int
env_GetTimeout(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
{
static const struct {
u_int32_t flag;
char *arg;
} timeout_flags[] = {
{ DB_SET_LOCK_TIMEOUT, "lock" },
{ DB_SET_REG_TIMEOUT, "reg" },
{ DB_SET_TXN_TIMEOUT, "txn" },
{ 0, NULL }
};
Tcl_Obj *res;
db_timeout_t timeout;
u_int32_t which;
int i, ret, result;
const char *arg;
COMPQUIET(timeout, 0);
if (objc != 3) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
arg = Tcl_GetStringFromObj(objv[2], NULL);
which = 0;
for (i = 0; timeout_flags[i].flag != 0; i++)
if (strcmp(arg, timeout_flags[i].arg) == 0)
which = timeout_flags[i].flag;
if (which == 0) {
ret = EINVAL;
goto err;
}
ret = dbenv->get_timeout(dbenv, &timeout, which);
err: if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_timeout")) == TCL_OK) {
res = Tcl_NewLongObj((long)timeout);
Tcl_SetObjResult(interp, res);
}
return (result);
}
/*
* env_GetVerbose --
* Implements the ENV->get_open_flags command.
*/
static int
env_GetVerbose(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv;
{
static const struct {
u_int32_t flag;
char *arg;
} verbose_flags[] = {
{ DB_VERB_DEADLOCK, "deadlock" },
{ DB_VERB_FILEOPS, "fileops" },
{ DB_VERB_FILEOPS_ALL, "fileops_all" },
{ DB_VERB_RECOVERY, "recovery" },
{ DB_VERB_REGISTER, "register" },
{ DB_VERB_REPLICATION, "rep" },
{ DB_VERB_REP_ELECT, "rep_elect" },
{ DB_VERB_REP_LEASE, "rep_lease" },
{ DB_VERB_REP_MISC, "rep_misc" },
{ DB_VERB_REP_MSGS, "rep_msgs" },
{ DB_VERB_REP_SYNC, "rep_sync" },
{ DB_VERB_REP_SYSTEM, "rep_system" },
{ DB_VERB_REP_TEST, "rep_test" },
{ DB_VERB_REPMGR_CONNFAIL, "repmgr_connfail" },
{ DB_VERB_REPMGR_MISC, "repmgr_misc" },
{ DB_VERB_WAITSFOR, "wait" },
{ 0, NULL }
};
Tcl_Obj *res;
u_int32_t which;
int i, onoff, ret, result;
const char *arg, *answer;
COMPQUIET(onoff, 0);
if (objc != 3) {
Tcl_WrongNumArgs(interp, 1, objv, NULL);
return (TCL_ERROR);
}
arg = Tcl_GetStringFromObj(objv[2], NULL);
which = 0;
for (i = 0; verbose_flags[i].flag != 0; i++)
if (strcmp(arg, verbose_flags[i].arg) == 0)
which = verbose_flags[i].flag;
if (which == 0) {
ret = EINVAL;
goto err;
}
ret = dbenv->get_verbose(dbenv, which, &onoff);
err: if ((result = _ReturnSetup(interp, ret, DB_RETOK_STD(ret),
"env get_verbose")) == 0) {
answer = onoff ? "on" : "off";
res = NewStringObj(answer, strlen(answer));
Tcl_SetObjResult(interp, res);
}
return (result);
}
/*
* PUBLIC: void tcl_EnvSetErrfile __P((Tcl_Interp *, DB_ENV *, DBTCL_INFO *,
* PUBLIC: char *));
*
* tcl_EnvSetErrfile --
* Implements the ENV->set_errfile command.
*/
void
tcl_EnvSetErrfile(interp, dbenv, ip, errf)
Tcl_Interp *interp; /* Interpreter */
DB_ENV *dbenv; /* Database pointer */
DBTCL_INFO *ip; /* Our internal info */
char *errf;
{
COMPQUIET(interp, NULL);
/*
* If the user already set one, free it.
*/
if (ip->i_err != NULL && ip->i_err != stdout &&
ip->i_err != stderr)
(void)fclose(ip->i_err);
if (strcmp(errf, "/dev/stdout") == 0)
ip->i_err = stdout;
else if (strcmp(errf, "/dev/stderr") == 0)
ip->i_err = stderr;
else
ip->i_err = fopen(errf, "a");
if (ip->i_err != NULL)
dbenv->set_errfile(dbenv, ip->i_err);
}
/*
* PUBLIC: void tcl_EnvSetMsgfile __P((Tcl_Interp *, DB_ENV *, DBTCL_INFO *,
* PUBLIC: char *));
*
* tcl_EnvSetMsgfile --
* Implements the ENV->set_msgfile command.
*/
void
tcl_EnvSetMsgfile(interp, dbenv, ip, msgf)
Tcl_Interp *interp; /* Interpreter */
DB_ENV *dbenv; /* Database pointer */
DBTCL_INFO *ip; /* Our internal info */
char *msgf;
{
COMPQUIET(interp, NULL);
/*
* If the user already set one, free it.
*/
if (ip->i_msg != NULL && ip->i_msg != stdout &&
ip->i_msg != stderr)
(void)fclose(ip->i_msg);
if (strcmp(msgf, "/dev/stdout") == 0)
ip->i_msg = stdout;
else if (strcmp(msgf, "/dev/stderr") == 0)
ip->i_msg = stderr;
else
ip->i_msg = fopen(msgf, "a");
if (ip->i_msg != NULL)
dbenv->set_msgfile(dbenv, ip->i_msg);
}
/*
* PUBLIC: int tcl_EnvSetErrpfx __P((Tcl_Interp *, DB_ENV *, DBTCL_INFO *,
* PUBLIC: char *));
*
* tcl_EnvSetErrpfx --
* Implements the ENV->set_errpfx command.
*/
int
tcl_EnvSetErrpfx(interp, dbenv, ip, pfx)
Tcl_Interp *interp; /* Interpreter */
DB_ENV *dbenv; /* Database pointer */
DBTCL_INFO *ip; /* Our internal info */
char *pfx;
{
int result, ret;
/*
* Assume success. The only thing that can fail is
* the __os_strdup.
*/
result = TCL_OK;
Tcl_SetResult(interp, "0", TCL_STATIC);
/*
* If the user already set one, free it.
*/
if (ip->i_errpfx != NULL) {
dbenv->set_errpfx(dbenv, NULL);
__os_free(dbenv->env, ip->i_errpfx);
ip->i_errpfx = NULL;
}
if ((ret = __os_strdup(dbenv->env, pfx, &ip->i_errpfx)) != 0) {
result = _ReturnSetup(interp, ret,
DB_RETOK_STD(ret), "__os_strdup");
ip->i_errpfx = NULL;
}
if (ip->i_errpfx != NULL)
dbenv->set_errpfx(dbenv, ip->i_errpfx);
return (result);
}
/*
* tcl_EnvStatPrint --
*
* PUBLIC: int tcl_EnvStatPrint __P((Tcl_Interp *, int,
* PUBLIC: Tcl_Obj * CONST*, DB_ENV *));
*/
int
tcl_EnvStatPrint(interp, objc, objv, dbenv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
DB_ENV *dbenv; /* Environment pointer */
{
static const char *envstatprtopts[] = {
"-all",
"-clear",
"-subsystem",
NULL
};
enum envstatprtopts {
ENVSTATPRTALL,
ENVSTATPRTCLEAR,
ENVSTATPRTSUB
};
u_int32_t flag;
int i, optindex, result, ret;
result = TCL_OK;
flag = 0;
i = 2;
while (i < objc) {
if (Tcl_GetIndexFromObj(interp, objv[i], envstatprtopts,
"option", TCL_EXACT, &optindex) != TCL_OK) {
result = IS_HELP(objv[i]);
goto error;
}
i++;
switch ((enum envstatprtopts)optindex) {
case ENVSTATPRTALL:
flag |= DB_STAT_ALL;
break;
case ENVSTATPRTCLEAR:
flag |= DB_STAT_CLEAR;
break;
case ENVSTATPRTSUB:
flag |= DB_STAT_SUBSYSTEM;
break;
}
if (result != TCL_OK)
break;
}
if (result != TCL_OK)
goto error;
_debug_check();
ret = dbenv->stat_print(dbenv, flag);
result = _ReturnSetup(interp,
ret, DB_RETOK_STD(ret), "dbenv stat_print");
error:
return (result);
}