libdb/lang/tcl/tcl_util.c
2012-11-14 16:35:20 -05:00

152 lines
3.2 KiB
C

/*-
* See the file LICENSE for redistribution information.
*
* Copyright (c) 1999, 2012 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/tcl_db.h"
/*
* bdb_RandCommand --
* Implements rand* functions.
*
* PUBLIC: int bdb_RandCommand __P((Tcl_Interp *, int, Tcl_Obj * CONST*));
*/
int
bdb_RandCommand(interp, objc, objv)
Tcl_Interp *interp; /* Interpreter */
int objc; /* How many arguments? */
Tcl_Obj *CONST objv[]; /* The argument objects */
{
static const char *rcmds[] = {
"rand", "random_int", "srand",
NULL
};
enum rcmds {
RRAND, RRAND_INT, RSRAND
};
Tcl_Obj *res;
int cmdindex, hi, lo, result, ret;
result = TCL_OK;
/*
* Get the command name index from the object based on the cmds
* defined above. This SHOULD NOT fail because we already checked
* in the 'berkdb' command.
*/
if (Tcl_GetIndexFromObj(interp,
objv[1], rcmds, "command", TCL_EXACT, &cmdindex) != TCL_OK)
return (IS_HELP(objv[1]));
res = NULL;
switch ((enum rcmds)cmdindex) {
case RRAND:
/*
* Must be 0 args. Error if different.
*/
if (objc != 2) {
Tcl_WrongNumArgs(interp, 2, objv, NULL);
return (TCL_ERROR);
}
#ifdef HAVE_RANDOM
ret = random();
#else
ret = rand();
#endif
res = Tcl_NewIntObj(ret);
break;
case RRAND_INT:
/*
* Must be 4 args. Error if different.
*/
if (objc != 4) {
Tcl_WrongNumArgs(interp, 2, objv, "lo hi");
return (TCL_ERROR);
}
if ((result =
Tcl_GetIntFromObj(interp, objv[2], &lo)) != TCL_OK)
return (result);
if ((result =
Tcl_GetIntFromObj(interp, objv[3], &hi)) != TCL_OK)
return (result);
if (lo < 0 || hi < 0) {
Tcl_SetResult(interp,
"Range value less than 0", TCL_STATIC);
return (TCL_ERROR);
}
_debug_check();
#ifdef HAVE_RANDOM
ret = lo + random() % ((hi - lo) + 1);
#else
ret = lo + rand() % ((hi - lo) + 1);
#endif
res = Tcl_NewIntObj(ret);
break;
case RSRAND:
/*
* Must be 1 arg. Error if different.
*/
if (objc != 3) {
Tcl_WrongNumArgs(interp, 2, objv, "seed");
return (TCL_ERROR);
}
if ((result =
Tcl_GetIntFromObj(interp, objv[2], &lo)) == TCL_OK) {
#ifdef HAVE_RANDOM
srandom((u_int)lo);
#else
srand((u_int)lo);
#endif
res = Tcl_NewIntObj(0);
}
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_LockMutex __P((DB_ENV *, db_mutex_t));
*/
int
tcl_LockMutex(dbenv, mutex)
DB_ENV *dbenv;
db_mutex_t mutex;
{
/*
* Why such a seemingly ridiculously trivial function? MUTEX_LOCK can't
* be invoked in a void function. The behavior of the macro could be
* unwrapped and duplicated in line; but by the time you account for
* HAVE_MUTEX_SUPPORT, checking for MUTEX_INVALID, etc., you've created
* a maintenance burden, and it's just not worth it.
*/
MUTEX_LOCK(dbenv->env, mutex);
return (0);
}
/*
* PUBLIC: int tcl_UnlockMutex __P((DB_ENV *, db_mutex_t));
*/
int
tcl_UnlockMutex(dbenv, mutex)
DB_ENV *dbenv;
db_mutex_t mutex;
{
MUTEX_UNLOCK(dbenv->env, mutex);
return (0);
}