X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FDevel%2FGlobalDestruction.pm;h=25b895881e2cd4f7a12ce3e62f89283c53ea7962;hb=23135b02f4a06a25ad3626ca40d8c171b379f25f;hp=cf94048e2f3f5af04d28fb3a64a6381d6d2a3fdf;hpb=0e762adeb08fb1fc9534c0e629e269d18951791a;p=p5sagit%2FDevel-GlobalDestruction.git diff --git a/lib/Devel/GlobalDestruction.pm b/lib/Devel/GlobalDestruction.pm index cf94048..25b8958 100644 --- a/lib/Devel/GlobalDestruction.pm +++ b/lib/Devel/GlobalDestruction.pm @@ -3,65 +3,34 @@ package Devel::GlobalDestruction; use strict; use warnings; -our $VERSION = '0.04'; +our $VERSION = '0.09'; -use Sub::Exporter -setup => { - exports => [ qw(in_global_destruction) ], - groups => { default => [ -all ] }, +use Sub::Exporter::Progressive -setup => { + exports => [ qw(in_global_destruction) ], + groups => { default => [ -all ] }, }; +# 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); - 1; + require XSLoader; + XSLoader::load(__PACKAGE__, $VERSION); + 1; }) { - # the eval already installed everything, nothing to do + # the eval already installed everything, nothing to do } else { - eval <<'PP_IGD' or die $@; - -my ($in_global_destruction, $before_is_installed); - -sub in_global_destruction { $in_global_destruction } - -END { - # SpeedyCGI runs END blocks every cycle but somehow keeps object instances - # hence lying about it seems reasonable...ish - $in_global_destruction = 1 unless $CGI::SpeedyCGI::i_am_speedy; -} - -# 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__ ]}"); - } -} - -1; # keep eval happy - -PP_IGD - + # internally, PL_main_start is nulled immediately before entering global destruction + # and we can use B to detect that + require B; + eval 'sub in_global_destruction () { B::main_start()->isa(q[B::NULL]) }; 1' + or die $@; } 1; # keep require happy @@ -101,7 +70,8 @@ destruction is in effect. =head1 EXPORTS -This module uses L so the exports may be renamed, aliased, etc. +This module uses L so the exports may be renamed, +aliased, etc. if L is present. =over 4 @@ -123,6 +93,10 @@ Jesse Luehrs Edoy@tozt.netE Peter Rabbitson Eribasushi@cpan.orgE +Arthur Axel 'fREW' Schmidt Efrioux@gmail.comE + +Elizabeth Mattijsen Eliz@dijkmat.nlE + =head1 COPYRIGHT Copyright (c) 2008 Yuval Kogman. All rights reserved