| 1 | #!./perl
|
|---|
| 2 |
|
|---|
| 3 | my $PERLIO;
|
|---|
| 4 |
|
|---|
| 5 | BEGIN {
|
|---|
| 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 |
|
|---|
| 28 | use Config;
|
|---|
| 29 |
|
|---|
| 30 | my $DOSISH = $^O =~ /^(?:MSWin32|os2|dos|NetWare|mint)$/ ? 1 : 0;
|
|---|
| 31 | $DOSISH = 1 if !$DOSISH and $^O =~ /^uwin/;
|
|---|
| 32 | my $NONSTDIO = exists $ENV{PERLIO} && $ENV{PERLIO} ne 'stdio' ? 1 : 0;
|
|---|
| 33 | my $FASTSTDIO = $Config{d_faststdio} && $Config{usefaststdio} ? 1 : 0;
|
|---|
| 34 | my $UNICODE_STDIN;
|
|---|
| 35 | if (${^UNICODE} & 1) {
|
|---|
| 36 | if (${^UNICODE} & 64) {
|
|---|
| 37 | # Conditional on the locale
|
|---|
| 38 | $UNICODE_STDIN = ${^UTF8LOCALE};
|
|---|
| 39 | } else {
|
|---|
| 40 | # Unconditional
|
|---|
| 41 | $UNICODE_STDIN = 1;
|
|---|
| 42 | }
|
|---|
| 43 | } else {
|
|---|
| 44 | $UNICODE_STDIN = 0;
|
|---|
| 45 | }
|
|---|
| 46 | my $NTEST = 44 - (($DOSISH || !$FASTSTDIO) ? 7 : 0) - ($DOSISH ? 5 : 0)
|
|---|
| 47 | + $UNICODE_STDIN;
|
|---|
| 48 |
|
|---|
| 49 | sub PerlIO::F_UTF8 () { 0x00008000 } # from perliol.h
|
|---|
| 50 |
|
|---|
| 51 | plan tests => $NTEST;
|
|---|
| 52 |
|
|---|
| 53 | print <<__EOH__;
|
|---|
| 54 | # PERLIO = $PERLIO
|
|---|
| 55 | # DOSISH = $DOSISH
|
|---|
| 56 | # NONSTDIO = $NONSTDIO
|
|---|
| 57 | # FASTSTDIO = $FASTSTDIO
|
|---|
| 58 | # UNICODE = ${^UNICODE}
|
|---|
| 59 | # UTF8LOCALE = ${^UTF8LOCALE}
|
|---|
| 60 | # UNICODE_STDIN = $UNICODE_STDIN
|
|---|
| 61 | __EOH__
|
|---|
| 62 |
|
|---|
| 63 | SKIP: {
|
|---|
| 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 | $UNICODE_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(sjis)"); # "sjis" will be canonized to "shiftjis"
|
|---|
| 135 |
|
|---|
| 136 | check([ PerlIO::get_layers(F) ],
|
|---|
| 137 | [ qw[stdio crlf encoding(shiftjis) utf8] ],
|
|---|
| 138 | ":encoding(sjis)");
|
|---|
| 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/);
|
|---|
| 220 | open(UTF, "<:raw:encoding(utf8)", "afile") or die $!;
|
|---|
| 221 | print ref *PerlIO::Layer::NoWarnings{CODE};
|
|---|
| 222 | EOT
|
|---|
| 223 |
|
|---|
| 224 | 1 while unlink "afile";
|
|---|
| 225 | }
|
|---|