Move the rest of the data munging into the test preparation loop.
[p5sagit/p5-mst-13.2.git] / t / op / sprintf.t
1 #!./perl
2
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
7 # specific to multi-byte characters (under the utf8 pragma and such).
8
9 BEGIN {
10     chdir 't' if -d 't';
11     @INC = '../lib';
12 }
13 use warnings;
14 use Config;
15 use strict;
16
17 my @tests = ();
18 my ($i, $template, $data, $result, $comment, $w, $x, $evalData, $n, $p);
19
20 my $Is_VMS_VAX = 0;
21 # We use HW_MODEL since ARCH_NAME was not in VMS V5.*
22 if ($^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;
26 }
27
28 # No %Config.
29 my $Is_Ultrix_VAX = $^O eq 'ultrix' && `uname -m` =~ /^VAX$/;
30
31 while (<DATA>) {
32     s/^\s*>//; s/<\s*$//;
33     ($template, $data, $result, $comment) = split(/<\s*>/, $_, 4);
34     if ($^O eq 'os390' || $^O eq 's390') { # non-IEEE (s390 is UTS)
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     }
40     if ($Is_VMS_VAX || $Is_Ultrix_VAX) {
41         # VAX DEC C 5.3 at least since there is no
42         # ccflags =~ /float=ieee/ on VAX.
43         # AXP is unaffected whether or not it's using ieee.
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     }
49
50     $evalData = eval $data;
51     $data = ref $evalData ? $evalData : [$evalData];
52     push @tests, [$template, $data, $result, $comment];
53 }
54
55 print '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
67 for ($i = 1; @tests; $i++) {
68     ($template, $data, $result, $comment) = @{shift @tests};
69     $w = undef;
70     $x = sprintf(">$template<", @$data);
71     substr($x, -1, 0) = $w if $w;
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         }
87     }
88
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;
102             $skip = $vsn ? ($osv <= $vsn ? 1 : 0) : 1;
103         }
104         $skip and $comment =~ s/$/, failure expected on $^O $osv/;
105     }
106
107     if ($x eq ">$result<") {
108         print "ok $i\n";
109     }
110     elsif ($skip) {
111         print "ok $i # skip $comment\n";
112     }
113     elsif ($y eq ">$result<")   # Some C libraries always give
114     {                           # three-digit exponent
115                 print("ok $i # >$result< $x three-digit exponent accepted\n");
116     }
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         {
123                 print("ok $i # >$template< >$data< >$result<",
124                           " Suppressed: exponent out of range?\n");
125         }
126     else {
127         $y = ($x eq $y ? "" : " => $y");
128         print("not ok $i >$template< >$data< >$result< $x$y",
129             $comment ? " # $comment\n" : "\n");
130     }
131 }
132
133 # In each of the following lines, there are three required fields:
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 #
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<
151 # >%.0g< >-0.0<        >-0<             >No minus skip: MSWin32 VMS hpux:10.20<
152 # >%d<   >4<           >1<              >4 != 1 skip: all<
153 #
154 # The following tests are not currently run, for the reasons stated:
155
156 =pod
157
158 =begin problematic
159
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
167 # template    data          result
168 __END__
169 >%6. 6s<    >''<          >%6. 6s INVALID< >(See use of $w in code above)<
170 >%6 .6s<    >''<          >%6 .6s INVALID<
171 >%6.6 s<    >''<          >%6.6 s INVALID<
172 >%A<        >''<          >%A INVALID<
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"<
179 >%G<        >1234567e96<  >1.23457E+102<
180 >%G<        >.1234567e-101< >1.23457E-102<
181 >%G<        >12345.6789<  >12345.7<
182 >%G<        >1234567e96<  >1.23457E+102<        >exponent too big skip: os390<
183 >%G<        >.1234567e-101< >1.23457E-102<      >exponent too small skip: os390<
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<
191 >%O<        >2**32-1<     >37777777777<    >Synonym for %lo<
192 >%P<        >''<          >%P INVALID<
193 >%Q<        >''<          >%Q INVALID<
194 >%R<        >''<          >%R INVALID<
195 >%S<        >''<          >%S INVALID<
196 >%T<        >''<          >%T INVALID<
197 >%U<        >2**32-1<     >4294967295<     >Synonym for %lu<
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<
245 >%0v3d<     >"\01\02\03"< >001.002.003<
246 >%-v3d<     >"\01\02\03"< >1  .2  .3  <
247 >%+-v3d<    >"\01\02\03"< >+1 .2  .3  <
248 >%v4.3d<    >"\01\02\03"< > 001. 002. 003<
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<
258 >%e<        >1234.875<    >1.234875e+03<
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<
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<
269 >%#.0e<     >1234.875<    >1.e+03<
270 >%.0e<      >1.875<       >2e+00<
271 >%.0e<      >0.875<       >9e-01<
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<
280 >%e<        >1234567E96<  >1.234567e+102<       >exponent too big skip: os390<
281 >%e<        >.1234567E-101< >1.234567e-102<     >exponent too small skip: os390<
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<
290 >%.0f<      >-0.1<        >-0<  >C library bug: no minus skip: VMS<
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<
300 >%f<        >0<           >0.000000<
301 >%.0f<      >0<           >0<
302 >%.0f<      >2**38<       >274877906944<   >Should have exact int'l rep'n<
303 >%.0f<      >0.1<         >0<
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<
308 >%.0f<      >1<           >1<
309 >%#.0f<     >1<           >1.<
310 >%.0lf<     >1<           >1<              >'l' should have no effect<
311 >%.0hf<     >1<           >%.0hf INVALID<  >'h' should be rejected<
312 >%g<        >12345.6789<  >12345.7<
313 >%+g<       >12345.6789<  >+12345.7<
314 >%#g<       >12345.6789<  >12345.7<
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<
316 >%.0g<      >12345.6789<  >1e+04<
317 >%#.0g<     >12345.6789<  >1.e+04<
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<
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<
339 >%13g<      >1234567.89<  >  1.23457e+06<
340 >%+13g<     >1234567.89<  > +1.23457e+06<
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<
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<
355 >%o<        >642<         >1202<          >check smaller octals across platforms<
356 >%+o<       >642<         >1202<
357 >%#o<       >642<         >01202<
358 >%d< >$p=sprintf('%p',$p);$p=~/^[0-9a-f]+$/< >1< >Coarse hack: hex from %p?<
359 >%d< >$p=sprintf('%-8p',$p);$p=~/^[0-9a-f]+\s*$/< >1< >Coarse hack: hex from %p?<
360 >%#p<       >''<          >%#p INVALID<
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<
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<
400 >%*3$2$d %d<    >[12, 34, 3]<   >%*3$2$d 12 INVALID<
401 >%2$d<          >12<    >0 UNINIT<
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<
407 >%*2$1d<        >[12, 3]<       >%*2$1d INVALID<
408 >%0v2.2d<       >''<    ><
409 >%vc,%d<        >[63, 64, 65]<  >%vc,63 INVALID<
410 >%v%,%d<        >[63, 64, 65]<  >%v%,63 INVALID<
411 >%vd,%d<        >[1, 2, 3]<     >49,2<
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<
418 >%vp<   >''<    >%vp INVALID<
419 >%vn<   >''<    >%vn INVALID<
420 >%vs,%d<        >[1, 2, 3]<     >%vs,1 INVALID<
421 >%v_<   >''<    >%v_ INVALID<
422 >%v#x<  >''<    >%v#x INVALID<
423 >%v02x< >"foo\012"<     >66.6f.6f.0a<
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<
429 >%*v*999\$d %d %d<      >[11, 22, 33]<  >%*v*999\$d 11 22 INVALID<
430 >%#b<           >0<     >0<
431 >%#o<           >0<     >0<
432 >%#x<           >0<     >0<
433 >%2918905856$v2d<       >''<    ><
434 >%*2918905856$v2d<      >''<    > UNINIT<