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