threads 1.66
[p5sagit/p5-mst-13.2.git] / ext / threads / t / thread.t
CommitLineData
0f1612a7 1use strict;
2use warnings;
f9dff5f5 3
4BEGIN {
0f1612a7 5 if ($ENV{'PERL_CORE'}){
6 chdir 't';
7 unshift @INC, '../lib';
8 }
9 use Config;
fc04eb16 10 if (! $Config{'useithreads'}) {
11 print("1..0 # Skip: Perl not compiled with 'useithreads'\n");
12 exit(0);
f9dff5f5 13 }
0f1612a7 14
15 require($ENV{PERL_CORE} ? "./test.pl" : "./t/test.pl");
f9dff5f5 16}
17
18use ExtUtils::testlib;
0f1612a7 19
58a3a76c 20use threads;
21
fc04eb16 22BEGIN {
58a3a76c 23 eval {
24 require threads::shared;
f3086ff0 25 threads::shared->import();
58a3a76c 26 };
27 if ($@ || ! $threads::shared::threads_shared) {
28 print("1..0 # Skip: threads::shared not available\n");
29 exit(0);
30 }
31
fc04eb16 32 $| = 1;
863e9b4a 33 print("1..34\n"); ### Number of tests that will be run ###
fc04eb16 34};
35
fc04eb16 36print("ok 1 - Loaded\n");
f9dff5f5 37
fc04eb16 38### Start of Testing ###
f9dff5f5 39
40sub content {
41 print shift;
42 return shift;
43}
44{
f4cc38af 45 my $t = threads->create(\&content, "ok 2\n", "ok 3\n", 1..1000);
f9dff5f5 46 print $t->join();
47}
48{
49 my $lock : shared;
50 my $t;
51 {
fc04eb16 52 lock($lock);
53 $t = threads->create(sub { lock($lock); print "ok 5\n"});
54 print "ok 4\n";
f9dff5f5 55 }
56 $t->join();
57}
58
59sub dorecurse {
60 my $val = shift;
61 my $ret;
74bf223e 62 print $val;
f9dff5f5 63 if(@_) {
fc04eb16 64 $ret = threads->create(\&dorecurse, @_);
65 $ret->join;
f9dff5f5 66 }
67}
68{
f4cc38af 69 my $t = threads->create(\&dorecurse, map { "ok $_\n" } 6..10);
74bf223e 70 $t->join();
f9dff5f5 71}
72
73{
74 # test that sleep lets other thread run
f4cc38af 75 my $t = threads->create(\&dorecurse, "ok 11\n");
da32f63e 76 threads->yield; # help out non-preemptive thread implementations
f9dff5f5 77 sleep 1;
74bf223e 78 print "ok 12\n";
79 $t->join();
f9dff5f5 80}
81{
82 my $lock : shared;
83 sub islocked {
fc04eb16 84 lock($lock);
85 my $val = shift;
86 my $ret;
87 print $val;
88 if (@_) {
89 $ret = threads->create(\&islocked, shift);
90 }
91 return $ret;
f9dff5f5 92 }
f4cc38af 93my $t = threads->create(\&islocked, "ok 13\n", "ok 14\n");
f9dff5f5 94$t->join->join;
95}
96
97
98
99sub testsprintf {
100 my $testno = shift;
101 my $same = sprintf( "%0.f", $testno);
8abd20a8 102 return $testno eq $same;
f9dff5f5 103}
104
105sub threaded {
8abd20a8 106 my ($string, $string_end) = @_;
f9dff5f5 107
108 # Do the match, saving the output in appropriate variables
109 $string =~ /(.*)(is)(.*)/;
110 # Yield control, allowing the other thread to fill in the match variables
111 threads->yield();
112 # Examine the match variable contents; on broken perls this fails
8abd20a8 113 return $3 eq $string_end;
f9dff5f5 114}
115
116
117{
74bf223e 118 curr_test(15);
8abd20a8 119
f4cc38af 120 my $thr1 = threads->create(\&testsprintf, 15);
121 my $thr2 = threads->create(\&testsprintf, 16);
f9dff5f5 122
123 my $short = "This is a long string that goes on and on.";
124 my $shorte = " a long string that goes on and on.";
125 my $long = "This is short.";
126 my $longe = " short.";
127 my $foo = "This is bar bar bar.";
128 my $fooe = " bar bar bar.";
8abd20a8 129 my $thr3 = new threads \&threaded, $short, $shorte;
130 my $thr4 = new threads \&threaded, $long, $longe;
74bf223e 131 my $thr5 = new threads \&testsprintf, 19;
132 my $thr6 = new threads \&testsprintf, 20;
8abd20a8 133 my $thr7 = new threads \&threaded, $foo, $fooe;
134
135 ok($thr1->join());
136 ok($thr2->join());
137 ok($thr3->join());
138 ok($thr4->join());
139 ok($thr5->join());
140 ok($thr6->join());
141 ok($thr7->join());
f9dff5f5 142}
38875929 143
144# test that 'yield' is importable
145
146package Test1;
147
148use threads 'yield';
149yield;
150main::ok(1);
151
152package main;
153
154
155# test async
156
157{
158 my $th = async {return 1 };
159 ok($th);
160 ok($th->join());
161}
9c98058e 162{
4dcb9e53 163 # There is a miniscule chance this test case may falsely fail
4acc73f2 164 # since it tests using rand()
9c98058e 165 my %rand : shared;
166 rand(10);
f4cc38af 167 threads->create( sub { $rand{int(rand(10000000000))}++ } ) foreach 1..25;
9c98058e 168 $_->join foreach threads->list;
4dcb9e53 169 ok((keys %rand >= 23), "Check that rand() is randomized in new threads");
9c98058e 170}
171
4e380990 172# bugid #24165
173
dcefd27c 174run_perl(prog => 'use threads 1.66;' .
60bd5ef6 175 'sub a{threads->create(shift)} $t = a sub{};' .
176 '$t->tid; $t->join; $t->tid',
69a9b4b8 177 nolib => ($ENV{PERL_CORE}) ? 0 : 1,
178 switches => ($ENV{PERL_CORE}) ? [] : [ '-Mblib' ]);
4e380990 179is($?, 0, 'coredump in global destruction');
9c98058e 180
09576c7d 181# Attempt to free unreferenced scalar...
182fresh_perl_is(<<'EOI', 'ok', { }, 'thread sub via scalar');
183 use threads;
184 my $test = sub {};
185 threads->create($test)->join();
186 print 'ok';
187EOI
188
189# Attempt to free unreferenced scalar...
190fresh_perl_is(<<'EOI', 'ok', { }, 'thread sub via $_[0]');
191 use threads;
192 sub thr { threads->new($_[0]); }
193 thr(sub { })->join;
194 print 'ok';
195EOI
196
863e9b4a 197# [perl #45053] Memory corruption from eval return in void context
198fresh_perl_is(<<'EOI', 'ok', { }, 'void eval return');
199 use threads;
200 threads->create(sub { eval '1' });
201 $_->join() for threads->list;
202 print 'ok';
203EOI
204
9660f481 205# test CLONE_SKIP() functionality
0f1612a7 206if ($] >= 5.008007) {
9660f481 207 my %c : shared;
208 my %d : shared;
209
210 # ---
211
212 package A;
213 sub CLONE_SKIP { $c{"A-$_[0]"}++; 1; }
214 sub DESTROY { $d{"A-". ref $_[0]}++ }
215
216 package A1;
217 our @ISA = qw(A);
218 sub CLONE_SKIP { $c{"A1-$_[0]"}++; 1; }
219 sub DESTROY { $d{"A1-". ref $_[0]}++ }
220
221 package A2;
222 our @ISA = qw(A1);
223
224 # ---
225
226 package B;
227 sub CLONE_SKIP { $c{"B-$_[0]"}++; 0; }
228 sub DESTROY { $d{"B-" . ref $_[0]}++ }
229
230 package B1;
231 our @ISA = qw(B);
232 sub CLONE_SKIP { $c{"B1-$_[0]"}++; 1; }
233 sub DESTROY { $d{"B1-" . ref $_[0]}++ }
234
235 package B2;
236 our @ISA = qw(B1);
237
238 # ---
239
240 package C;
241 sub CLONE_SKIP { $c{"C-$_[0]"}++; 1; }
242 sub DESTROY { $d{"C-" . ref $_[0]}++ }
243
244 package C1;
245 our @ISA = qw(C);
246 sub CLONE_SKIP { $c{"C1-$_[0]"}++; 0; }
247 sub DESTROY { $d{"C1-" . ref $_[0]}++ }
248
249 package C2;
250 our @ISA = qw(C1);
251
252 # ---
253
254 package D;
255 sub DESTROY { $d{"D-" . ref $_[0]}++ }
256
257 package D1;
258 our @ISA = qw(D);
259
260 package main;
261
262 {
fc04eb16 263 my @objs;
264 for my $class (qw(A A1 A2 B B1 B2 C C1 C2 D D1)) {
265 push @objs, bless [], $class;
266 }
267
268 sub f {
269 my $depth = shift;
270 my $cloned = ""; # XXX due to recursion, doesn't get initialized
271 $cloned .= "$_" =~ /ARRAY/ ? '1' : '0' for @objs;
272 is($cloned, ($depth ? '00010001111' : '11111111111'),
273 "objs clone skip at depth $depth");
274 threads->create( \&f, $depth+1)->join if $depth < 2;
275 @objs = ();
276 }
277 f(0);
9660f481 278 }
279
280 curr_test(curr_test()+2);
281 ok(eq_hash(\%c,
fc04eb16 282 {
283 qw(
284 A-A 2
285 A1-A1 2
286 A1-A2 2
287 B-B 2
288 B1-B1 2
289 B1-B2 2
290 C-C 2
291 C1-C1 2
292 C1-C2 2
293 )
294 }),
295 "counts of calls to CLONE_SKIP");
9660f481 296 ok(eq_hash(\%d,
fc04eb16 297 {
298 qw(
299 A-A 1
300 A1-A1 1
301 A1-A2 1
302 B-B 3
303 B1-B1 1
304 B1-B2 1
305 C-C 1
306 C1-C1 3
307 C1-C2 3
308 D-D 3
309 D-D1 3
310 )
311 }),
312 "counts of calls to DESTROY");
0f1612a7 313
314} else {
09576c7d 315 print("ok 29 # Skip objs clone skip at depth 0\n");
316 print("ok 30 # Skip objs clone skip at depth 1\n");
317 print("ok 31 # Skip objs clone skip at depth 2\n");
318 print("ok 32 # Skip counts of calls to CLONE_SKIP\n");
319 print("ok 33 # Skip counts of calls to DESTROY\n");
9660f481 320}
38875929 321
0f1612a7 322# EOF