1081 lines
28 KiB
C
1081 lines
28 KiB
C
/*
|
||
* Implementation of most standard Tcl list processing commands
|
||
* suitable for operation on thread shared (list) variables.
|
||
*
|
||
* Copyright (c) 2002 by Zoran Vasiljevic.
|
||
*
|
||
* See the file "license.terms" for information on usage and redistribution
|
||
* of this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||
* ----------------------------------------------------------------------------
|
||
*/
|
||
|
||
#include "threadSvCmd.h"
|
||
#include "threadSvListCmd.h"
|
||
|
||
#if defined(USE_TCL_STUBS)
|
||
/* Little hack to eliminate the need for "tclInt.h" here:
|
||
Just copy a small portion of TclIntStubs, just
|
||
enough to make it work */
|
||
typedef struct TclIntStubs {
|
||
int magic;
|
||
void *hooks;
|
||
void (*dummy[34]) (void); /* dummy entries 0-33, not used */
|
||
int (*tclGetIntForIndex) (Tcl_Interp *interp, Tcl_Obj *objPtr, int endValue, int *indexPtr); /* 34 */
|
||
} TclIntStubs;
|
||
extern const TclIntStubs *tclIntStubsPtr;
|
||
|
||
# undef Tcl_GetIntForIndex
|
||
# define Tcl_GetIntForIndex(interp, obj, max, ptr) ((tclIntStubsPtr->tclGetIntForIndex == NULL)? \
|
||
((int (*)(Tcl_Interp*, Tcl_Obj *, int, int*))(void *)((&(tclStubsPtr->tcl_PkgProvideEx))[645]))((interp), (obj), (max), (ptr)): \
|
||
tclIntStubsPtr->tclGetIntForIndex((interp), (obj), (max), (ptr)))
|
||
#elif TCL_MINOR_VERSION < 7
|
||
extern int TclGetIntForIndex(Tcl_Interp*, Tcl_Obj *, int, int*);
|
||
# define Tcl_GetIntForIndex TclGetIntForIndex
|
||
#endif
|
||
|
||
|
||
/*
|
||
* Implementation of list commands for shared variables.
|
||
* Most of the standard Tcl list commands are implemented.
|
||
* There are also two new commands: "lpop" and "lpush".
|
||
* Those are very convenient for simple stack operations.
|
||
*
|
||
* Main difference to standard Tcl commands is that our commands
|
||
* operate on list variable per-reference instead per-value.
|
||
* This way we avoid frequent object shuffling between shared
|
||
* containers and current interpreter, thus increasing speed.
|
||
*/
|
||
|
||
static Tcl_ObjCmdProc SvLpopObjCmd; /* lpop */
|
||
static Tcl_ObjCmdProc SvLpushObjCmd; /* lpush */
|
||
static Tcl_ObjCmdProc SvLappendObjCmd; /* lappend */
|
||
static Tcl_ObjCmdProc SvLreplaceObjCmd; /* lreplace */
|
||
static Tcl_ObjCmdProc SvLlengthObjCmd; /* llength */
|
||
static Tcl_ObjCmdProc SvLindexObjCmd; /* lindex */
|
||
static Tcl_ObjCmdProc SvLinsertObjCmd; /* linsert */
|
||
static Tcl_ObjCmdProc SvLrangeObjCmd; /* lrange */
|
||
static Tcl_ObjCmdProc SvLsearchObjCmd; /* lsearch */
|
||
static Tcl_ObjCmdProc SvLsetObjCmd; /* lset */
|
||
|
||
/*
|
||
* Inefficient list duplicator function which,
|
||
* however, produces deep list copies, unlike
|
||
* the original, which just makes shallow copies.
|
||
*/
|
||
|
||
static void DupListObjShared(Tcl_Obj*, Tcl_Obj*);
|
||
|
||
/*
|
||
* This mutex protects a static variable which tracks
|
||
* registration of commands and object types.
|
||
*/
|
||
|
||
static Tcl_Mutex initMutex;
|
||
|
||
/*
|
||
* Functions for implementing the "lset" list command
|
||
*/
|
||
|
||
static Tcl_Obj*
|
||
SvLsetFlat(Tcl_Interp *interp, Tcl_Obj *listPtr, int indexCount,
|
||
Tcl_Obj **indexArray, Tcl_Obj *valuePtr);
|
||
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* Sv_RegisterListCommands --
|
||
*
|
||
* Register list commands with shared variable module.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* Memory gets allocated
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
void
|
||
Sv_RegisterListCommands(void)
|
||
{
|
||
static int initialized = 0;
|
||
|
||
if (initialized == 0) {
|
||
Tcl_MutexLock(&initMutex);
|
||
if (initialized == 0) {
|
||
/* Create list with 1 empty element. */
|
||
Tcl_Obj *listobj = Tcl_NewObj();
|
||
listobj = Tcl_NewListObj(1, &listobj);
|
||
Sv_RegisterObjType(listobj->typePtr, DupListObjShared);
|
||
Tcl_DecrRefCount(listobj);
|
||
|
||
Sv_RegisterCommand("lpop", SvLpopObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("lpush", SvLpushObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("lappend", SvLappendObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("lreplace", SvLreplaceObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("linsert", SvLinsertObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("llength", SvLlengthObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("lindex", SvLindexObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("lrange", SvLrangeObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("lsearch", SvLsearchObjCmd, NULL, 0);
|
||
Sv_RegisterCommand("lset", SvLsetObjCmd, NULL, 0);
|
||
|
||
initialized = 1;
|
||
}
|
||
Tcl_MutexUnlock(&initMutex);
|
||
}
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLpopObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lpop" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLpopObjCmd (
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
int ret, off, llen, iarg = 0;
|
||
int index = 0;
|
||
Tcl_Obj *elPtr = NULL;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lpop array key ?index?
|
||
* $list lpop ?index?
|
||
*/
|
||
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, 0);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) > 1) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "?index?");
|
||
goto cmd_err;
|
||
}
|
||
if ((objc - off) == 1) {
|
||
iarg = off;
|
||
}
|
||
ret = Tcl_ListObjLength(interp, svObj->tclObj, &llen);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
if (iarg) {
|
||
ret = Tcl_GetIntForIndex(interp, objv[iarg], llen-1, &index);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
}
|
||
if ((index < 0) || (index >= llen)) {
|
||
goto cmd_ok; /* Ignore out-of bounds, like Tcl does */
|
||
}
|
||
ret = Tcl_ListObjIndex(interp, svObj->tclObj, index, &elPtr);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
|
||
Tcl_IncrRefCount(elPtr);
|
||
ret = Tcl_ListObjReplace(interp, svObj->tclObj, index, 1, 0, NULL);
|
||
if (ret != TCL_OK) {
|
||
Tcl_DecrRefCount(elPtr);
|
||
goto cmd_err;
|
||
}
|
||
Tcl_SetObjResult(interp, elPtr);
|
||
Tcl_DecrRefCount(elPtr);
|
||
|
||
cmd_ok:
|
||
return Sv_PutContainer(interp, svObj, SV_CHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLpushObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lpush" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLpushObjCmd (
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
int off, ret, flg, llen;
|
||
int index = 0;
|
||
Tcl_Obj *args[1];
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lpush array key element ?index?
|
||
* $list lpush element ?index?
|
||
*/
|
||
|
||
flg = FLAGS_CREATEARRAY | FLAGS_CREATEVAR;
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, flg);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) < 1) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "element ?index?");
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_ListObjLength(interp, svObj->tclObj, &llen);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
if ((objc - off) == 2) {
|
||
ret = Tcl_GetIntForIndex(interp, objv[off+1], llen, &index);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
if (index < 0) {
|
||
index = 0;
|
||
} else if (index > llen) {
|
||
index = llen;
|
||
}
|
||
}
|
||
|
||
args[0] = Sv_DuplicateObj(objv[off]);
|
||
ret = Tcl_ListObjReplace(interp, svObj->tclObj, index, 0, 1, args);
|
||
if (ret != TCL_OK) {
|
||
Tcl_DecrRefCount(args[0]);
|
||
goto cmd_err;
|
||
}
|
||
|
||
return Sv_PutContainer(interp, svObj, SV_CHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLappendObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lappend" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLappendObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
int i, ret, flg, off;
|
||
Tcl_Obj *dup;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lappend array key value ?value ...?
|
||
* $list lappend value ?value ...?
|
||
*/
|
||
|
||
flg = FLAGS_CREATEARRAY | FLAGS_CREATEVAR;
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, flg);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) < 1) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "value ?value ...?");
|
||
goto cmd_err;
|
||
}
|
||
for (i = off; i < objc; i++) {
|
||
dup = Sv_DuplicateObj(objv[i]);
|
||
ret = Tcl_ListObjAppendElement(interp, svObj->tclObj, dup);
|
||
if (ret != TCL_OK) {
|
||
Tcl_DecrRefCount(dup);
|
||
goto cmd_err;
|
||
}
|
||
}
|
||
|
||
Tcl_SetObjResult(interp, Sv_DuplicateObj(svObj->tclObj));
|
||
|
||
return Sv_PutContainer(interp, svObj, SV_CHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLreplaceObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lreplace" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLreplaceObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
const char *firstArg;
|
||
size_t argLen;
|
||
int ret, off, llen, ndel, nargs, i, j;
|
||
int first, last;
|
||
Tcl_Obj **args = NULL;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lreplace array key first last ?element ...?
|
||
* $list lreplace first last ?element ...?
|
||
*/
|
||
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, 0);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) < 2) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "first last ?element ...?");
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_ListObjLength(interp, svObj->tclObj, &llen);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_GetIntForIndex(interp, objv[off], llen-1, &first);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_GetIntForIndex(interp, objv[off+1], llen-1, &last);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
|
||
firstArg = Tcl_GetString(objv[off]);
|
||
argLen = objv[off]->length;
|
||
if (first < 0) {
|
||
first = 0;
|
||
}
|
||
if (llen && first >= llen && strncmp(firstArg, "end", argLen)) {
|
||
Tcl_AppendResult(interp, "list doesn't have element ", firstArg, NULL);
|
||
goto cmd_err;
|
||
}
|
||
if (last >= llen) {
|
||
last = llen - 1;
|
||
}
|
||
if (first <= last) {
|
||
ndel = last - first + 1;
|
||
} else {
|
||
ndel = 0;
|
||
}
|
||
|
||
nargs = objc - (off + 2);
|
||
if (nargs) {
|
||
args = (Tcl_Obj**)ckalloc(nargs * sizeof(Tcl_Obj*));
|
||
for(i = off + 2, j = 0; i < objc; i++, j++) {
|
||
args[j] = Sv_DuplicateObj(objv[i]);
|
||
}
|
||
}
|
||
|
||
ret = Tcl_ListObjReplace(interp, svObj->tclObj, first, ndel, nargs, args);
|
||
if (args) {
|
||
if (ret != TCL_OK) {
|
||
for(i = off + 2, j = 0; i < objc; i++, j++) {
|
||
Tcl_DecrRefCount(args[j]);
|
||
}
|
||
}
|
||
ckfree((char*)args);
|
||
}
|
||
|
||
return Sv_PutContainer(interp, svObj, SV_CHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLrangeObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lrange" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLrangeObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
int ret, off, llen, nargs, j;
|
||
int first, last, i;
|
||
Tcl_Obj **elPtrs, **args;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lrange array key first last
|
||
* $list lrange first last
|
||
*/
|
||
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, 0);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) != 2) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "first last");
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_ListObjGetElements(interp, svObj->tclObj, &llen, &elPtrs);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_GetIntForIndex(interp, objv[off], llen-1, &first);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_GetIntForIndex(interp, objv[off+1], llen-1, &last);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
if (first < 0) {
|
||
first = 0;
|
||
}
|
||
if (last >= llen) {
|
||
last = llen - 1;
|
||
}
|
||
if (first > last) {
|
||
goto cmd_ok;
|
||
}
|
||
|
||
nargs = last - first + 1;
|
||
args = (Tcl_Obj**)ckalloc(nargs * sizeof(Tcl_Obj*));
|
||
for (i = first, j = 0; i <= last; i++, j++) {
|
||
args[j] = Sv_DuplicateObj(elPtrs[i]);
|
||
}
|
||
|
||
Tcl_ResetResult(interp);
|
||
Tcl_SetListObj(Tcl_GetObjResult(interp), nargs, args);
|
||
ckfree((char*)args);
|
||
|
||
cmd_ok:
|
||
return Sv_PutContainer(interp, svObj, SV_UNCHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLinsertObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::linsert" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLinsertObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
int off, ret, flg, llen, nargs, i, j;
|
||
int index = 0;
|
||
Tcl_Obj **args;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::linsert array key index element ?element ...?
|
||
* $list linsert element ?element ...?
|
||
*/
|
||
|
||
flg = FLAGS_CREATEARRAY | FLAGS_CREATEVAR;
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, flg);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) < 2) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "index element ?element ...?");
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_ListObjLength(interp, svObj->tclObj, &llen);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_GetIntForIndex(interp, objv[off], llen, &index);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
if (index < 0) {
|
||
index = 0;
|
||
} else if (index > llen) {
|
||
index = llen;
|
||
}
|
||
|
||
nargs = objc - off - 1;
|
||
args = (Tcl_Obj**)ckalloc(nargs * sizeof(Tcl_Obj*));
|
||
for (i = off + 1, j = 0; i < objc; i++, j++) {
|
||
args[j] = Sv_DuplicateObj(objv[i]);
|
||
}
|
||
ret = Tcl_ListObjReplace(interp, svObj->tclObj, index, 0, nargs, args);
|
||
if (ret != TCL_OK) {
|
||
for (i = off + 1, j = 0; i < objc; i++, j++) {
|
||
Tcl_DecrRefCount(args[j]);
|
||
}
|
||
ckfree((char*)args);
|
||
goto cmd_err;
|
||
}
|
||
|
||
ckfree((char*)args);
|
||
|
||
return Sv_PutContainer(interp, svObj, SV_CHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLlengthObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::llength" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLlengthObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
int llen, off, ret;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::llength array key
|
||
* $list llength
|
||
*/
|
||
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, 0);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
|
||
ret = Tcl_ListObjLength(interp, svObj->tclObj, &llen);
|
||
if (ret == TCL_OK) {
|
||
Tcl_SetObjResult(interp, Tcl_NewIntObj(llen));
|
||
}
|
||
if (Sv_PutContainer(interp, svObj, SV_UNCHANGED) != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
|
||
return ret;
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLsearchObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lsearch" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLsearchObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
size_t length;
|
||
int ret, off, listc, mode, imode, ipatt, index, match, i;
|
||
const char *patBytes;
|
||
Tcl_Obj **listv;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
static const char *modes[] = {"-exact", "-glob", "-regexp", NULL};
|
||
enum {LS_EXACT, LS_GLOB, LS_REGEXP};
|
||
|
||
mode = LS_GLOB;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lsearch array key ?mode? pattern
|
||
* $list lsearch ?mode? pattern
|
||
*/
|
||
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, 0);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) == 2) {
|
||
imode = off;
|
||
ipatt = off + 1;
|
||
} else if ((objc - off) == 1) {
|
||
imode = 0;
|
||
ipatt = off;
|
||
} else {
|
||
Tcl_WrongNumArgs(interp, off, objv, "?mode? pattern");
|
||
goto cmd_err;
|
||
}
|
||
if (imode) {
|
||
ret = Tcl_GetIndexFromObjStruct(interp, objv[imode], modes, sizeof(char *), "search mode",
|
||
0, &mode);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
}
|
||
ret = Tcl_ListObjGetElements(interp, svObj->tclObj, &listc, &listv);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
|
||
index = -1;
|
||
patBytes = Tcl_GetString(objv[ipatt]);
|
||
length = objv[ipatt]->length;
|
||
|
||
for (i = 0; i < listc; i++) {
|
||
match = 0;
|
||
switch (mode) {
|
||
case LS_GLOB:
|
||
match = Tcl_StringCaseMatch(Tcl_GetString(listv[i]), patBytes, 0);
|
||
break;
|
||
|
||
case LS_EXACT: {
|
||
const char *bytes = Tcl_GetString(listv[i]);
|
||
if (length == (size_t)listv[i]->length) {
|
||
match = (memcmp(bytes, patBytes, length) == 0);
|
||
}
|
||
break;
|
||
}
|
||
case LS_REGEXP:
|
||
match = Tcl_RegExpMatchObj(interp, listv[i], objv[ipatt]);
|
||
if (match < 0) {
|
||
goto cmd_err;
|
||
}
|
||
break;
|
||
}
|
||
if (match) {
|
||
index = i;
|
||
break;
|
||
}
|
||
}
|
||
|
||
Tcl_SetObjResult(interp, Tcl_NewIntObj(index));
|
||
|
||
return Sv_PutContainer(interp, svObj, SV_UNCHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLindexObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lindex" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLindexObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
Tcl_Obj **elPtrs;
|
||
int ret, off, llen;
|
||
int index;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lindex array key index
|
||
* $list lindex index
|
||
*/
|
||
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, 0);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) != 1) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "index");
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_ListObjGetElements(interp, svObj->tclObj, &llen, &elPtrs);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
ret = Tcl_GetIntForIndex(interp, objv[off], llen-1, &index);
|
||
if (ret != TCL_OK) {
|
||
goto cmd_err;
|
||
}
|
||
if ((index >= 0) && (index < llen)) {
|
||
Tcl_SetObjResult(interp, Sv_DuplicateObj(elPtrs[index]));
|
||
}
|
||
|
||
return Sv_PutContainer(interp, svObj, SV_UNCHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* SvLsetObjCmd --
|
||
*
|
||
* This procedure is invoked to process the "tsv::lset" command.
|
||
* See the user documentation for details on what it does.
|
||
*
|
||
* Results:
|
||
* A standard Tcl result.
|
||
*
|
||
* Side effects:
|
||
* See the user documentation.
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static int
|
||
SvLsetObjCmd(
|
||
ClientData arg,
|
||
Tcl_Interp *interp,
|
||
int objc,
|
||
Tcl_Obj *const objv[]
|
||
) {
|
||
Tcl_Obj *lPtr;
|
||
int ret, argc, off;
|
||
Container *svObj = (Container*)arg;
|
||
|
||
/*
|
||
* Syntax:
|
||
* tsv::lset array key index ?index ...? value
|
||
* $list lset index ?index ...? value
|
||
*/
|
||
|
||
ret = Sv_GetContainer(interp, objc, objv, &svObj, &off, 0);
|
||
if (ret != TCL_OK) {
|
||
return TCL_ERROR;
|
||
}
|
||
if ((objc - off) < 2) {
|
||
Tcl_WrongNumArgs(interp, off, objv, "index ?index...? value");
|
||
goto cmd_err;
|
||
}
|
||
|
||
lPtr = svObj->tclObj;
|
||
argc = objc - off - 1;
|
||
|
||
if (!SvLsetFlat(interp, lPtr, argc, (Tcl_Obj**)objv+off,objv[objc-1])) {
|
||
return TCL_ERROR;
|
||
}
|
||
|
||
Tcl_SetObjResult(interp, Sv_DuplicateObj(lPtr));
|
||
|
||
return Sv_PutContainer(interp, svObj, SV_CHANGED);
|
||
|
||
cmd_err:
|
||
return Sv_PutContainer(interp, svObj, SV_ERROR);
|
||
}
|
||
|
||
/*
|
||
*-----------------------------------------------------------------------------
|
||
*
|
||
* DupListObjShared --
|
||
*
|
||
* Help function to make a proper deep copy of the list object.
|
||
* This is used as the replacement-hook for list object native
|
||
* DupInternalRep function. We need it since the native function
|
||
* does a shallow list copy, i.e. retains references to list
|
||
* element objects from the original list. This gives us trouble
|
||
* when making the list object shared between threads.
|
||
*
|
||
* Results:
|
||
* None.
|
||
*
|
||
* Side effects;
|
||
* This is not a very efficient implementation, but that's all what's
|
||
* available to Tcl API programmer. We could include the tclInt.h and
|
||
* get the copy more efficient using list internals, but ...
|
||
*
|
||
*-----------------------------------------------------------------------------
|
||
*/
|
||
|
||
static void
|
||
DupListObjShared(
|
||
Tcl_Obj *srcPtr, /* Object with internal rep to copy. */
|
||
Tcl_Obj *copyPtr /* Object with internal rep to set. */
|
||
) {
|
||
int i, llen;
|
||
Tcl_Obj *elObj, **newObjList;
|
||
|
||
Tcl_ListObjLength(NULL, srcPtr, &llen);
|
||
if (llen == 0) {
|
||
(*srcPtr->typePtr->dupIntRepProc)(srcPtr, copyPtr);
|
||
copyPtr->refCount = 0;
|
||
return;
|
||
}
|
||
|
||
newObjList = (Tcl_Obj**)ckalloc(llen*sizeof(Tcl_Obj*));
|
||
|
||
for (i = 0; i < llen; i++) {
|
||
Tcl_ListObjIndex(NULL, srcPtr, i, &elObj);
|
||
newObjList[i] = Sv_DuplicateObj(elObj);
|
||
}
|
||
|
||
Tcl_SetListObj(copyPtr, llen, newObjList);
|
||
|
||
ckfree((char*)newObjList);
|
||
}
|
||
|
||
/*
|
||
*----------------------------------------------------------------------
|
||
*
|
||
* SvLsetFlat --
|
||
*
|
||
* Almost exact copy from the TclLsetFlat found in tclListObj.c.
|
||
* Simplified in a sense that thread shared objects are guaranteed
|
||
* to be non-shared.
|
||
*
|
||
* Actual return value of this procedure is irrelevant to the caller,
|
||
* and it should be either NULL or non-NULL.
|
||
*
|
||
*----------------------------------------------------------------------
|
||
*/
|
||
|
||
static Tcl_Obj*
|
||
SvLsetFlat(
|
||
Tcl_Interp *interp, /* Tcl interpreter */
|
||
Tcl_Obj *listPtr, /* Pointer to the list being modified */
|
||
int indexCount, /* Number of index args */
|
||
Tcl_Obj **indexArray,
|
||
Tcl_Obj *valuePtr /* Value arg to 'lset' */
|
||
) {
|
||
int elemCount, result, i;
|
||
int index;
|
||
Tcl_Obj **elemPtrs, *chainPtr, *subListPtr;
|
||
|
||
/*
|
||
* Determine whether the index arg designates a list
|
||
* or a single index.
|
||
*/
|
||
|
||
if (indexCount == 1 &&
|
||
Tcl_ListObjGetElements(interp, indexArray[0], &indexCount,
|
||
&indexArray) != TCL_OK) {
|
||
/*
|
||
* Index arg designates something that is neither an index
|
||
* nor a well formed list.
|
||
*/
|
||
|
||
return NULL;
|
||
}
|
||
|
||
/*
|
||
* If there are no indices, then simply return the new value,
|
||
* counting the returned pointer as a reference
|
||
*/
|
||
|
||
if (indexCount == 0) {
|
||
return valuePtr;
|
||
}
|
||
|
||
/*
|
||
* Anchor the linked list of Tcl_Obj's whose string reps must be
|
||
* invalidated if the operation succeeds.
|
||
*/
|
||
|
||
chainPtr = NULL;
|
||
|
||
/*
|
||
* Handle each index arg by diving into the appropriate sublist
|
||
*/
|
||
|
||
for (i = 0; ; ++i) {
|
||
|
||
/*
|
||
* Take the sublist apart.
|
||
*/
|
||
|
||
result = Tcl_ListObjGetElements(interp,listPtr,&elemCount,&elemPtrs);
|
||
if (result != TCL_OK) {
|
||
break;
|
||
}
|
||
|
||
listPtr->internalRep.twoPtrValue.ptr2 = (void*)chainPtr;
|
||
|
||
/*
|
||
* Determine the index of the requested element.
|
||
*/
|
||
|
||
result = Tcl_GetIntForIndex(interp, indexArray[i], elemCount-1, &index);
|
||
if (result != TCL_OK) {
|
||
break;
|
||
}
|
||
|
||
/*
|
||
* Check that the index is in range.
|
||
*/
|
||
|
||
if ((index < 0) || (index >= elemCount)) {
|
||
Tcl_SetObjResult(interp,
|
||
Tcl_NewStringObj("list index out of range", -1));
|
||
result = TCL_ERROR;
|
||
break;
|
||
}
|
||
|
||
/*
|
||
* Break the loop after extracting the innermost sublist
|
||
*/
|
||
|
||
if (i + 1 >= indexCount) {
|
||
result = TCL_OK;
|
||
break;
|
||
}
|
||
|
||
/*
|
||
* Extract the appropriate sublist and chain it onto the linked
|
||
* list of Tcl_Obj's whose string reps must be spoilt.
|
||
*/
|
||
|
||
subListPtr = elemPtrs[index];
|
||
chainPtr = listPtr;
|
||
listPtr = subListPtr;
|
||
}
|
||
|
||
/* Store the result in the list element */
|
||
|
||
if (result == TCL_OK) {
|
||
result = Tcl_ListObjGetElements(interp,listPtr,&elemCount,&elemPtrs);
|
||
if (result == TCL_OK) {
|
||
Tcl_DecrRefCount(elemPtrs[index]);
|
||
elemPtrs[index] = Sv_DuplicateObj(valuePtr);
|
||
Tcl_IncrRefCount(elemPtrs[index]);
|
||
}
|
||
}
|
||
|
||
if (result == TCL_OK) {
|
||
listPtr->internalRep.twoPtrValue.ptr2 = (void*)chainPtr;
|
||
/* Spoil all the string reps */
|
||
while (listPtr != NULL) {
|
||
subListPtr = (Tcl_Obj*)listPtr->internalRep.twoPtrValue.ptr2;
|
||
Tcl_InvalidateStringRep(listPtr);
|
||
listPtr->internalRep.twoPtrValue.ptr2 = NULL;
|
||
listPtr = subListPtr;
|
||
}
|
||
|
||
return valuePtr;
|
||
}
|
||
|
||
return NULL;
|
||
}
|
||
|
||
/* EOF $RCSfile: threadSvListCmd.c,v $ */
|
||
|
||
/* Emacs Setup Variables */
|
||
/* Local Variables: */
|
||
/* mode: C */
|
||
/* indent-tabs-mode: nil */
|
||
/* c-basic-offset: 4 */
|
||
/* End: */
|
||
|