-#!perl -T
+#!perl
use strict;
use warnings;
-use Config qw/%Config/;
+use lib 't/lib';
+use VPIT::TestHelpers (
+ threads => [ 'Variable::Magic' => 'Variable::Magic::VMG_THREADSAFE()' ],
+);
-BEGIN {
- if (!$Config{useithreads}) {
- require Test::More;
- Test::More->import;
- plan(skip_all => 'This perl wasn\'t built to support threads');
- }
-}
+use Test::Leaner 'no_plan';
-use threads; # Before Test::More
-use threads::shared;
+my $destroyed : shared = 0;
-use Test::More;
+sub try {
+ my ($dispell, $op_info) = @_;
+ my $tid = threads->tid;
-use Variable::Magic qw/wizard cast dispell getdata VMG_THREADSAFE VMG_OP_INFO_NAME VMG_OP_INFO_OBJECT/;
+ my $c = 0;
+ my $wiz;
-if (VMG_THREADSAFE) {
- plan tests => 2 * (4 * 18 + 1) + 2 * (4 * 13 + 1);
- my $v = $threads::VERSION;
- diag "Using threads $v" if defined $v;
- $v = $threads::shared::VERSION;
- diag "Using threads::shared $v" if defined $v;
-} else {
- plan skip_all => 'This Variable::Magic isn\'t thread safe';
-}
+ {
+ local $@;
+ eval { require Variable::Magic; 1 } or return;
+ }
-my $destroyed : shared = 0;
-my $sig = undef;
+ {
+ local $@;
+ $wiz = eval {
+ Variable::Magic::wizard(
+ data => sub { $_[1] + $tid },
+ get => sub { ++$c; 0 },
+ set => sub {
+ my $op = $_[-1];
+
+ my $assign_op = ("$]" < 5.037_003) ? 'sassign': 'padsv_store';
+ my $assign_op_cl = ("$]" < 5.037_003) ? 'B::BINOP': 'B::UNOP';
+
+ if ($op_info eq 'object') {
+ is_deeply { class => ref($op), name => $op->name },
+ { class => $assign_op_cl, name => $assign_op },
+ "op object in thread $tid is correct";
+ } else {
+ is $op, $assign_op, "op name in thread $tid is correct";
+ }
+
+ return 0;
+ },
+ free => sub { lock $destroyed; ++$destroyed; 0 },
+ op_info => $op_info eq 'object' ? Variable::Magic::VMG_OP_INFO_OBJECT()
+ : Variable::Magic::VMG_OP_INFO_NAME()
+ );
+ };
+ is $@, '', "wizard in thread $tid doesn't croak";
+ isnt $wiz, undef, "wizard in thread $tid is defined";
+ is $c, 0, "wizard in thread $tid doesn't trigger magic";
+ }
-sub try {
- my ($dispell, $op_info) = @_;
- my $tid = threads->tid();
- my $c = 0;
- my $wiz = eval {
- wizard data => sub { $_[1] + $tid },
- sig => $sig,
- get => sub { ++$c; 0 },
- set => sub {
- my $name = $_[-1];
- $name = $name->name if $op_info == VMG_OP_INFO_OBJECT;
- is $name, 'sassign', "opname for op_info $op_info in thread $tid is correct";
- 0
- },
- free => sub { ++$destroyed; 0 },
- op_info => $op_info
- };
- is($@, '', "wizard in thread $tid doesn't croak");
- isnt($wiz, undef, "wizard in thread $tid is defined");
- is($c, 0, "wizard in thread $tid doesn't trigger magic");
my $a = 3;
- my $res = eval { cast $a, $wiz, sub { 5 }->() };
- is($@, '', "cast in thread $tid doesn't croak");
- is($c, 0, "cast in thread $tid doesn't trigger magic");
- my $b;
- eval { $b = $a };
- is($@, '', "get in thread $tid doesn't croak");
- is($b, 3, "get in thread $tid returns the right thing");
- is($c, 1, "get in thread $tid triggers magic");
- my $d = eval { getdata $a, $wiz };
- 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");
- 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;
+
+ {
+ local $@;
+ my $res = eval { &Variable::Magic::cast(\$a, $wiz, sub { 5 }->()) };
+ is $@, '', "cast in thread $tid doesn't croak";
+ is $c, 0, "cast in thread $tid doesn't trigger magic";
+ }
+
+ {
+ local $@;
+ my $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");
+ is $@, '', "get in thread $tid doesn't croak";
+ is $b, 3, "get in thread $tid returns the right thing";
+ is $c, 1, "get in thread $tid triggers magic";
+ }
+
+ {
+ local $@;
+ my $d = eval { &Variable::Magic::getdata(\$a, $wiz) };
+ 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";
+ }
+
+ {
+ local $@;
+ eval { $a = 9 };
+ is $@, '', "set in thread $tid (check opname) doesn't croak";
+ }
+
+ if ($dispell) {
+ {
+ local $@;
+ my $res = eval { &Variable::Magic::dispell(\$a, $wiz) };
+ is $@, '', "dispell in thread $tid doesn't croak";
+ is $c, 1, "dispell in thread $tid doesn't trigger magic";
+ }
+
+ {
+ local $@;
+ my $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
+
+ return 1;
}
for my $dispell (1, 0) {
- for my $sig (undef, Variable::Magic::gensig()) {
+ {
+ 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, (1 - $dispell) * 4, 'destructors');
+ }
+
+ my $completed = 0;
+
+ my @threads = map spawn(\&try, $dispell, $_), ('name') x 2, ('object') x 2;
+ for my $thr (@threads) {
+ my $res = $thr->join;
+ $completed += $res if defined $res;
+ }
+
+ {
+ lock $destroyed;
+ is $destroyed, (1 - $dispell) * $completed, 'destructors';
}
}