1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
|
#!./perl -w
print "1..4016\n";
my $test = 0;
my %templates = (
'UTF-8' => 'C0U',
'UTF-16BE' => 'n',
'UTF-16LE' => 'v',
);
sub bytes_to_utf {
my ($enc, $content, $do_bom) = @_;
my $template = $templates{$enc};
die "Unsupported encoding $enc" unless $template;
my @chars = unpack "U*", $content;
if ($enc ne 'UTF-8') {
# Make surrogate pairs
my @remember_that_utf_16_is_variable_length;
foreach my $ord (@chars) {
if ($ord < 0x10000) {
push @remember_that_utf_16_is_variable_length,
$ord;
} else {
$ord -= 0x10000;
push @remember_that_utf_16_is_variable_length,
(0xD800 | ($ord >> 10)), (0xDC00 | ($ord & 0x3FF));
}
}
@chars = @remember_that_utf_16_is_variable_length;
}
return pack "$template*", ($do_bom ? 0xFEFF : ()), @chars;
}
sub test {
my ($enc, $write, $expect, $bom, $nl, $name) = @_;
open my $fh, ">", "utf$$.pl" or die "utf.pl: $!";
binmode $fh;
print $fh bytes_to_utf($enc, $write . ($nl ? "\n" : ''), $bom);
close $fh or die $!;
my $got = do "./utf$$.pl";
$test = $test + 1;
if (!defined $got) {
if ($@ =~ /^(Unsupported script encoding \Q$enc\E)/) {
print "ok $test # skip $1\n";
} else {
print "not ok $test # $enc $bom $nl $name; got undef\n";
}
} elsif ($got ne $expect) {
print "not ok $test # $enc $bom $nl $name; got '$got'\n";
} else {
print "ok $test # $enc $bom $nl $name\n";
}
}
for my $bom (0, 1) {
for my $enc (qw(UTF-16LE UTF-16BE UTF-8)) {
for my $nl (1, 0) {
for my $value (123, 1234, 12345) {
test($enc, $value, $value, $bom, $nl, $value);
# This has the unfortunate side effect of causing an infinite
# loop without the bug fix it corresponds to:
test($enc, "($value)", $value, $bom, $nl, "($value)");
}
next if $enc eq 'UTF-8';
# Arguably a bug that currently string literals from UTF-8 file
# handles are not implicitly "use utf8", but don't FIXME that
# right now, as here we're testing the input filter itself.
for my $expect ("N", "\xFF", "\x{100}", "\x{010a}", "\x{0a23}",
"\x{10000}", "\x{64321}", "\x{10FFFD}",
"\x{1000a}", # 0xD800 0xDC0A
"\x{12800}", # 0xD80A 0xDC00
) {
# A space so that the UTF-16 heuristic triggers - " '" gives two
# characters of ASCII.
my $write = " '$expect'";
my $name = 'chrs ' . join ', ', map {ord $_} split '', $expect;
test($enc, $write, $expect, $bom, $nl, $name);
}
# This is designed to try to trip over the end of the buffer,
# with similar results to U-1000A and U-12800 above.
for my $pad (2 .. 162) {
for my $chr ("\x{10000}", "\x{1000a}", "\x{12800}") {
my $padding = ' ' x $pad;
# Need 4 octets that were from 2 ASCII characters to trigger
# the heuristic that detects UTF-16 without a BOM. For
# UTF-16BE, one space and the newline will do, as the
# newline's high octet comes first. But for UTF-16LE, a
# newline is "\n\0", so it doesn't trigger it.
test($enc, " \n$padding'$chr'", $chr, $bom, $nl,
sprintf "'\\x{%x}' with $pad spaces before it", ord $chr);
}
}
}
}
}
END {
1 while unlink "utf$$.pl";
}
|