Move the rest of the data munging into the test preparation loop.
[p5sagit/p5-mst-13.2.git] / t / op / sprintf.t
CommitLineData
8d063cd8 1#!./perl
2
be3174d2 3# Tests sprintf, excluding handling of 64-bit integers or long
4# doubles (if supported), of machine-specific short and long
5# integers, machine-specific floating point exceptions (infinity,
6# not-a-number ...), of the effects of locale, and of features
169da838 7# specific to multi-byte characters (under the utf8 pragma and such).
8d063cd8 8
9f1b1f2d 9BEGIN {
10 chdir 't' if -d 't';
20822f61 11 @INC = '../lib';
e24bffee 12}
9f1b1f2d 13use warnings;
e24bffee 14use Config;
0a52d15b 15use strict;
8234e14b 16
8234e14b 17my @tests = ();
18my ($i, $template, $data, $result, $comment, $w, $x, $evalData, $n, $p);
9f1b1f2d 19
8234e14b 20my $Is_VMS_VAX = 0;
2fba3065 21# We use HW_MODEL since ARCH_NAME was not in VMS V5.*
22if ($^O eq 'VMS') {
23 my $hw_model;
24 chomp($hw_model = `write sys\$output f\$getsyi("HW_MODEL")`);
25 $Is_VMS_VAX = $hw_model < 1024 ? 1 : 0;
8234e14b 26}
27
eaf637cf 28# No %Config.
29my $Is_Ultrix_VAX = $^O eq 'ultrix' && `uname -m` =~ /^VAX$/;
30
0a52d15b 31while (<DATA>) {
32 s/^\s*>//; s/<\s*$//;
33 ($template, $data, $result, $comment) = split(/<\s*>/, $_, 4);
e95e2653 34 if ($^O eq 'os390' || $^O eq 's390') { # non-IEEE (s390 is UTS)
12ebcc11 35 $data =~ s/([eE])96$/${1}63/; # smaller exponents
36 $result =~ s/([eE]\+)102$/${1}69/; # " "
37 $data =~ s/([eE])\-101$/${1}-56/; # larger exponents
38 $result =~ s/([eE])\-102$/${1}-57/; # " "
39 }
eaf637cf 40 if ($Is_VMS_VAX || $Is_Ultrix_VAX) {
e24bffee 41 # VAX DEC C 5.3 at least since there is no
eaf637cf 42 # ccflags =~ /float=ieee/ on VAX.
43 # AXP is unaffected whether or not it's using ieee.
8234e14b 44 $data =~ s/([eE])96$/${1}26/; # smaller exponents
45 $result =~ s/([eE]\+)102$/${1}32/; # " "
46 $data =~ s/([eE])\-101$/${1}-24/; # larger exponents
47 $result =~ s/([eE])\-102$/${1}-25/; # " "
48 }
57c348a9 49
50 $evalData = eval $data;
51 $data = ref $evalData ? $evalData : [$evalData];
0a52d15b 52 push @tests, [$template, $data, $result, $comment];
53}
54
55print '1..', scalar @tests, "\n";
56
57$SIG{__WARN__} = sub {
58 if ($_[0] =~ /^Invalid conversion/) {
59 $w = ' INVALID';
60 } elsif ($_[0] =~ /^Use of uninitialized value/) {
61 $w = ' UNINIT';
62 } else {
63 warn @_;
64 }
65};
66
67for ($i = 1; @tests; $i++) {
68 ($template, $data, $result, $comment) = @{shift @tests};
be3174d2 69 $w = undef;
57c348a9 70 $x = sprintf(">$template<", @$data);
be3174d2 71 substr($x, -1, 0) = $w if $w;
48237dde 72 # $x may have 3 exponent digits, not 2
73 my $y = $x;
74 if ($y =~ s/([Ee][-+])0(\d)/$1$2/) {
75 # if result is left-adjusted, append extra space
76 if ($template =~ /%\+?\-/ and $result =~ / $/) {
77 $y =~ s/<$/ </;
78 }
79 # if result is zero-filled, add extra zero
80 elsif ($template =~ /%\+?0/ and $result =~ /^0/) {
81 $y =~ s/^>0/>00/;
82 }
83 # if result is right-adjusted, prepend extra space
84 elsif ($result =~ /^ /) {
85 $y =~ s/^>/> /;
86 }
65c97e0f 87 }
88
e24bffee 89 my $skip = 0;
90 if ($comment =~ s/\s+skip:\s*(.*)//) {
91 my $os = $1;
92 my $osv = exists $Config{osvers} ? $Config{osvers} : "0";
93 # >comment skip: all<
94 if ($os =~ /\ball\b/i) {
95 $skip = 1;
96 # >comment skip: VMS hpux:10.20<
97 } elsif ($os =~ /\b$^O(?::(\S+))?\b/i) {
98 my $vsn = defined $1 ? $1 : "0";
99 # Only compare on the the first pair of digits, as numeric
100 # compares don't like 2.6.10-3mdksmp or 2.6.8-24.10-default
101 s/^(\d+(\.\d+)?).*/$1/ for $osv, $vsn;
6f1f3b4a 102 $skip = $vsn ? ($osv <= $vsn ? 1 : 0) : 1;
e24bffee 103 }
104 $skip and $comment =~ s/$/, failure expected on $^O $osv/;
105 }
106
be3174d2 107 if ($x eq ">$result<") {
108 print "ok $i\n";
109 }
e24bffee 110 elsif ($skip) {
111 print "ok $i # skip $comment\n";
112 }
48237dde 113 elsif ($y eq ">$result<") # Some C libraries always give
65c97e0f 114 { # three-digit exponent
00b6c67e 115 print("ok $i # >$result< $x three-digit exponent accepted\n");
65c97e0f 116 }
f7137e37 117 elsif ($result =~ /[-+]\d{3}$/ &&
118 # Suppress tests with modulo of exponent >= 100 on platforms
119 # which can't handle such magnitudes (or where we can't tell).
120 ((!eval {require POSIX}) || # Costly: only do this if we must!
121 (length(&POSIX::DBL_MAX) - rindex(&POSIX::DBL_MAX, '+')) == 3))
122 {
00b6c67e 123 print("ok $i # >$template< >$data< >$result<",
e24bffee 124 " Suppressed: exponent out of range?\n");
f7137e37 125 }
be3174d2 126 else {
48237dde 127 $y = ($x eq $y ? "" : " => $y");
128 print("not ok $i >$template< >$data< >$result< $x$y",
65c97e0f 129 $comment ? " # $comment\n" : "\n");
fb73857a 130 }
131}
48237dde 132
a6d05634 133# In each of the following lines, there are three required fields:
be3174d2 134# printf template, data to be formatted (as a Perl expression), and
135# expected result of formatting. An optional fourth field can contain
136# a comment. Each field is delimited by a starting '>' and a
137# finishing '<'; any whitespace outside these start and end marks is
138# not part of the field. If formatting requires more than one data
139# item (for example, if variable field widths are used), the Perl data
140# expression should return a reference to an array having the requisite
141# number of elements. Even so, subterfuge is sometimes required: see
142# tests for %n and %p.
143#
e24bffee 144# Tests that are expected to fail on a certain OS can be marked as such
145# by trailing the comment with a skip: section. Skips are tags separated
146# bu space consisting of a $^O optionally trailed with :osvers. In the
147# latter case, all os-levels below that are expected to fail. A special
148# tag 'all' is allowed for todo tests that should fail on any system
149#
150# >%G< >1234567e96< >1.23457E+102< >exponent too big skip: os390<
6f1f3b4a 151# >%.0g< >-0.0< >-0< >No minus skip: MSWin32 VMS hpux:10.20<
e24bffee 152# >%d< >4< >1< >4 != 1 skip: all<
153#
02a4ca6d 154# The following tests are not currently run, for the reasons stated:
155
156=pod
157
158=begin problematic
159
02a4ca6d 160>%.0f< >1.5< >2< >Standard vague: no rounding rules<
161>%.0f< >2.5< >2< >Standard vague: no rounding rules<
162
163=end problematic
164
165=cut
166
be3174d2 167# template data result
168__END__
be3174d2 169>%6. 6s< >''< >%6. 6s INVALID< >(See use of $w in code above)<
c2e66d9e 170>%6 .6s< >''< >%6 .6s INVALID<
be3174d2 171>%6.6 s< >''< >%6.6 s INVALID<
c2e66d9e 172>%A< >''< >%A INVALID<
be3174d2 173>%B< >''< >%B INVALID<
174>%C< >''< >%C INVALID<
175>%D< >0x7fffffff< >2147483647< >Synonym for %ld<
176>%E< >123456.789< >1.234568E+05< >Like %e, but using upper-case "E"<
177>%F< >123456.789< >123456.789000< >Synonym for %f<
178>%G< >1234567.89< >1.23457E+06< >Like %g, but using upper-case "E"<
c2e66d9e 179>%G< >1234567e96< >1.23457E+102<
180>%G< >.1234567e-101< >1.23457E-102<
be3174d2 181>%G< >12345.6789< >12345.7<
e24bffee 182>%G< >1234567e96< >1.23457E+102< >exponent too big skip: os390<
183>%G< >.1234567e-101< >1.23457E-102< >exponent too small skip: os390<
be3174d2 184>%H< >''< >%H INVALID<
185>%I< >''< >%I INVALID<
186>%J< >''< >%J INVALID<
187>%K< >''< >%K INVALID<
188>%L< >''< >%L INVALID<
189>%M< >''< >%M INVALID<
190>%N< >''< >%N INVALID<
8234e14b 191>%O< >2**32-1< >37777777777< >Synonym for %lo<
be3174d2 192>%P< >''< >%P INVALID<
193>%Q< >''< >%Q INVALID<
194>%R< >''< >%R INVALID<
195>%S< >''< >%S INVALID<
196>%T< >''< >%T INVALID<
8234e14b 197>%U< >2**32-1< >4294967295< >Synonym for %lu<
be3174d2 198>%V< >''< >%V INVALID<
199>%W< >''< >%W INVALID<
200>%X< >2**32-1< >FFFFFFFF< >Like %x, but with u/c letters<
201>%#X< >2**32-1< >0XFFFFFFFF<
202>%Y< >''< >%Y INVALID<
203>%Z< >''< >%Z INVALID<
204>%a< >''< >%a INVALID<
205>%b< >2**32-1< >11111111111111111111111111111111<
206>%+b< >2**32-1< >11111111111111111111111111111111<
207>%#b< >2**32-1< >0b11111111111111111111111111111111<
208>%34b< >2**32-1< > 11111111111111111111111111111111<
209>%034b< >2**32-1< >0011111111111111111111111111111111<
210>%-34b< >2**32-1< >11111111111111111111111111111111 <
211>%-034b< >2**32-1< >11111111111111111111111111111111 <
212>%c< >ord('A')< >A<
213>%10c< >ord('A')< > A<
214>%#10c< >ord('A')< > A< ># modifier: no effect<
215>%010c< >ord('A')< >000000000A<
216>%10lc< >ord('A')< > A< >l modifier: no effect<
217>%10hc< >ord('A')< > A< >h modifier: no effect<
218>%10.5c< >ord('A')< > A< >precision: no effect<
219>%-10c< >ord('A')< >A <
220>%d< >123456.789< >123456<
221>%d< >-123456.789< >-123456<
222>%d< >0< >0<
223>%+d< >0< >+0<
224>%0d< >0< >0<
225>%.0d< >0< ><
226>%+.0d< >0< >+<
227>%.0d< >1< >1<
228>%d< >1< >1<
229>%+d< >1< >+1<
230>%#3.2d< >1< > 01< ># modifier: no effect<
231>%3.2d< >1< > 01<
232>%03.2d< >1< >001<
233>%-3.2d< >1< >01 <
234>%-03.2d< >1< >01 < >zero pad + left just.: no effect<
235>%d< >-1< >-1<
236>%+d< >-1< >-1<
237>%hd< >1< >1< >More extensive testing of<
238>%ld< >1< >1< >length modifiers would be<
239>%Vd< >1< >1< >platform-specific<
240>%vd< >chr(1)< >1<
241>%+vd< >chr(1)< >+1<
242>%#vd< >chr(1)< >1<
243>%vd< >"\01\02\03"< >1.2.3<
244>%v.3d< >"\01\02\03"< >001.002.003<
211dfcf1 245>%0v3d< >"\01\02\03"< >001.002.003<
246>%-v3d< >"\01\02\03"< >1 .2 .3 <
247>%+-v3d< >"\01\02\03"< >+1 .2 .3 <
be3174d2 248>%v4.3d< >"\01\02\03"< > 001. 002. 003<
211dfcf1 249>%0v4.3d< >"\01\02\03"< >0001.0002.0003<
250>%0*v2d< >['-', "\0\7\14"]< >00-07-12<
251>%v.*d< >["\01\02\03", 3]< >001.002.003<
252>%0v*d< >["\01\02\03", 3]< >001.002.003<
253>%-v*d< >["\01\02\03", 3]< >1 .2 .3 <
254>%+-v*d< >["\01\02\03", 3]< >+1 .2 .3 <
255>%v*.*d< >["\01\02\03", 4, 3]< > 001. 002. 003<
256>%0v*.*d< >["\01\02\03", 4, 3]< >0001.0002.0003<
257>%0*v*d< >['-', "\0\7\13", 2]< >00-07-11<
be3174d2 258>%e< >1234.875< >1.234875e+03<
c2e66d9e 259>%e< >0.000012345< >1.234500e-05<
260>%e< >1234567E96< >1.234567e+102<
261>%e< >0< >0.000000e+00<
262>%e< >.1234567E-101< >1.234567e-102<
be3174d2 263>%+e< >1234.875< >+1.234875e+03<
264>%#e< >1234.875< >1.234875e+03<
265>%e< >-1234.875< >-1.234875e+03<
266>%+e< >-1234.875< >-1.234875e+03<
267>%#e< >-1234.875< >-1.234875e+03<
268>%.0e< >1234.875< >1e+03<
02a4ca6d 269>%#.0e< >1234.875< >1.e+03<
20f6aaab 270>%.0e< >1.875< >2e+00<
271>%.0e< >0.875< >9e-01<
be3174d2 272>%.*e< >[0, 1234.875]< >1e+03<
273>%.1e< >1234.875< >1.2e+03<
274>%-12.4e< >1234.875< >1.2349e+03 <
275>%12.4e< >1234.875< > 1.2349e+03<
276>%+-12.4e< >1234.875< >+1.2349e+03 <
277>%+12.4e< >1234.875< > +1.2349e+03<
278>%+-12.4e< >-1234.875< >-1.2349e+03 <
279>%+12.4e< >-1234.875< > -1.2349e+03<
e24bffee 280>%e< >1234567E96< >1.234567e+102< >exponent too big skip: os390<
281>%e< >.1234567E-101< >1.234567e-102< >exponent too small skip: os390<
be3174d2 282>%f< >1234.875< >1234.875000<
283>%+f< >1234.875< >+1234.875000<
284>%#f< >1234.875< >1234.875000<
285>%f< >-1234.875< >-1234.875000<
286>%+f< >-1234.875< >-1234.875000<
287>%#f< >-1234.875< >-1234.875000<
288>%6f< >1234.875< >1234.875000<
289>%*f< >[6, 1234.875]< >1234.875000<
e24bffee 290>%.0f< >-0.1< >-0< >C library bug: no minus skip: VMS<
be3174d2 291>%.0f< >1234.875< >1235<
292>%.1f< >1234.875< >1234.9<
293>%-8.1f< >1234.875< >1234.9 <
294>%8.1f< >1234.875< > 1234.9<
295>%+-8.1f< >1234.875< >+1234.9 <
296>%+8.1f< >1234.875< > +1234.9<
297>%+-8.1f< >-1234.875< >-1234.9 <
298>%+8.1f< >-1234.875< > -1234.9<
299>%*.*f< >[5, 2, 12.3456]< >12.35<
c2e66d9e 300>%f< >0< >0.000000<
301>%.0f< >0< >0<
302>%.0f< >2**38< >274877906944< >Should have exact int'l rep'n<
d5365ef1 303>%.0f< >0.1< >0<
20f6aaab 304>%.0f< >0.6< >1< >Known to fail with sfio, (irix|nonstop-ux|powerux); -DHAS_LDBL_SPRINTF_BUG may fix<
305>%.0f< >-0.6< >-1< >Known to fail with sfio, (irix|nonstop-ux|powerux); -DHAS_LDBL_SPRINTF_BUG may fix<
306>%.0f< >1.6< >2<
307>%.0f< >-1.6< >-2<
02a4ca6d 308>%.0f< >1< >1<
309>%#.0f< >1< >1.<
00e17364 310>%.0lf< >1< >1< >'l' should have no effect<
311>%.0hf< >1< >%.0hf INVALID< >'h' should be rejected<
be3174d2 312>%g< >12345.6789< >12345.7<
313>%+g< >12345.6789< >+12345.7<
314>%#g< >12345.6789< >12345.7<
65ff66fc 315>%.0g< >-0.0< >-0< >C99 standard mandates minus sign but C89 does not skip: MSWin32 VMS hpux:10.20 openbsd netbsd:1.5 irix<
be3174d2 316>%.0g< >12345.6789< >1e+04<
02a4ca6d 317>%#.0g< >12345.6789< >1.e+04<
be3174d2 318>%.2g< >12345.6789< >1.2e+04<
319>%.*g< >[2, 12345.6789]< >1.2e+04<
320>%.9g< >12345.6789< >12345.6789<
321>%12.9g< >12345.6789< > 12345.6789<
322>%012.9g< >12345.6789< >0012345.6789<
323>%-12.9g< >12345.6789< >12345.6789 <
324>%*.*g< >[-12, 9, 12345.6789]< >12345.6789 <
325>%-012.9g< >12345.6789< >12345.6789 <
326>%g< >-12345.6789< >-12345.7<
327>%+g< >-12345.6789< >-12345.7<
328>%g< >1234567.89< >1.23457e+06<
329>%+g< >1234567.89< >+1.23457e+06<
330>%#g< >1234567.89< >1.23457e+06<
331>%g< >-1234567.89< >-1.23457e+06<
332>%+g< >-1234567.89< >-1.23457e+06<
333>%#g< >-1234567.89< >-1.23457e+06<
c2e66d9e 334>%g< >0.00012345< >0.00012345<
335>%g< >0.000012345< >1.2345e-05<
336>%g< >1234567E96< >1.23457e+102<
337>%g< >.1234567E-101< >1.23457e-102<
338>%g< >0< >0<
be3174d2 339>%13g< >1234567.89< > 1.23457e+06<
340>%+13g< >1234567.89< > +1.23457e+06<
e24bffee 341>%013g< >1234567.89< >001.23457e+06<
342>%-13g< >1234567.89< >1.23457e+06 <
343>%g< >.1234567E-101< >1.23457e-102< >exponent too small skip: os390<
344>%g< >1234567E96< >1.23457e+102< >exponent too big skip: os390<
be3174d2 345>%h< >''< >%h INVALID<
346>%i< >123456.789< >123456< >Synonym for %d<
347>%j< >''< >%j INVALID<
348>%k< >''< >%k INVALID<
349>%l< >''< >%l INVALID<
350>%m< >''< >%m INVALID<
351>%s< >sprintf('%%n%n %d', $n, $n)< >%n 2< >Slight sneakiness to test %n<
352>%o< >2**32-1< >37777777777<
353>%+o< >2**32-1< >37777777777<
354>%#o< >2**32-1< >037777777777<
8234e14b 355>%o< >642< >1202< >check smaller octals across platforms<
356>%+o< >642< >1202<
357>%#o< >642< >01202<
be3174d2 358>%d< >$p=sprintf('%p',$p);$p=~/^[0-9a-f]+$/< >1< >Coarse hack: hex from %p?<
0dbb1585 359>%d< >$p=sprintf('%-8p',$p);$p=~/^[0-9a-f]+\s*$/< >1< >Coarse hack: hex from %p?<
d5365ef1 360>%#p< >''< >%#p INVALID<
be3174d2 361>%q< >''< >%q INVALID<
362>%r< >''< >%r INVALID<
363>%s< >'string'< >string<
364>%10s< >'string'< > string<
365>%+10s< >'string'< > string<
366>%#10s< >'string'< > string<
367>%010s< >'string'< >0000string<
368>%0*s< >[10, 'string']< >0000string<
369>%-10s< >'string'< >string <
370>%3s< >'string'< >string<
371>%.3s< >'string'< >str<
372>%.*s< >[3, 'string']< >str<
373>%t< >''< >%t INVALID<
374>%u< >2**32-1< >4294967295<
375>%+u< >2**32-1< >4294967295<
376>%#u< >2**32-1< >4294967295<
377>%12u< >2**32-1< > 4294967295<
378>%012u< >2**32-1< >004294967295<
379>%-12u< >2**32-1< >4294967295 <
380>%-012u< >2**32-1< >4294967295 <
381>%v< >''< >%v INVALID<
382>%w< >''< >%w INVALID<
383>%x< >2**32-1< >ffffffff<
384>%+x< >2**32-1< >ffffffff<
385>%#x< >2**32-1< >0xffffffff<
386>%10x< >2**32-1< > ffffffff<
387>%010x< >2**32-1< >00ffffffff<
388>%-10x< >2**32-1< >ffffffff <
389>%-010x< >2**32-1< >ffffffff <
390>%0-10x< >2**32-1< >ffffffff <
391>%0*x< >[-10, ,2**32-1]< >ffffffff <
392>%y< >''< >%y INVALID<
393>%z< >''< >%z INVALID<
eb3fce90 394>%2$d %1$d< >[12, 34]< >34 12<
395>%*2$d< >[12, 3]< > 12<
396>%2$d %d< >[12, 34]< >34 12<
397>%2$d %d %d< >[12, 34]< >34 12 34<
398>%3$d %d %d< >[12, 34, 56]< >56 12 34<
399>%2$*3$d %d< >[12, 34, 3]< > 34 12<
58e33a90 400>%*3$2$d %d< >[12, 34, 3]< >%*3$2$d 12 INVALID<
211dfcf1 401>%2$d< >12< >0 UNINIT<
eb3fce90 402>%0$d< >12< >%0$d INVALID<
403>%1$$d< >12< >%1$$d INVALID<
404>%1$1$d< >12< >%1$1$d INVALID<
405>%*2$*2$d< >[12, 3]< >%*2$*2$d INVALID<
406>%*2*2$d< >[12, 3]< >%*2*2$d INVALID<
8896765a 407>%*2$1d< >[12, 3]< >%*2$1d INVALID<
211dfcf1 408>%0v2.2d< >''< ><
26372e71 409>%vc,%d< >[63, 64, 65]< >%vc,63 INVALID<
410>%v%,%d< >[63, 64, 65]< >%v%,63 INVALID<
be75b157 411>%vd,%d< >[1, 2, 3]< >49,2<
26372e71 412>%vf,%d< >[1, 2, 3]< >%vf,1 INVALID<
413>%vF,%d< >[1, 2, 3]< >%vF,1 INVALID<
414>%ve,%d< >[1, 2, 3]< >%ve,1 INVALID<
415>%vE,%d< >[1, 2, 3]< >%vE,1 INVALID<
416>%vg,%d< >[1, 2, 3]< >%vg,1 INVALID<
417>%vG,%d< >[1, 2, 3]< >%vG,1 INVALID<
be75b157 418>%vp< >''< >%vp INVALID<
26372e71 419>%vn< >''< >%vn INVALID<
420>%vs,%d< >[1, 2, 3]< >%vs,1 INVALID<
be75b157 421>%v_< >''< >%v_ INVALID<
f3583277 422>%v#x< >''< >%v#x INVALID<
63c6dcc1 423>%v02x< >"foo\012"< >66.6f.6f.0a<
58e33a90 424>%V-%s< >["Hello"]< >%V-Hello INVALID<
425>%K %d %d< >[13, 29]< >%K 13 29 INVALID<
426>%*.*K %d< >[13, 29, 76]< >%*.*K 13 INVALID<
427>%4$K %d< >[45, 67]< >%4$K 45 INVALID<
428>%d %K %d< >[23, 45]< >23 %K 45 INVALID<
3a7a539e 429>%*v*999\$d %d %d< >[11, 22, 33]< >%*v*999\$d 11 22 INVALID<
ed2b91d2 430>%#b< >0< >0<
431>%#o< >0< >0<
432>%#x< >0< >0<
7ad96abb 433>%2918905856$v2d< >''< ><
434>%*2918905856$v2d< >''< > UNINIT<