summaryrefslogtreecommitdiff
path: root/ext/Opcode
diff options
context:
space:
mode:
authorDave Mitchell <davem@fdisolutions.com>2003-09-07 20:14:44 +0100
committerRafael Garcia-Suarez <rgarciasuarez@gmail.com>2003-09-07 18:25:23 +0000
commit35ed0d3c8f1120489361bf37c4e66472d2262576 (patch)
tree3736a733510d2e163ff9627d2b84fbd219bd8a1b /ext/Opcode
parent4c90a460cdb7d8d7e800ddcf78d3a123f45708a6 (diff)
downloadperl-35ed0d3c8f1120489361bf37c4e66472d2262576.tar.gz
Re: [perl #23656] Safe reval bleeds local variable values
Message-ID: <20030907181444.GA7058@fdgroup.com> p4raw-id: //depot/perl@21063
Diffstat (limited to 'ext/Opcode')
-rw-r--r--ext/Opcode/Safe.pm32
1 files changed, 22 insertions, 10 deletions
diff --git a/ext/Opcode/Safe.pm b/ext/Opcode/Safe.pm
index b090e4078d..5036943cd7 100644
--- a/ext/Opcode/Safe.pm
+++ b/ext/Opcode/Safe.pm
@@ -3,7 +3,27 @@ package Safe;
use 5.003_11;
use strict;
-$Safe::VERSION = "2.09";
+$Safe::VERSION = "2.10";
+
+# *** Don't declare any lexicals above this point ***
+#
+# This function should return a closure which contains an eval that can't
+# see any lexicals in scope (apart from __ExPr__ which is unavoidable)
+
+sub lexless_anon_sub {
+ # $_[0] is package;
+ # $_[1] is strict flag;
+ my $__ExPr__ = $_[2]; # must be a lexical to create the closure that
+ # can be used to pass the value into the safe
+ # world
+
+ # Create anon sub ref in root of compartment.
+ # Uses a closure (on $__ExPr__) to pass in the code to be executed.
+ # (eval on one line to keep line numbers as expected by caller)
+ eval sprintf
+ 'package %s; %s strict; sub { @_=(); eval q[my $__ExPr__;] . $__ExPr__; }',
+ $_[0], $_[1] ? 'use' : 'no';
+}
use Carp;
@@ -211,15 +231,7 @@ sub reval {
my ($obj, $expr, $strict) = @_;
my $root = $obj->{Root};
- # Create anon sub ref in root of compartment.
- # Uses a closure (on $expr) to pass in the code to be executed.
- # (eval on one line to keep line numbers as expected by caller)
- my $evalcode = sprintf('package %s; sub { @_ = (); eval $expr; }', $root);
- my $evalsub;
-
- if ($strict) { use strict; $evalsub = eval $evalcode; }
- else { no strict; $evalsub = eval $evalcode; }
-
+ my $evalsub = lexless_anon_sub($root,$strict, $expr);
return Opcode::_safe_call_sv($root, $obj->{Mask}, $evalsub);
}