diff options
author | Karl Williamson <public@khwilliamson.com> | 2014-01-27 15:35:00 -0700 |
---|---|---|
committer | Karl Williamson <public@khwilliamson.com> | 2014-01-27 23:03:48 -0700 |
commit | 31f05a37c4e9c37a7263491f2fc0237d836e1a80 (patch) | |
tree | 7537c7e179350243b3de0f3a99d6747c9c7812e6 /ext | |
parent | cea315b64e0c4b1890867df0c925cafc8823ba38 (diff) | |
download | perl-31f05a37c4e9c37a7263491f2fc0237d836e1a80.tar.gz |
Work properly under UTF-8 LC_CTYPE locales
This large (sorry, I couldn't figure out how to meaningfully split it
up) commit causes Perl to fully support LC_CTYPE operations (case
changing, character classification) in UTF-8 locales.
As a side effect it resolves [perl #56820].
The basics are easy, but there were a lot of details, and one
troublesome edge case discussed below.
What essentially happens is that when the locale is changed to a UTF-8
one, a global variable is set TRUE (FALSE when changed to a non-UTF-8
locale). Within the scope of 'use locale', this variable is checked,
and if TRUE, the code that Perl uses for non-locale behavior is used
instead of the code for locale behavior. Since Perl's internal
representation is UTF-8, we get UTF-8 behavior for a UTF-8 locale.
More work had to be done for regular expressions. There are three
cases.
1) The character classes \w, [[:punct:]] needed no extra work, as
the changes fall out from the base work.
2) Strings that are to be matched case-insensitively. These form
EXACTFL regops (nodes). Notice that if such a string contains only
characters above-Latin1 that match only themselves, that the node can be
downgraded to an EXACT-only node, which presents better optimization
possibilities, as we now have a fixed string known at compile time to be
required to be in the target string to match. Similarly if all
characters in the string match only other above-Latin1 characters
case-insensitively, the node can be downgraded to a regular EXACTFU node
(match, folding, using Unicode, not locale, rules). The code changes
for this could be done without accepting UTF-8 locales fully, but there
were edge cases which needed to be handled differently if I stopped
there, so I continued on.
In an EXACTFL node, all such characters are now folded at compile time
(just as before this commit), while the other characters whose folds are
locale-dependent are left unfolded. This means that they have to be
folded at execution time based on the locale in effect at the moment.
Again, this isn't a change from before. The difference is that now some
of the folds that need to be done at execution time (in regexec) are
potentially multi-char. Some of the code in regexec was trivial to
extend to account for this because of existing infrastructure, but the
part dealing with regex quantifiers, had to have more work.
Also the code that joins EXACTish nodes together had to be expanded to
account for the possibility of multi-character folds within locale
handling. This was fairly easy, because it already has infrastructure
to handle these under somewhat different circumstances.
3) In bracketed character classes, represented by ANYOF nodes, a new
inversion list was created giving the characters that should be matched
by this node when the runtime locale is UTF-8. The list is ignored
except under that circumstance. To do this, I created a new ANYOF type
which has an extra SV for the inversion list.
The edge case that caused the most difficulty is folding involving the
MICRO SIGN, U+00B5. It folds to the GREEK SMALL LETTER MU, as does the
GREEK CAPITAL LETTER MU. The MICRO SIGN is the only 0-255 range
character that folds to outside that range. The issue is that it
doesn't naturally fall out that it will match the CAP MU. If we let the
CAP MU fold to the samll mu at compile time (which it can because both
are above-Latin1 and so the fold is the same no matter what locale is in
effect), it could appear that the regnode can be downgraded away from
EXACTFL to EXACTFU, but doing so would cause the MICRO SIGN to not case
insensitvely match the CAP MU. This could be special cased in regcomp
and regexec, but I wanted to avoid that. Instead the mktables tables
are set up to include the CAP MU as a character whose presence forbids
the downgrading, so the special casing is in mktables, and not in the C
code.
Diffstat (limited to 'ext')
-rw-r--r-- | ext/XS-APItest/t/handy.t | 92 |
1 files changed, 86 insertions, 6 deletions
diff --git a/ext/XS-APItest/t/handy.t b/ext/XS-APItest/t/handy.t index 3a8abc9935..9a70ec2589 100644 --- a/ext/XS-APItest/t/handy.t +++ b/ext/XS-APItest/t/handy.t @@ -1,5 +1,9 @@ #!perl -w +BEGIN { + require 'loc_tools.pl'; # Contains find_utf8_locale() +} + use strict; use Test::More; use Config; @@ -13,6 +17,7 @@ sub truth($) { # Converts values so is() works } my $locale; +my $utf8_locale; if($Config{d_setlocale}) { require POSIX; $locale = POSIX::setlocale( &POSIX::LC_ALL, "C"); @@ -31,6 +36,8 @@ if($Config{d_setlocale}) { last; } } + + $utf8_locale = find_utf8_locale(); } } @@ -149,13 +156,30 @@ foreach my $name (sort keys %properties) { if (defined $locale) { require locale; import locale; + POSIX::setlocale( &POSIX::LC_ALL, "C"); $ret = truth eval "test_is${function}_LC($i)"; if ($@) { fail($@); } else { my $truth = truth($matches && $i < 128); - is ($ret, $truth, "is${function}_LC( $display_name ) == $truth"); + is ($ret, $truth, "is${function}_LC( $display_name ) == $truth (C locale)"); + } + } + + if (defined $utf8_locale) { + use locale; + + POSIX::setlocale( &POSIX::LC_ALL, $utf8_locale); + $ret = truth eval "test_is${function}_LC($i)"; + if ($@) { + fail($@); + } + else { + + # UTF-8 locale works on full range 0-255 + my $truth = truth($matches && $i < 256); + is ($ret, $truth, "is${function}_LC( $display_name ) == $truth ($utf8_locale)"); } } } @@ -171,13 +195,28 @@ foreach my $name (sort keys %properties) { if (defined $locale && $name ne 'vertws') { require locale; import locale; + POSIX::setlocale( &POSIX::LC_ALL, "C"); $ret = truth eval "test_is${function}_LC_uvchr('$i')"; if ($@) { fail($@); } else { my $truth = truth($matches && ($i < 128 || $i > 255)); - is ($ret, $truth, "is${function}_LC_uvchr( $display_name ) == $truth"); + is ($ret, $truth, "is${function}_LC_uvchr( $display_name ) == $truth (C locale)"); + } + } + + if (defined $utf8_locale && $name ne 'vertws') { + use locale; + + POSIX::setlocale( &POSIX::LC_ALL, $utf8_locale); + $ret = truth eval "test_is${function}_LC_uvchr('$i')"; + if ($@) { + fail($@); + } + else { + my $truth = truth($matches); + is ($ret, $truth, "is${function}_LC_uvchr( $display_name ) == $truth ($utf8_locale)"); } } @@ -195,13 +234,28 @@ foreach my $name (sort keys %properties) { if ($name ne 'vertws' && defined $locale) { require locale; import locale; + POSIX::setlocale( &POSIX::LC_ALL, "C"); $ret = truth eval "test_is${function}_LC_utf8('$char')"; if ($@) { fail($@); } else { my $truth = truth($matches && ($i < 128 || $i > 255)); - is ($ret, $truth, "is${function}_LC_utf8( $display_name ) == $truth"); + is ($ret, $truth, "is${function}_LC_utf8( $display_name ) == $truth (C locale)"); + } + } + + if ($name ne 'vertws' && defined $utf8_locale) { + use locale; + + POSIX::setlocale( &POSIX::LC_ALL, $utf8_locale); + $ret = truth eval "test_is${function}_LC_utf8('$char')"; + if ($@) { + fail($@); + } + else { + my $truth = truth($matches); + is ($ret, $truth, "is${function}_LC_utf8( $display_name ) == $truth ($utf8_locale)"); } } } @@ -292,10 +346,11 @@ foreach my $name (sort keys %to_properties) { } } - if ($name ne 'TITLE' && defined $locale) { + if ($name ne 'TITLE') { # Test _LC; titlecase is not defined in locales. + if (defined $locale) { require locale; import locale; - # Test _LC; titlecase is not defined in locales. + POSIX::setlocale( &POSIX::LC_ALL, "C"); $ret = eval "test_to${function}_LC($j)"; if ($@) { fail($@); @@ -304,7 +359,32 @@ foreach my $name (sort keys %to_properties) { my $should_be = ($i < 128 && $map_ref->[$index] != $missing) ? $map_ref->[$index] + $j - $list_ref->[$index] : $j; - is ($ret, $should_be, sprintf("to${function}_LC( $display_name ) == 0x%02X", $should_be)); + is ($ret, $should_be, sprintf("to${function}_LC( $display_name ) == 0x%02X (C locale)", $should_be)); + } + } + + if (defined $utf8_locale) { + use locale; + + SKIP: { + skip "to${property}_LC does not work for LATIN SMALL LETTER SHARP S", 1 + if $j == 0xDF && ($name eq 'FOLD' || $name eq 'UPPER'); + + POSIX::setlocale( &POSIX::LC_ALL, $utf8_locale); + $ret = eval "test_to${function}_LC($j)"; + if ($@) { + fail($@); + } + else { + my $should_be = ($i < 256 + && ! ref $map_ref->[$index] + && $map_ref->[$index] != $missing + ) + ? $map_ref->[$index] + $j - $list_ref->[$index] + : $j; + is ($ret, $should_be, sprintf("to${function}_LC( $display_name ) == 0x%02X ($utf8_locale)", $should_be)); + } + } } } |