diff options
| author | apnadkarni <apnmbx-wits@yahoo.com> | 2023-07-12 02:51:50 (GMT) |
|---|---|---|
| committer | apnadkarni <apnmbx-wits@yahoo.com> | 2023-07-12 02:51:50 (GMT) |
| commit | 6c4636e94b19b9b0d5972e4e76048e3a94351f82 (patch) | |
| tree | 6d95db4ef0e11ac90efa0c06cb08b44d0f5a7c54 | |
| parent | cd97018033e773481b8a5b71645a3560e14daca6 (diff) | |
| parent | 4ee94e4f670279c82c6af59820e2d7d82c774ca5 (diff) | |
| download | tcl-6c4636e94b19b9b0d5972e4e76048e3a94351f82.zip tcl-6c4636e94b19b9b0d5972e4e76048e3a94351f82.tar.gz tcl-6c4636e94b19b9b0d5972e4e76048e3a94351f82.tar.bz2 | |
Fix [54329e39c7]
| -rw-r--r-- | generic/tclExecute.c | 29 | ||||
| -rw-r--r-- | tests/lseq.test | 21 |
2 files changed, 47 insertions, 3 deletions
diff --git a/generic/tclExecute.c b/generic/tclExecute.c index 120dac6..973ba8e 100644 --- a/generic/tclExecute.c +++ b/generic/tclExecute.c @@ -6537,11 +6537,18 @@ TEBCresume( for (i = 0; i < numLists; i++) { varListPtr = infoPtr->varLists[i]; numVars = varListPtr->numVars; + int hasAbstractList; listPtr = OBJ_AT_DEPTH(listTmpDepth); + hasAbstractList = TclObjTypeHasProc(listPtr, indexProc) != 0; DECACHE_STACK_INFO(); - status = TclListObjGetElementsM( - interp, listPtr, &listLen, &elements); + if (hasAbstractList) { + status = Tcl_ListObjLength(interp, listPtr, &listLen); + elements = NULL; + } else { + status = TclListObjGetElementsM( + interp, listPtr, &listLen, &elements); + } if (status != TCL_OK) { CACHE_STACK_INFO(); goto gotError; @@ -6554,7 +6561,23 @@ TEBCresume( if (valIndex >= listLen) { TclNewObj(valuePtr); } else { - valuePtr = elements[valIndex]; + DECACHE_STACK_INFO(); + if (elements) { + valuePtr = elements[valIndex]; + } else { + status = Tcl_ListObjIndex( + interp, listPtr, valIndex, &valuePtr); + if (status != TCL_OK) { + /* Could happen for abstract lists */ + CACHE_STACK_INFO(); + goto gotError; + } + if (valuePtr == NULL) { + /* Permitted for Tcl_LOI to return NULL */ + TclNewObj(valuePtr); + } + } + CACHE_STACK_INFO(); } varIndex = varListPtr->varIndexes[j]; diff --git a/tests/lseq.test b/tests/lseq.test index 6ec9bb2..a149908 100644 --- a/tests/lseq.test +++ b/tests/lseq.test @@ -21,6 +21,16 @@ testConstraint knownBug 0 testConstraint has64BitLengths [expr {$tcl_platform(pointerSize) == 8}] testConstraint has32BitLengths [expr {$tcl_platform(pointerSize) == 4}] +proc memusage {} { + set fd [open /proc/[pid]/statm] + set line [gets $fd] + if {[llength $line] != 7} { + error "Unexpected /proc/pid/statm format" + } + return [lindex $line 5] +} +testConstraint hasMemUsage [expr {![catch {memusage}]}] + # Arg errors test lseq-1.1 {error cases} -body { lseq @@ -707,6 +717,17 @@ test lseq-convertToList {does not result in a memory error} -body { list [catch {set var1 [lindex [lreplace [lseq 1 2] 1 1 hello] 0]} cres] $cres } -cleanup {unset var1 cres} -result {1 {can't set "var1": this is an error}} +test lseq-bug-54329e39c7 {does not cause memory bloat} -constraints { + hasMemUsage +} -body { + set l [lseq 1000000] + proc p l {foreach x $l {}} + set premem [memusage] + p $l + set postmem [memusage] + expr {($postmem - $premem) < 10} +} -result 1 + # cleanup ::tcltest::cleanupTests |
