OSDN Git Service

Please enter the commit message for your changes. Lines starting
[eos/base.git] / util / src / TclTk / tcl8.6.12 / pkgs / thread2.8.7 / generic / threadSvListCmd.c
diff --git a/util/src/TclTk/tcl8.6.12/pkgs/thread2.8.7/generic/threadSvListCmd.c b/util/src/TclTk/tcl8.6.12/pkgs/thread2.8.7/generic/threadSvListCmd.c
new file mode 100644 (file)
index 0000000..182a53c
--- /dev/null
@@ -0,0 +1,1080 @@
+/*
+ * 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);
+
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+    }
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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;
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *-----------------------------------------------------------------------------
+ *
+ * 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);
+}
+\f
+/*
+ *----------------------------------------------------------------------
+ *
+ * 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:                  */
+