X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FDevel%2FGlobalDestruction.pm;h=9030ae7e63754857b3e02f49f5ef3896df7b6b10;hb=19a66cb4bd2afb3036d4ad05c6e30713023bc107;hp=8cf559c61febd481f5dfa816bce4f3f4e64e14ea;hpb=aaa7f60fc32505c436095fb1e5a91b8bc11f3e3a;p=p5sagit%2FDevel-GlobalDestruction.git diff --git a/lib/Devel/GlobalDestruction.pm b/lib/Devel/GlobalDestruction.pm index 8cf559c..9030ae7 100644 --- a/lib/Devel/GlobalDestruction.pm +++ b/lib/Devel/GlobalDestruction.pm @@ -1,31 +1,91 @@ -#!/usr/bin/perl - package Devel::GlobalDestruction; use strict; use warnings; -use XSLoader; - -our $VERSION = '0.04'; +our $VERSION = '0.06'; use Sub::Exporter -setup => { - exports => [ qw(in_global_destruction) ], - groups => { default => [ -all ] }, + 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] }'; } -else { +# try to load the xs version if it was compiled +# +elsif (eval { + require XSLoader; XSLoader::load(__PACKAGE__, $VERSION); + 1; +}) { + # the eval already installed everything, nothing to do } +# Not core nor XS +# +else { -__PACKAGE__ + # 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; -__END__ + eval <<'PP_IGD' or die $@; + +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__ ]}"); + } +} -=pod +1; # keep eval happy + +PP_IGD + +} + +1; # keep require happy + + +__END__ =head1 NAME @@ -34,16 +94,16 @@ destruction. =head1 SYNOPSIS - package Foo; - use Devel::GlobalDestruction; + package Foo; + use Devel::GlobalDestruction; - use namespace::clean; # to avoid having an "in_global_destruction" method + use namespace::clean; # to avoid having an "in_global_destruction" method - sub DESTROY { - return if in_global_destruction; + sub DESTROY { + return if in_global_destruction; - do_something_a_little_tricky(); - } + do_something_a_little_tricky(); + } =head1 DESCRIPTION @@ -71,12 +131,6 @@ current value of C. =back -=head1 VERSION CONTROL - -This module is maintained using Darcs. You can get the latest version from -L, and use C to commit -changes. - =head1 AUTHORS Yuval Kogman Enothingmuch@woobling.orgE @@ -85,12 +139,12 @@ Florian Ragwitz Erafl@debian.orgE Jesse Luehrs Edoy@tozt.netE +Peter Rabbitson Eribasushi@cpan.orgE + =head1 COPYRIGHT - Copyright (c) 2008 Yuval Kogman. All rights reserved - This program is free software; you can redistribute - it and/or modify it under the same terms as Perl itself. + Copyright (c) 2008 Yuval Kogman. All rights reserved + This program is free software; you can redistribute + it and/or modify it under the same terms as Perl itself. =cut - -