X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FDevel%2FGlobalDestruction.pm;h=36a1feb73bb33f4bd076cbe9ea68523b77bb877f;hb=b1bee216b13ba067c620587ce49899927e103970;hp=99e3eec071331e9f3ec48619471165a9176622ac;hpb=53daa8381ac1962b80e5becbffc1d1a2017c9df5;p=p5sagit%2FDevel-GlobalDestruction.git diff --git a/lib/Devel/GlobalDestruction.pm b/lib/Devel/GlobalDestruction.pm index 99e3eec..36a1feb 100644 --- a/lib/Devel/GlobalDestruction.pm +++ b/lib/Devel/GlobalDestruction.pm @@ -3,7 +3,7 @@ package Devel::GlobalDestruction; use strict; use warnings; -our $VERSION = '0.08'; +our $VERSION = '0.09'; use Sub::Exporter::Progressive -setup => { exports => [ qw(in_global_destruction) ], @@ -13,75 +13,30 @@ use Sub::Exporter::Progressive -setup => { # we run 5.14+ - everything is in core # if (defined ${^GLOBAL_PHASE}) { - eval 'sub in_global_destruction () { ${^GLOBAL_PHASE} eq q[DESTRUCT] }'; + eval 'sub in_global_destruction () { ${^GLOBAL_PHASE} eq q[DESTRUCT] }; 1' + or die $@; } # try to load the xs version if it was compiled # elsif (eval { - require XSLoader; - XSLoader::load(__PACKAGE__, $VERSION); + require Devel::GlobalDestruction::XS; + *in_global_destruction = \&Devel::GlobalDestruction::XS::in_global_destruction; 1; }) { # the eval already installed everything, nothing to do } -# Not core nor XS -# The whole thing is in an eval to prevent perl from parsing it in the -# first place under perls where none of this is needed -# else { - eval <<'PP_IGD' or die $@; - -# SpeedyCGI runs END blocks every cycle but somehow keeps object instances -# hence DIAF -die("The pure-perl version of @{[__PACKAGE__]} can not function correctly under CGI::SpeedyCGI. " - . "Please ensure you have a working compiler, and reinstall @{[__PACKAGE__]} to enable the XS " - . "codepath.\n" -) if $CGI::SpeedyCGI::i_am_speedy; - - -my ($in_global_destruction, $before_is_installed); - -sub in_global_destruction () { $in_global_destruction } - -# This block will fire towards the end of the program execution -# Since there is no way for us to generate an END which will execute *last* -# this is *NOT 100% INCOMPATIBLE* with XS/${^GLOBAL_PHASE}. We *may* end up -# with a true in_gloal_destruction() in the middle of another END block -# There are no practical cases where this matters. -# -END { - $in_global_destruction = 1; -} - -# threads do not execute the global ENDs (it would be stupid). However -# one can register a new END via simple string eval within a thread, and -# achieve the same result. A logical place to do this would be CLONE, which -# is claimed to run in the context of the new thread. However this does -# not really seem to be the case - any END evaled in a CLONE is ignored :( -# Hence blatantly hooking threads::create -# -if ($INC{'threads.pm'}) { - my $orig_create = threads->can('create'); - no warnings 'redefine'; - *threads::create = sub { - { local $@; eval 'END { $in_global_destruction = 1 }' }; - goto $orig_create; - }; - $before_is_installed = 1; -} - -# just in case threads got loaded after us (silly) -sub CLONE { - unless ($before_is_installed) { - require Carp; - Carp::croak("You must load the 'threads' module before @{[ __PACKAGE__ ]}"); + # internally, PL_main_start is nulled immediately before entering global destruction + # and we can use B to detect that. It will also be null before the main runloop starts, + # so we check install a CHECK if needed to detect that. + require B; + my $started = !B::main_start()->isa(q[B::NULL]); + unless ($started) { + eval 'CHECK { $started = 1 }; 1' + or die $@; } -} - -1; # keep eval happy - -PP_IGD - + eval 'sub in_global_destruction () { $started && B::main_start()->isa(q[B::NULL]) }; 1' + or die $@; } 1; # keep require happy @@ -129,8 +84,8 @@ aliased, etc. if L is present. =item in_global_destruction Returns true if the interpreter is in global destruction. In perl 5.14+, this -returns C<${^GLOBAL_PHASE} eq 'DESTRUCT'>, and on earlier perls, it returns the -current value of C. +returns C<${^GLOBAL_PHASE} eq 'DESTRUCT'>, and on earlier perls, detects it using +the value of C or C. =back @@ -146,6 +101,10 @@ Peter Rabbitson Eribasushi@cpan.orgE Arthur Axel 'fREW' Schmidt Efrioux@gmail.comE +Elizabeth Mattijsen Eliz@dijkmat.nlE + +Greham Knop Ehaarg@haarg.orgE + =head1 COPYRIGHT Copyright (c) 2008 Yuval Kogman. All rights reserved