diff options
author | hobbs <hobbs> | 1999-10-29 03:03:59 (GMT) |
---|---|---|
committer | hobbs <hobbs> | 1999-10-29 03:03:59 (GMT) |
commit | 92efc11980dd19d497d47daca03f5082a28a63f4 (patch) | |
tree | 2550a48f32cd8d0e46db002b52b75fa1deff247a /generic/tclStringObj.c | |
parent | bba32831be50692711739fc652d23c4d3e258c9e (diff) | |
download | tcl-92efc11980dd19d497d47daca03f5082a28a63f4.zip tcl-92efc11980dd19d497d47daca03f5082a28a63f4.tar.gz tcl-92efc11980dd19d497d47daca03f5082a28a63f4.tar.bz2 |
* generic/tclStringObj.c: fixed Tcl_AppendResultVA so it only
iterates once over the va_list (avoiding a memcpy of it,
which is not portable).
* generic/tclEnv.c: fixed possible ABR error in environ array
* tests/scan.test:
* generic/tclScan.c: added support for use of inline scan,
XPG3 currently not included
* tests/incr.test:
* tests/set.test:
* generic/tclCompCmds.c: fixed improper bytecode handling of
'eval {set array($unknownvar) 5}' (also for incr)
* win/tclWinTest.c: added testvolumetype command, as atime is
completely ignored for Windows FAT file systems
* win/tclWinPort.h: added sys/utime.h to includes
* unix/tclUnixPort.h: added utime.h to includes
* doc/file.n:
* tests/cmdAH.test:
* generic/tclCmdAH.c: added time arguments to atime and mtime
file command methods (support 'touch' functionality)
Diffstat (limited to 'generic/tclStringObj.c')
-rw-r--r-- | generic/tclStringObj.c | 45 |
1 files changed, 38 insertions, 7 deletions
diff --git a/generic/tclStringObj.c b/generic/tclStringObj.c index bc18fb3..f9c9589 100644 --- a/generic/tclStringObj.c +++ b/generic/tclStringObj.c @@ -33,7 +33,7 @@ * See the file "license.terms" for information on usage and redistribution * of this file, and for a DISCLAIMER OF ALL WARRANTIES. * - * RCS: @(#) $Id: tclStringObj.c,v 1.13 1999/09/02 16:26:34 hobbs Exp $ */ + * RCS: @(#) $Id: tclStringObj.c,v 1.14 1999/10/29 03:04:00 hobbs Exp $ */ #include "tclInt.h" @@ -1170,10 +1170,14 @@ Tcl_AppendStringsToObjVA (objPtr, argList) Tcl_Obj *objPtr; /* Points to the object to append to. */ va_list argList; /* Variable argument list. */ { +#define STATIC_LIST_SIZE 16 String *stringPtr; - va_list tmpArgList; int newLength, oldLength; register char *string, *dst; + char *static_list[STATIC_LIST_SIZE]; + char **args = static_list; + int nargs_space = STATIC_LIST_SIZE; + int nargs, i; if (Tcl_IsShared(objPtr)) { panic("Tcl_AppendStringsToObj called with shared object"); @@ -1188,17 +1192,33 @@ Tcl_AppendStringsToObjVA (objPtr, argList) * (notably OS/390) the argList is an array so we need to use memcpy. */ - memcpy ((VOID *) &tmpArgList, (VOID *) &argList, sizeof (tmpArgList)); + nargs = 0; newLength = oldLength = objPtr->length; while (1) { - string = va_arg(tmpArgList, char *); + string = va_arg(argList, char *); if (string == NULL) { break; } + if (nargs >= nargs_space) { + /* + * Expand the args buffer + */ + nargs_space += STATIC_LIST_SIZE; + if (args == static_list) { + args = (void *)ckalloc(nargs_space * sizeof(char *)); + for (i = 0; i < nargs; ++i) { + args[i] = static_list[i]; + } + } else { + args = (void *)ckrealloc((void *)args, + nargs_space * sizeof(char *)); + } + } newLength += strlen(string); + args[nargs++] = string; } if (newLength == oldLength) { - return; + goto done; } stringPtr = GET_STRING(objPtr); @@ -1222,8 +1242,8 @@ Tcl_AppendStringsToObjVA (objPtr, argList) */ dst = objPtr->bytes + oldLength; - while (1) { - string = va_arg(argList, char *); + for (i = 0; i < nargs; ++i) { + string = args[i]; if (string == NULL) { break; } @@ -1245,6 +1265,17 @@ Tcl_AppendStringsToObjVA (objPtr, argList) *dst = 0; } objPtr->length = newLength; + + done: + /* + * If we had to allocate a buffer from the heap, + * free it now. + */ + + if (args != static_list) { + ckfree((void *)args); + } +#undef STATIC_LIST_SIZE } /* |