summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorapnadkarni <apnmbx-wits@yahoo.com>2023-07-12 02:51:50 (GMT)
committerapnadkarni <apnmbx-wits@yahoo.com>2023-07-12 02:51:50 (GMT)
commit6c4636e94b19b9b0d5972e4e76048e3a94351f82 (patch)
tree6d95db4ef0e11ac90efa0c06cb08b44d0f5a7c54
parentcd97018033e773481b8a5b71645a3560e14daca6 (diff)
parent4ee94e4f670279c82c6af59820e2d7d82c774ca5 (diff)
downloadtcl-6c4636e94b19b9b0d5972e4e76048e3a94351f82.zip
tcl-6c4636e94b19b9b0d5972e4e76048e3a94351f82.tar.gz
tcl-6c4636e94b19b9b0d5972e4e76048e3a94351f82.tar.bz2
Fix [54329e39c7]
-rw-r--r--generic/tclExecute.c29
-rw-r--r--tests/lseq.test21
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