-
Notifications
You must be signed in to change notification settings - Fork 561
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Some closures leak memory in v5.40 #22547
Comments
Bisect says: 386907f is the first bad commit
|
Confirmed. It went into blead on Sep 25.
@tonycoz can you take a look? Thanks. |
The script shown here detects a leak on perls before a0d2bbd (5.16). |
If we can fix, it should be backported to perl-5.40.1. |
Yes, my change reverted this to pre-5.18 behaviour, since a0d2bbd broke eval in package DB, which is a documented feature. I asked for guidance on the list and received no feedback that could resolve the issue. The only feedback I received on the PR #21489 was to apply the reversion and consider mitigations, of which I didn't see any Note that if your closure contains a string eval you'll still see the reference loop. |
I've been thinking about this. I now think your example: use strict;
package DB { sub myeval { eval shift or die } }
my $y = 1; # something true so eval in myeval above is true
my $has_outer = sub { $y; my $x; sub { DB::myeval '$y'; eval ""; ++$x; } }->();
$has_outer->(); # has an outside pointer and can resolve $y
my $no_outer = sub { $y; my $x; sub { DB::myeval '$y'; ++$x; } }->();
$no_outer->(); # dies since we can't see $y ... should die in either case, no matter if there is an Rationale: A closure should only capture variables whose names appear lexically in the sub's body. Capturing more is not just inefficient (using more memory than expected), but can introduce leaks due to hidden reference cycles. Runtime code from It is nice if attempts to access such a variable at runtime work anyway or trigger a use strict;
use warnings;
sub f {
my $x;
sub { eval '$x' }
}
f()->();
# Variable "$x" is not available at (eval 1) line 1. But I'm totally fine with getting an undeclared variable error from either static (lexically present in the inner sub) or dynamic (via a call to a sub in package |
What if we add a |
Then your code behaves differently in terms of references in the debugger, which makes the debugger not fit for purpose. |
The same mechanism is used to search for lexical subs, so outer lexical subs would also fail to be found in an eval. |
Given the choice between the 5.38 and 5.40 behaviors, I think I would prefer to revert to 5.38. If a closure doesn't capture a lexical, it can't expect it to be available. And this includes lexical subs. Implicitly capturing everything, leading to reference loops and lifetime issues, for the sake of the debugger seems like the wrong approach. And given how long the behavior existed before there were any complaints, it seems like it has a smaller impact. If there was an option to use weak references, as implied by #21489 (comment), that would probably be preferable to either of the behaviors that have been implemented so far. |
This reverts commit 386907f. Reinstates the behaviour of CV outside references from 5.38, fixing Perl#22547 Breaks Perl#19370
This reverts commit 386907f. Reinstates the behaviour of CV outside references from 5.38, fixing Perl#22547 Breaks Perl#19370
Sorry for the delay, I've added the extra tests I wanted to #22635 |
Description
Starting in v5.40, some closures leak memory despite there being no reference cycles.
Steps to Reproduce
This used to work in v5.38. It is broken in v5.40 and blead.
Expected behavior
The text was updated successfully, but these errors were encountered: