From: Daisuke Maki Date: Wed, 3 Dec 2008 08:14:28 +0000 (+0000) Subject: 024 fails, but successfully loads and uses the XS versions of Scalar::Util functions X-Git-Tag: 0.19~136^2~55 X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?p=gitmo%2FMouse.git;a=commitdiff_plain;h=eae8075956ba01581ea7488b4ddb2506db1111da;hp=11ac534bdfe4aab1f8bfb575769dee68f456c1d9 024 fails, but successfully loads and uses the XS versions of Scalar::Util functions --- diff --git a/lib/Mouse.pm b/lib/Mouse.pm index b8e8f97..708b0a5 100644 --- a/lib/Mouse.pm +++ b/lib/Mouse.pm @@ -1,31 +1,35 @@ -#!/usr/bin/env perl + package Mouse; use strict; use warnings; +use 5.006; use base 'Exporter'; -our $VERSION = '0.12'; -use 5.006; +our $VERSION; +our $PurePerl; -if ($ENV{SHIKA_DEBUG}) { - *DEBUG = sub (){ 1 }; -} else { - *DEBUG = sub (){ 0 }; -} +BEGIN { + $VERSION = '0.12'; -our $PurePerl; -$PurePerl = $ENV{SHIKA_PUREPERL} unless defined $PurePerl; + if ($ENV{MOUSE_DEBUG}) { + *DEBUG = sub (){ 1 }; + } else { + *DEBUG = sub (){ 0 }; + } + if (! defined $PurePerl && $ENV{MOUSE_PUREPERL} && $ENV{MOUSE_PUREPERL} =~ /^(.+)$/) { + $PurePerl = $1; + } -if (! $PurePerl) { - local $@; - local $^W = 0; - require XSLoader; - $PurePerl = !eval{ XSLoader::load(__PACKAGE__, $VERSION); 1 }; - warn "Failed to load XS mode: $@" if $@ && Mouse::DEBUG(); + if (! $PurePerl) { + local $@; + local $^W = 0; + require XSLoader; + $PurePerl = ! eval{ XSLoader::load(__PACKAGE__, $VERSION); 1 }; + warn "Failed to load XS mode: $@" if $@; # && Mouse::DEBUG(); + } } - use Carp 'confess'; use Mouse::Util 'blessed'; @@ -110,7 +114,7 @@ sub import { no warnings 'redefine'; *{$caller.'::meta'} = sub { $meta }; - Mouse->export_to_level(1, @_); + __PACKAGE__->export_to_level( 1, @_); } sub unimport { diff --git a/lib/Mouse/Util.pm b/lib/Mouse/Util.pm index 476075d..4c194fd 100644 --- a/lib/Mouse/Util.pm +++ b/lib/Mouse/Util.pm @@ -2,13 +2,97 @@ package Mouse::Util; use strict; use warnings; -use base 'Exporter'; +use Exporter 'import'; use Carp; +our @EXPORT_OK = qw( + blessed + get_linear_isa + looks_like_number + openhandle + reftype + weaken +); +our %EXPORT_TAGS = ( + all => \@EXPORT_OK, +); + +# We only have to do this nastiness if we haven't loaded XS version of +# Mouse.pm, so check if we're running under PurePerl or not BEGIN { - our %dependencies = ( - 'Scalar::Util' => { + if ($Mouse::PurePerl) { + _install_pp_func(); + } else { + # If we're running under XS, we can provide + # blessed + # looks_like_number + # reftype + # weaken + # other functions need to be loaded from our respective sources + + if (defined &Scalar::Util::openhandle) { + *openhandle = \&Scalar::Util::openhandle; + } else { + # XXX - room for improvement + *openhandle = sub { + local($@, $SIG{__DIE__}, $SIG{__WARN__}); + my $r = shift; + my $t; + + length($t = ref($r)) or return undef; + + # This eval will fail if the reference is not blessed + eval { $r->a_sub_not_likely_to_be_here; 1 } + ? do { + $t = eval { + # we have a GLOB or an IO. Stringify a GLOB gives it's name + my $q = *$r; + $q =~ /^\*/ ? "GLOB" : "IO"; + } + or do { + # OK, if we don't have a GLOB what parts of + # a glob will it populate. + # NOTE: A glob always has a SCALAR + local *glob = $r; + defined *glob{ARRAY} && "ARRAY" + or defined *glob{HASH} && "HASH" + or defined *glob{CODE} && "CODE" + or length(ref(${$r})) ? "REF" : "SCALAR"; + } + } + : $t + }; + } + if (defined &mro::get_linear_isa) { + *get_linear_isa = \&mro::get_linear_isa; + } else { + # this recurses so it isn't pretty + my $code; + *get_linear_isa = $code = sub { + no strict 'refs'; + + my $classname = shift; + + my @lin = ($classname); + my %stored; + foreach my $parent (@{"$classname\::ISA"}) { + my $plin = $code->($parent); + foreach (@$plin) { + next if exists $stored{$_}; + push(@lin, $_); + $stored{$_} = 1; + } + } + return \@lin; + }; + } + } +} + +sub _install_pp_func { + my %dependencies = ( + 'Scalar::Util' => { # VVVVV CODE TAKEN FROM SCALAR::UTIL VVVVV 'blessed' => do { *UNIVERSAL::a_sub_not_likely_to_be_here = sub { @@ -123,13 +207,7 @@ BEGIN { }, ); - our %loaded; - - our @EXPORT_OK = map { keys %$_ } values %dependencies; - our %EXPORT_TAGS = ( - all => \@EXPORT_OK, - ); - + my %loaded; for my $module_name (keys %dependencies) { my $loaded = do { local $SIG{__DIE__} = 'DEFAULT';