This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
Add skip_all_without_extension() to test.pl, for the common skip idiom.
[perl5.git] / t / io / layers.t
1 #!./perl
2
3 my $PERLIO;
4
5 BEGIN {
6     chdir 't' if -d 't';
7     @INC = '../lib';
8     require './test.pl';
9     skip_all('not perlio') unless (find PerlIO::Layer 'perlio');
10     # FIXME - more of these could be tested without Encode or full perl
11     skip_all_if_miniperl("no dynamic loading on miniperl, no Encode");
12     skip_all_without_extension('Encode');
13
14     # Makes testing easier.
15     $ENV{PERLIO} = 'stdio' if exists $ENV{PERLIO} && $ENV{PERLIO} eq '';
16     if (exists $ENV{PERLIO} && $ENV{PERLIO} !~ /^(stdio|perlio|mmap)$/) {
17         # We are not prepared for anything else.
18         print "1..0 # PERLIO='$ENV{PERLIO}' unknown\n";
19         exit 0;
20     }
21     $PERLIO = exists $ENV{PERLIO} ? $ENV{PERLIO} : "(undef)";
22 }
23
24 use Config;
25
26 my $DOSISH    = $^O =~ /^(?:MSWin32|os2|dos|NetWare)$/ ? 1 : 0;
27    $DOSISH    = 1 if !$DOSISH and $^O =~ /^uwin/;
28 my $NONSTDIO  = exists $ENV{PERLIO} && $ENV{PERLIO} ne 'stdio'     ? 1 : 0;
29 my $FASTSTDIO = $Config{d_faststdio} && $Config{usefaststdio}      ? 1 : 0;
30 my $UTF8_STDIN;
31 if (${^UNICODE} & 1) {
32     if (${^UNICODE} & 64) {
33         # Conditional on the locale
34         $UTF8_STDIN = ${^UTF8LOCALE};
35     } else {
36         # Unconditional
37         $UTF8_STDIN = 1;
38     }
39 } else {
40     $UTF8_STDIN = 0;
41 }
42 my $NTEST = 55 - (($DOSISH || !$FASTSTDIO) ? 7 : 0) - ($DOSISH ? 7 : 0)
43     + $UTF8_STDIN;
44
45 sub PerlIO::F_UTF8 () { 0x00008000 } # from perliol.h
46
47 plan tests => $NTEST;
48
49 print <<__EOH__;
50 # PERLIO        = $PERLIO
51 # DOSISH        = $DOSISH
52 # NONSTDIO      = $NONSTDIO
53 # FASTSTDIO     = $FASTSTDIO
54 # UNICODE       = ${^UNICODE}
55 # UTF8LOCALE    = ${^UTF8LOCALE}
56 # UTF8_STDIN = $UTF8_STDIN
57 __EOH__
58
59 {
60     sub check {
61         my ($result, $expected, $id) = @_;
62         # An interesting dance follows where we try to make the following
63         # IO layer stack setups to compare equal:
64         #
65         # PERLIO     UNIX-like                   DOS-like
66         #
67         # unset / "" unix perlio / stdio [1]     unix crlf
68         # stdio      unix perlio / stdio [1]     stdio
69         # perlio     unix perlio                 unix perlio
70         # mmap       unix mmap                   unix mmap
71         #
72         # [1] "stdio" if Configure found out how to do "fast stdio" (depends
73         # on the stdio implementation) and in Perl 5.8, otherwise "unix perlio"
74         #
75         if ($NONSTDIO) {
76             # Get rid of "unix".
77             shift @$result if $result->[0] eq "unix";
78             # Change expectations.
79             if ($FASTSTDIO) {
80                 $expected->[0] = $ENV{PERLIO};
81             } else {
82                 $expected->[0] = $ENV{PERLIO} if $expected->[0] eq "stdio";
83             }
84         } elsif (!$FASTSTDIO && !$DOSISH) {
85             splice(@$result, 0, 2, "stdio")
86                 if @$result >= 2 &&
87                    $result->[0] eq "unix" &&
88                    $result->[1] eq "perlio";
89         } elsif ($DOSISH) {
90             splice(@$result, 0, 2, "stdio")
91                 if @$result >= 2 &&
92                    $result->[0] eq "unix" &&
93                    $result->[1] eq "crlf";
94         }
95         if ($DOSISH && grep { $_ eq 'crlf' } @$expected) {
96             # 5 tests potentially skipped because
97             # DOSISH systems already have a CRLF layer
98             # which will make new ones not stick.
99             splice @$expected, 1, 1 if $expected->[1] eq 'crlf';
100         }
101         my $n = scalar @$expected;
102         is(scalar @$result, $n, "$id - layers == $n");
103         for (my $i = 0; $i < $n; $i++) {
104             my $j = $expected->[$i];
105             if (ref $j eq 'CODE') {
106                 ok($j->($result->[$i]), "$id - $i is ok");
107             } else {
108                 is($result->[$i], $j,
109                    sprintf("$id - $i is %s",
110                            defined $j ? $j : "undef"));
111             }
112         }
113     }
114
115     check([ PerlIO::get_layers(STDIN) ],
116           $UTF8_STDIN ? [ "stdio", "utf8" ] : [ "stdio" ],
117           "STDIN");
118
119     my $afile = tempfile();
120     open(F, ">:crlf", $afile);
121
122     check([ PerlIO::get_layers(F) ],
123           [ qw(stdio crlf) ],
124           "open :crlf");
125
126     binmode(F, ":crlf");
127
128     check([ PerlIO::get_layers(F) ],
129           [ qw(stdio crlf) ],
130           "binmode :crlf");
131
132     binmode(F, ":encoding(cp1047)"); 
133
134     check([ PerlIO::get_layers(F) ],
135           [ qw[stdio crlf encoding(cp1047) utf8] ],
136           ":encoding(cp1047)");
137
138     binmode(F, ":crlf");
139
140     check([ PerlIO::get_layers(F) ],
141           [ qw[stdio crlf encoding(cp1047) utf8 crlf utf8] ],
142           ":encoding(cp1047):crlf");
143     
144     binmode(F, ":pop:pop");
145
146     check([ PerlIO::get_layers(F) ],
147           [ qw(stdio crlf) ],
148           ":pop");
149
150     binmode(F, ":raw");
151
152     check([ PerlIO::get_layers(F) ],
153           [ "stdio" ],
154           ":raw");
155
156     binmode(F, ":utf8");
157
158     check([ PerlIO::get_layers(F) ],
159           [ qw(stdio utf8) ],
160           ":utf8");
161
162     binmode(F, ":bytes");
163
164     check([ PerlIO::get_layers(F) ],
165           [ "stdio" ],
166           ":bytes");
167
168     binmode(F, ":encoding(utf8)");
169
170     check([ PerlIO::get_layers(F) ],
171             [ qw[stdio encoding(utf8) utf8] ],
172             ":encoding(utf8)");
173
174     binmode(F, ":raw :crlf");
175
176     check([ PerlIO::get_layers(F) ],
177           [ qw(stdio crlf) ],
178           ":raw:crlf");
179
180     binmode(F, ":raw :encoding(latin1)"); # "latin1" will be canonized
181
182     # 7 tests potentially skipped.
183     unless ($DOSISH || !$FASTSTDIO) {
184         my @results = PerlIO::get_layers(F, details => 1);
185
186         # Get rid of the args and the flags.
187         splice(@results, 1, 2) if $NONSTDIO;
188
189         check([ @results ],
190               [ "stdio",    undef,        sub { $_[0] > 0 },
191                 "encoding", "iso-8859-1", sub { $_[0] & PerlIO::F_UTF8() } ],
192               ":raw:encoding(latin1)");
193     }
194
195     binmode(F);
196
197     check([ PerlIO::get_layers(F) ],
198           [ "stdio" ],
199           "binmode");
200
201     # RT78844
202     {
203         local $@ = "foo";
204         binmode(F, ":encoding(utf8)");
205         is( $@, "foo", '$@ not clobbered by binmode and :encoding');
206     }
207
208     close F;
209
210     {
211         use open(IN => ":crlf", OUT => ":encoding(cp1252)");
212
213         open F, '<', $afile;
214         open G, '>', $afile;
215
216         check([ PerlIO::get_layers(F, input  => 1) ],
217               [ qw(stdio crlf) ],
218               "use open IN");
219         
220         check([ PerlIO::get_layers(G, output => 1) ],
221               [ qw[stdio encoding(cp1252) utf8] ],
222               "use open OUT");
223
224         close F;
225         close G;
226     }
227
228     # Check that PL_sigwarn's reference count is correct, and that 
229     # &PerlIO::Layer::NoWarnings isn't prematurely freed.
230     fresh_perl_like (<<"EOT", qr/^CODE/);
231 open(UTF, "<:raw:encoding(utf8)", '$afile') or die \$!;
232 print ref *PerlIO::Layer::NoWarnings{CODE};
233 EOT
234 }