summaryrefslogtreecommitdiff
path: root/op.c
diff options
context:
space:
mode:
authorDave Mitchell <davem@fdisolutions.com>2006-12-13 12:47:35 +0000
committerDave Mitchell <davem@fdisolutions.com>2006-12-13 12:47:35 +0000
commitd2c837a0af0259b12bbac41550ec7dc972b876d9 (patch)
tree73ecb539088be198671a9fb94bc9ab8b21a41b2f /op.c
parent7b75a55b6ed321d45dd8a2bb65f2bb2a846904d2 (diff)
downloadperl-d2c837a0af0259b12bbac41550ec7dc972b876d9.tar.gz
fix double free introduced by #29543 (spotted by Nicholas)
p4raw-id: //depot/perl@29547
Diffstat (limited to 'op.c')
-rw-r--r--op.c20
1 files changed, 17 insertions, 3 deletions
diff --git a/op.c b/op.c
index 1a3baa34c5..cacad6ee0b 100644
--- a/op.c
+++ b/op.c
@@ -277,6 +277,20 @@ Perl_allocmy(pTHX_ const char *const name)
return off;
}
+/* free the body of an op without examining its contents.
+ * Always use this rather than FreeOp directly */
+
+void
+S_op_destroy(pTHX_ OP *o)
+{
+ if (o->op_latefree) {
+ o->op_latefreed = 1;
+ return;
+ }
+ FreeOp(o);
+}
+
+
/* Destructor */
void
@@ -2044,7 +2058,7 @@ Perl_newPROG(pTHX_ OP *o)
if (o->op_type == OP_STUB) {
PL_comppad_name = 0;
PL_compcv = 0;
- FreeOp(o);
+ S_op_destroy(aTHX_ o);
return;
}
PL_main_root = scope(sawparens(scalarvoid(o)));
@@ -2386,7 +2400,7 @@ Perl_append_list(pTHX_ I32 type, LISTOP *first, LISTOP *last)
last->op_madprop = 0;
#endif
- FreeOp(last);
+ S_op_destroy(aTHX_ (OP*)last);
return (OP*)first;
}
@@ -4579,7 +4593,7 @@ Perl_newFOROP(pTHX_ I32 flags, char *label, line_t forline, OP *sv, OP *expr, OP
LOOP *tmp;
NewOp(1234,tmp,1,LOOP);
Copy(loop,tmp,1,LISTOP);
- FreeOp(loop);
+ S_op_destroy(aTHX_ loop);
loop = tmp;
}
#else