]> git.vpit.fr Git - perl/modules/Variable-Magic.git/blobdiff - t/40-threads.t
Replace $] by "$]"
[perl/modules/Variable-Magic.git] / t / 40-threads.t
index 2215c592aaa3a72e739e4f1f11d08d3ac573ca4c..f67dfc313bf182f6f46229684a667c0f956011dd 100644 (file)
@@ -3,38 +3,64 @@
 use strict;
 use warnings;
 
-use Config qw/%Config/;
+sub skipall {
+ my ($msg) = @_;
+ require Test::More;
+ Test::More::plan(skip_all => $msg);
+}
+
+use Config qw<%Config>;
 
 BEGIN {
- if (!$Config{useithreads}) {
-  require Test::More;
-  Test::More->import;
-  plan(skip_all => 'This Perl wasn\'t built to support threads');
- }
+ my $force = $ENV{PERL_VARIABLE_MAGIC_TEST_THREADS} ? 1 : !1;
+ my $t_v   = $force ? '0' : '1.67';
+ my $ts_v  = $force ? '0' : '1.14';
+ skipall 'This perl wasn\'t built to support threads'
+                                                    unless $Config{useithreads};
+ skipall 'perl 5.13.4 required to test thread safety'
+                                              unless $force or "$]" >= 5.013004;
+ skipall "threads $t_v required to test thread safety"
+                                              unless eval "use threads $t_v; 1";
+ skipall "threads::shared $ts_v required to test thread safety"
+                                     unless eval "use threads::shared $ts_v; 1";
 }
 
-use threads; # Before Test::More
-use threads::shared;
+use Test::More; # after threads
 
-use Test::More;
+use Variable::Magic qw<
+ wizard cast dispell getdata
+ VMG_THREADSAFE VMG_OP_INFO_NAME VMG_OP_INFO_OBJECT
+>;
 
-use Variable::Magic qw/wizard cast dispell getdata VMG_THREADSAFE/;
-
-if (VMG_THREADSAFE) {
- plan tests => 2 * 16 + 1;
-} else {
- plan skip_all => 'This Variable::Magic isn\'t thread safe';
+BEGIN {
+ skipall 'This Variable::Magic isn\'t thread safe' unless VMG_THREADSAFE;
+ plan tests => (4 * 18 + 1) + (4 * 13 + 1);
+ defined and diag "Using threads $_"         for $threads::VERSION;
+ defined and diag "Using threads::shared $_" for $threads::shared::VERSION;
 }
 
 my $destroyed : shared = 0;
 
 sub try {
+ my ($dispell, $op_info) = @_;
  my $tid = threads->tid();
  my $c   = 0;
  my $wiz = eval {
-  wizard get  => sub { ++$c },
-         data => sub { $_[1] + $tid },
-         free => sub { ++$destroyed };
+  wizard data    => sub { $_[1] + $tid },
+         get     => sub { ++$c; 0 },
+         set     => sub {
+                     my $op = $_[-1];
+                     if ($op_info == VMG_OP_INFO_OBJECT) {
+                      is_deeply { class => ref($op),   name => $op->name },
+                                { class => 'B::BINOP', name => 'sassign' },
+                                "op object in thread $tid is correct";
+                     } else {
+                      is $op, 'sassign', "op name in thread $tid is correct";
+                     }
+                     0
+                    },
+         free    => sub { lock $destroyed; ++$destroyed; 0 },
+         op_info => $op_info
  };
  is($@,     '',    "wizard in thread $tid doesn't croak");
  isnt($wiz, undef, "wizard in thread $tid is defined");
@@ -52,19 +78,33 @@ sub try {
  is($@, '',       "getdata in thread $tid doesn't croak");
  is($d, 5 + $tid, "getdata in thread $tid returns the right thing");
  is($c, 1,        "getdata in thread $tid doesn't trigger magic");
- $res = eval { dispell $a, $wiz };
- is($@, '', "dispell in thread $tid doesn't croak");
- is($c, 1,  "dispell in thread $tid doesn't trigger magic");
- undef $b;
- eval { $b = $a };
- is($@, '', "get in thread $tid after dispell doesn't croak");
- is($b, 3,  "get in thread $tid after dispell returns the right thing");
- is($c, 1,  "get in thread $tid after dispell doesn't trigger magic");
- return;
+ eval { $a = 9 };
+ is($@, '', "set in thread $tid (check opname) doesn't croak");
+ if ($dispell) {
+  $res = eval { dispell $a, $wiz };
+  is($@, '', "dispell in thread $tid doesn't croak");
+  is($c, 1,  "dispell in thread $tid doesn't trigger magic");
+  undef $b;
+  eval { $b = $a };
+  is($@, '', "get in thread $tid after dispell doesn't croak");
+  is($b, 9,  "get in thread $tid after dispell returns the right thing");
+  is($c, 1,  "get in thread $tid after dispell doesn't trigger magic");
+ }
+ return; # Ugly if not here
 }
 
-my @t = map { threads->create(\&try) } 1 .. 2;
-$t[0]->join;
-$t[1]->join;
+for my $dispell (1, 0) {
+ {
+  lock $destroyed;
+  $destroyed = 0;
+ }
+
+ my @t = map { threads->create(\&try, $dispell, $_) }
+                              (VMG_OP_INFO_NAME) x 2, (VMG_OP_INFO_OBJECT) x 2;
+ $_->join for @t;
 
-is($destroyed, 0, 'destructors didn\'t fired');
+ {
+  lock $destroyed;
+  is $destroyed, (1 - $dispell) * 4, 'destructors';
+ }
+}