diff options
-rwxr-xr-x | Configure | 2 | ||||
-rw-r--r-- | ext/POSIX/hints/dynixptx.pl | 2 | ||||
-rwxr-xr-x | myconfig | 8 | ||||
-rwxr-xr-x | t/op/grent.t | 56 | ||||
-rwxr-xr-x | t/op/pwent.t | 49 | ||||
-rwxr-xr-x | t/op/undef.t | 5 | ||||
-rwxr-xr-x | t/pragma/locale.t | 526 | ||||
-rw-r--r-- | util.c | 56 |
8 files changed, 421 insertions, 283 deletions
@@ -5207,7 +5207,7 @@ EOM To build perl, you must add the current working directory to your $xxx environment variable before running make. You can do this with - $xxx=\`pwd\`; export $xxx + $xxx=\`pwd\`:\$$xxx; export $xxx for Bourne-style shells, or setenv $xxx \`pwd\` for Csh-style shells. You *MUST* do this before running make. diff --git a/ext/POSIX/hints/dynixptx.pl b/ext/POSIX/hints/dynixptx.pl index 05cf0f8765..9b63684382 100644 --- a/ext/POSIX/hints/dynixptx.pl +++ b/ext/POSIX/hints/dynixptx.pl @@ -1,4 +1,4 @@ # Need to add an extra '-lc' to the end to work around a DYNIX/ptx bug # PR#227670 - linker error on fpgetround() -$self->{LIBS} = ['-ldb -lc']; +$self->{LIBS} = ['-ldb -lm -lc']; @@ -23,15 +23,7 @@ Summary of my $package ($baserev patchlevel $PATCHLEVEL subversion $SUBVERSION) uname='$myuname' hint=$hint, useposix=$useposix, d_sigaction=$d_sigaction usethreads=$usethreads useperlio=$useperlio d_sfio=$d_sfio -!GROK!THIS! -case "$use64bits$usemultiplicity" in -*define*) - $spitshell <<!GROK!THIS! use64bits=$use64bits usemultiplicity=$usemultiplicity -!GROK!THIS! - ;; -esac -$spitshell <<!GROK!THIS! Compiler: cc='$cc', optimize='$optimize', gccversion=$gccversion cppflags='$cppflags' diff --git a/t/op/grent.t b/t/op/grent.t index 9d2b01d51a..e0cd7a82c5 100755 --- a/t/op/grent.t +++ b/t/op/grent.t @@ -7,10 +7,18 @@ BEGIN { my $GR = "/etc/group"; - if (($^O eq 'next' and not open(GR, "nidump group .|")) - or (defined $Config{'i_grp'} and $Config{'i_grp'} ne 'define') - or not -f $GR or not open(GR, $GR) - ) { + $where = $GR; + + if (-x "/usr/bin/nidump") { + if (open(GR, "nidump group . |")) { + $where = "NetInfo"; + } else { + print "1..0\n"; + exit 0; + } + } elsif ((defined $Config{'i_grp'} and $Config{'i_grp'} ne 'define') + or not -f $GR or not open(GR, $GR) + ) { print "1..0\n"; exit 0; } @@ -19,19 +27,27 @@ BEGIN { print "1..1\n"; # Go through at most this many groups. -my $max = 25; # +my $max = 25; -my $n = 0; -my $not; +my $n = 0; my $tst = 1; +my %suspect; +my %seen; -$not = 0; while (<GR>) { - last if $n == $max; chomp; - @s = split /:/; + my @s = split /:/; + my ($name_s,$passwd_s,$gid_s,$members_s) = @s; + if (@s) { + push @{ $seen{$name_s} }, $.; + } else { + warn "# Your $where line $. is empty.\n"; + next; + } + next if $n == $max; + # In principle we could whine if @s != 4 but do we know enough + # of group file formats everywhere? if (@s == 4) { - my ($name_s,$passwd_s,$gid_s,$members_s) = @s; $members_s =~ s/\s*,\s*/,/g; $members_s =~ s/\s+$//; $members_s =~ s/^\s+//; @@ -46,10 +62,10 @@ while (<GR>) { next if $name_s ne $name; } $members =~ s/\s+/,/g; - $not = 1, last + $suspect{$name_s}++ if $name ne $name_s or # Shadow passwords confuse this. -# Not that group passwords are used much but still. +# Not that group passwords are used much but better not assume anything. # $passwd ne $passwd_s or $gid ne $gid_s or $members ne $members_s; @@ -57,7 +73,19 @@ while (<GR>) { $n++; } -print "not " if $not; +# Drop the multiply defined groups. + +foreach (sort keys %seen) { + my $times = @{ $seen{$_} }; + if ($times > 1) { + # Multiply defined groups are rarely intentional. + local $" = ", "; + warn "# Group '$_' defined multiple times in $where, lines: @{$seen{$_}}.\n"; + delete $suspect{$_}; + } +} + +print "not " if keys %suspect; print "ok ", $tst++, "\n"; close(GR); diff --git a/t/op/pwent.t b/t/op/pwent.t index 87b2ac1f78..1365588004 100755 --- a/t/op/pwent.t +++ b/t/op/pwent.t @@ -7,10 +7,17 @@ BEGIN { my $PW = "/etc/passwd"; - if (($^O eq 'next' and not open(PW, "nidump passwd .|")) - or (defined $Config{'i_pwd'} and $Config{'i_pwd'} ne 'define') - or not -f $PW or not open(PW, $PW) - ) { + $where = $PW; + + if (-x "/usr/bin/nidump") { + if (open(PW, "nidump passwd . |")) { + $where = "NetInfo"; + } else { + print "1..0\n"; + exit 0; + } + } elsif ((defined $Config{'i_pwd'} and $Config{'i_pwd'} ne 'define') + or not -f $PW or not open(PW, $PW)) { print "1..0\n"; exit 0; } @@ -22,16 +29,24 @@ print "1..1\n"; my $max = 25; # my $n = 0; -my $not; my $tst = 1; +my %suspect; +my %seen; -$not = 0; while (<PW>) { - last if $n == $max; chomp; - @s = split /:/; + my @s = split /:/; + my ($name_s, $passwd_s, $uid_s, $gid_s, $gcos_s, $home_s, $shell_s) = @s; + if (@s) { + push @{ $seen{$name_s} }, $.; + } else { + warn "# Your $where line $. is empty.\n"; + next; + } + next if $n == $max; + # In principle we could whine if @s != 7 but do we know enough + # of passwd file formats everywhere? if (@s == 7) { - my ($name_s, $passwd_s, $uid_s, $gid_s, $gcos_s, $home_s, $shell_s) = @s; @n = getpwuid($uid_s); # 'nobody' et al. next unless @n; @@ -42,7 +57,7 @@ while (<PW>) { ($name,$passwd,$uid,$gid,$quota,$comment,$gcos,$home,$shell) = @n; next if $name_s ne $name; } - $not = 1, last + $suspect{$name_s}++ if $name ne $name_s or # Shadow passwords confuse this. # Think about non-crypt(3) encryptions, too, before you do anything rash. @@ -56,7 +71,19 @@ while (<PW>) { $n++; } -print "not " if $not; +# Drop the multiply defined users. + +foreach (sort keys %seen) { + my $times = @{ $seen{$_} }; + if ($times > 1) { + # Multiply defined users are rarely intentional. + local $" = ", "; + warn "# User '$_' defined multiple times in $where, lines: @{$seen{$_}}.\n"; + delete $suspect{$_}; + } +} + +print "not " if keys %suspect; print "ok ", $tst++, "\n"; close(PW); diff --git a/t/op/undef.t b/t/op/undef.t index 3bfe1a3118..8944ee3976 100755 --- a/t/op/undef.t +++ b/t/op/undef.t @@ -1,5 +1,10 @@ #!./perl +BEGIN { + chdir 't' if -d 't'; + unshift @INC, '../lib'; +} + print "1..27\n"; print defined($a) ? "not ok 1\n" : "ok 1\n"; diff --git a/t/pragma/locale.t b/t/pragma/locale.t index 7e3df8c3f1..b53a22809a 100755 --- a/t/pragma/locale.t +++ b/t/pragma/locale.t @@ -2,7 +2,7 @@ BEGIN { chdir 't' if -d 't'; - @INC = '../lib'; + unshift @INC, '../lib'; require Config; import Config; if (!$Config{d_setlocale} || $Config{ccflags} =~ /\bD?NO_LOCALE\b/) { print "1..0\n"; @@ -12,6 +12,8 @@ BEGIN { use strict; +my $debug = 1; + my $have_setlocale = 0; eval { require POSIX; @@ -19,19 +21,21 @@ eval { $have_setlocale++; }; +use vars qw(&LC_ALL); + # Visual C's CRT goes silly on strings of the form "en_US.ISO8859-1" # and mingw32 uses said silly CRT $have_setlocale = 0 if $^O eq 'MSWin32' && $Config{cc} =~ /^(cl|gcc)/i; -# 103 (the last test) may fail but that is okay. +# 103 (the last test) may fail but that is sort-of okay. # (It indicates something broken in the environment, not Perl) -# Therefore .. only until 102, not 103. -print "1..", ($have_setlocale ? 102 : 98), "\n"; + +print "1..", ($have_setlocale ? 103 : 98), "\n"; use vars qw($a $English $German $French $Spanish @C @English @German @French @Spanish - $Locale @Locale %iLocale %UPPER %lower @Neoalpha); + $Locale @Locale %UPPER %lower %bothcase @Neoalpha); $a = 'abc %'; @@ -219,269 +223,363 @@ check_taint_not 98, $a; # I think we've seen quite enough of taint. # Let us do some *real* locale work now, -# unless setlocale() is missing (i.e. minitest). +# unless setlocale() is missing (i.e. minitest). exit unless $have_setlocale; -sub getalnum { +# Find locales. + +my $locales = <<EOF; +Arabic:ar:dz eg sa:6 arabic8 +Bulgarian:bg:bg:5 +Chinese:zh:cn tw:cn.EUC eucCN eucTW euc.CN euc.TW tw.EUC +Croation:hr:hr:2 +Czech:cs:cz:2 +Danish:dk:da:1 +Danish:dk:da:1 +Dutch:nl:nl:1 +English American British:en:au ca gb ie nz us uk:1 cp850 +Estonian:et:ee:1 +Finnish:fi:fi:1 +French:fr:be ca ch fr:1 +German:de:de at ch:1 +Greek:el:gr:7 g8 +Hebrew:iw:il:8 hebrew8 +Hungarian:hu:hu:2 +Icelandic:is:is:1 +Italian:it:it:1 +Japanese:ja:jp:euc eucJP jp.EUC sjis +Korean:ko:kr: +Latin:la:va:1 +Latvian:lv:lv:1 +Lithuanian:lt:lt:1 +Polish:pl:pl:2 +Portuguese:po:po br:1 +Rumanian:ro:ro:2 +Russian:ru:ru su:5 koi8 koi8r koi8u cp1251 +Slovak:sk:sk:2 +Slovene:sl:si:2 +Spanish:es:ar bo cl co cr ec es gt mx ni pa pe py sv uy ve:1 +Swedish:sv:se:1 +Thai:th:th:tis620 +Turkish:tr:tr:9 turkish8 +EOF + +my @Locale; +my $Locale; +my @Alnum_; + +sub getalnum_ { sort grep /\w/, map { chr } 0..255 } -sub locatelocale ($$@) { - my ($lcall, $alnum, @try) = @_; +sub trylocale { + my $locale = shift; + if (setlocale(LC_ALL, $locale)) { + push @Locale, $locale; + } +} - undef $$lcall; +sub decode_encodings { + my @enc; - for (@try) { - local $^W = 0; # suppress "Subroutine LC_ALL redefined" - if (setlocale(&LC_ALL, $_)) { - $$lcall = $_; - @$alnum = &getalnum; - last; + foreach (split(/ /, shift)) { + if (/^(\d+)$/) { + push @enc, "ISO8859-$1"; + push @enc, "iso8859$1"; # HP + if ($1 eq '1') { + push @enc, "roman8"; # HP + } + } else { + push @enc, $_; } } - @$alnum = () unless (defined $$lcall); + return @enc; } -# Find some default locale - -locatelocale(\$Locale, \@Locale, qw(C POSIX)); - -# Find some English locale - -locatelocale(\$English, \@English, - qw(en_US.ISO8859-1 en_GB.ISO8859-1 - en en_US en_UK en_IE en_CA en_AU en_NZ - english english.iso88591 - american american.iso88591 - british british.iso88591 - )); - -# Find some German locale - -locatelocale(\$German, \@German, - qw(de_DE.ISO8859-1 de_AT.ISO8859-1 de_CH.ISO8859-1 - de de_DE de_AT de_CH - german german.iso88591)); - -# Find some French locale - -locatelocale(\$French, \@French, - qw(fr_FR.ISO8859-1 fr_BE.ISO8859-1 fr_CA.ISO8859-1 fr_CH.ISO8859-1 - fr fr_FR fr_BE fr_CA fr_CH - french french.iso88591)); - -# Find some Spanish locale - -locatelocale(\$Spanish, \@Spanish, - qw(es_AR.ISO8859-1 es_BO.ISO8859-1 es_CL.ISO8859-1 - es_CO.ISO8859-1 es_CR.ISO8859-1 es_EC.ISO8859-1 - es_ES.ISO8859-1 es_GT.ISO8859-1 es_MX.ISO8859-1 - es_NI.ISO8859-1 es_PA.ISO8859-1 es_PE.ISO8859-1 - es_PY.ISO8859-1 es_SV.ISO8859-1 es_UY.ISO8859-1 es_VE.ISO8859-1 - es es_AR es_BO es_CL - es_CO es_CR es_EC - es_ES es_GT es_MX - es_NI es_PA es_PE - es_PY es_SV es_UY es_VE - spanish spanish.iso88591)); - -# Select the largest of the alpha(num)bets. - -($Locale, @Locale) = ($English, @English) - if (@English > @Locale); -($Locale, @Locale) = ($German, @German) - if (@German > @Locale); -($Locale, @Locale) = ($French, @French) - if (@French > @Locale); -($Locale, @Locale) = ($Spanish, @Spanish) - if (@Spanish > @Locale); - -{ - local $^W = 0; - setlocale(&LC_ALL, $Locale); +trylocale("C"); +trylocale("POSIX"); +foreach (0..15) { + trylocale("ISO8859-$_"); + trylocale("iso_8859_$_"); + trylocale("iso8859$_"); } -# Sort it now that LC_ALL has been set. +foreach my $locale (split(/\n/, $locales)) { + my ($locale_name, $language_codes, $country_codes, $encodings) = + split(/:/, $locale); + my @enc = decode_encodings($encodings); + foreach my $loc (split(/ /, $locale_name)) { + trylocale($loc); + foreach my $enc (@enc) { + trylocale("$loc.$enc"); + } + $loc = lc $loc; + foreach my $enc (@enc) { + trylocale("$loc.$enc"); + } + } + foreach my $lang (split(/ /, $language_codes)) { + trylocale($lang); + foreach my $country (split(/ /, $country_codes)) { + my $lc = "${lang}_${country}"; + trylocale($lc); + foreach my $enc (@enc) { + trylocale("$lc.$enc"); + } + my $lC = "${lang}_\U${country}"; + trylocale($lC); + foreach my $enc (@enc) { + trylocale("$lC.$enc"); + } + } + } +} @Locale = sort @Locale; -print "# Locale = $Locale\n"; -print "# Alnum_ = @Locale\n"; - -{ - my $i = 0; +sub debug { + print @_ if $debug; +} - for (@Locale) { - $iLocale{$_} = $i++; - } +sub debugf { + printf @_ if $debug; } -# Sieve the uppercase and the lowercase. +debug "# Locales = @Locale\n"; -for (@Locale) { - if (/[^\d_]/) { # skip digits and the _ - if (lc eq $_) { - $UPPER{$_} = uc; - } else { - $lower{$_} = lc; +my %Problem; + +foreach $Locale (@Locale) { + debug "# Locale = $Locale\n"; + @Alnum_ = getalnum_(); + debug "# \\w = @Alnum_\n"; + + unless (setlocale(LC_ALL, $Locale)) { + foreach (99..103) { + $Problem{$_}{$Locale} = -1; } + next; } -} -# Find the alphabets that are not alphabets in the default locale. + # Sieve the uppercase and the lowercase. + + %UPPER = %lower = %bothcase = (); + for (@Alnum_) { + if (/[^\d_]/) { # skip digits and the _ + if (uc($_) eq $_) { + $UPPER{$_} = $_; + } + if (lc($_) eq $_) { + $lower{$_} = $_; + } + } + } + foreach (keys %UPPER) { + $bothcase{$_}++ if exists $lower{$_}; + } + foreach (keys %lower) { + $bothcase{$_}++ if exists $UPPER{$_}; + } + foreach (keys %bothcase) { + delete $UPPER{$_}; + delete $lower{$_}; + } + + debug "# UPPER = ", join(" ", sort keys %UPPER ), "\n"; + debug "# lower = ", join(" ", sort keys %lower ), "\n"; + debug "# bothcase = ", join(" ", sort keys %bothcase), "\n"; + + # Find the alphabets that are not alphabets in the default locale. -{ - no locale; + { + no locale; - for (keys %UPPER, keys %lower) { - push(@Neoalpha, $_) if (/\W/); + @Neoalpha = (); + for (keys %UPPER, keys %lower) { + push(@Neoalpha, $_) if (/\W/); + } } -} -@Neoalpha = sort @Neoalpha; + @Neoalpha = sort @Neoalpha; -# Test \w. + debug "# Neoalpha = @Neoalpha\n"; -{ - my $word = join('', @Neoalpha); + if (@Neoalpha == 0) { + # If we have no Neoalphas the remaining tests are no-ops. + debug "# no Neoalpha, skipping tests 99..103 for locale '$Locale'\n"; + next; + } - $word =~ /^(\w*)$/; + # Test \w. + + debug "# testing 99 with locale '$Locale'\n"; + { + my $word = join('', @Neoalpha); - print 'not ' if ($1 ne $word); -} -print "ok 99\n"; + $word =~ /^(\w+)$/; -# Find places where the collation order differs from the default locale. + if ($1 ne $word) { + $Problem{99}{$Locale} = 1; + debug "# failed 99 ($1 vs $word)\n"; + } + } -print "# testing 100\n"; -{ - my (@k, $i, $j, @d); + # Test #100 removed but to preserve historical test number + # consistency we do not renumber the remaining tests. - { - no locale; + # Cross-check whole character set. - @k = sort (keys %UPPER, keys %lower); + debug "# testing 101 with locale '$Locale'\n"; + for (map { chr } 0..255) { + if ((/\w/ and /\W/) or (/\d/ and /\D/) or (/\s/ and /\S/)) { + $Problem{101}{$Locale} = 1; + debug "# failed 101\n"; + last; + } } - for ($i = 0; $i < @k; $i++) { - for ($j = $i + 1; $j < @k; $j++) { - if ($iLocale{$k[$j]} < $iLocale{$k[$i]}) { - push(@d, [$k[$j], $k[$i]]); + # Test for read-only scalars' locale vs non-locale comparisons. + + debug "# testing 102 with locale '$Locale'\n"; + { + no locale; + $a = "qwerty"; + { + use locale; + if ($a cmp "qwerty") { + $Problem{102}{$Locale} = 1; + debug "# failed 102\n"; } } } - # Cross-check those places. - - for (@d) { - ($i, $j) = @$_; - if ($i gt $j) { - print "# failed 100 at:\n"; - print "# i = $i, j = $j, i ", - $i le $j ? 'le' : 'gt', " j\n"; - print 'not '; - last; + # This test must be the last one because its failure is not fatal. + # The @Alnum_ should be internally consistent. + # Thanks to Hallvard Furuseth <h.b.furuseth@usit.uio.no> + # for inventing a way to test for ordering consistency + # without requiring any particular order. + # <jhi@iki.fi> + + debug "# testing 103 with locale '$Locale'\n"; + { + my ($from, $to, $lesser, $greater, + @test, %test, $test, $yes, $no, $sign); + + for (0..9) { + # Select a slice. + $from = int(($_*@Alnum_)/10); + $to = $from + int(@Alnum_/10); + $to = $#Alnum_ if ($to > $#Alnum_); + $lesser = join('', @Alnum_[$from..$to]); + # Select a slice one character on. + $from++; $to++; + $to = $#Alnum_ if ($to > $#Alnum_); + $greater = join('', @Alnum_[$from..$to]); + ($yes, $no, $sign) = ($lesser lt $greater + ? (" ", "not ", 1) + : ("not ", " ", -1)); + # all these tests should FAIL (return 0). + # Exact lt or gt cannot be tested because + # in some locales, say, eacute and E may test equal. + @test = + ( + $no.' ($lesser le $greater)', # 1 + 'not ($lesser ne $greater)', # 2 + ' ($lesser eq $greater)', # 3 + $yes.' ($lesser ge $greater)', # 4 + $yes.' ($lesser ge $greater)', # 5 + $yes.' ($greater le $lesser )', # 7 + 'not ($greater ne $lesser )', # 8 + ' ($greater eq $lesser )', # 9 + $no.' ($greater ge $lesser )', # 10 + 'not (($lesser cmp $greater) == -$sign)' # 12 + ); + @test{@test} = 0 x @test; + $test = 0; + for my $ti (@test) { $test{$ti} = eval $ti ; $test ||= $test{$ti} } + if ($test) { + $Problem{103}{$Locale} = 1; + debug "# failed 103 at:\n"; + debug "# lesser = '$lesser'\n"; + debug "# greater = '$greater'\n"; + debug "# lesser cmp greater = ", $lesser cmp $greater, "\n"; + debug "# greater cmp lesser = ", $greater cmp $lesser, "\n"; + debug "# (greater) from = $from, to = $to\n"; + for my $ti (@test) { + debugf("# %-40s %-4s", $ti, + $test{$ti} ? 'FAIL' : 'ok'); + if ($ti =~ /\(\.*(\$.+ +cmp +\$[^\)]+)\.*\)/) { + debugf("(%s == %4d)", $1, eval $1); + } + debug "\n#"; + } + + last; + } } } } -print "ok 100\n"; - -# Cross-check whole character set. - -print "# testing 101\n"; -for (map { chr } 0..255) { - if (/\w/ and /\W/) { print 'not '; last } - if (/\d/ and /\D/) { print 'not '; last } - if (/\s/ and /\S/) { print 'not '; last } - if (/\w/ and /\D/ and not /_/ and - not (exists $UPPER{$_} or exists $lower{$_})) { - print "# failed 101 at:\n"; - print "# ", ord($_), " '$_'\n"; - print 'not '; - last; + +no locale; + +foreach (99..103) { + if ($Problem{$_}) { + if ($_ == 103) { + print "# The failure of test 103 is not necessarily fatal.\n"; + print "# It usually indicates a problem in the enviroment,\n"; + print "# not in Perl itself.\n"; + } + print "not "; } + print "ok $_\n"; } -print "ok 101\n"; - -# Test for read-onlys. -print "# testing 102\n"; -{ - no locale; - $a = "qwerty"; - { - use locale; - print "not " if $a cmp "qwerty"; +my $didwarn = 0; + +foreach (99..103) { + if ($Problem{$_}) { + my @f = sort keys %{ $Problem{$_} }; + my $f = join(" ", @f); + $f =~ s/(.{50,60}) /$1\n#\t/g; + warn + "# The locale ", (@f == 1 ? "definition" : "definitions"), "\n#\n", + "#\t", $f, "\n#\n", + "# on your system may have errors because the locale test $_\n", + "# failed in ", (@f == 1 ? "that locale" : "those locales"), + ".\n"; + warn <<EOW; +# +# If your users are not using these locales you are safe for the moment, +# but please report this failure first to perlbug\@perl.com using the +# perlbug script (as described in the INSTALL file) so that the exact +# details of the failures can be sorted out first and then your operating +# system supplier can be alerted about these anomalies. +# +EOW + $didwarn = 1; } } -print "ok 102\n"; - -# This test must be the last one because its failure is not fatal. -# The @Locale should be internally consistent. -# Thanks to Hallvard Furuseth <h.b.furuseth@usit.uio.no> -# for inventing a way to test for ordering consistency -# without requiring any particular order. -# <jhi@iki.fi> - -print "# testing 103\n"; -{ - my ($from, $to, $lesser, $greater, @test, %test, $test, $yes, $no, $sign); - - for (0..9) { - # Select a slice. - $from = int(($_*@Locale)/10); - $to = $from + int(@Locale/10); - $to = $#Locale if ($to > $#Locale); - $lesser = join('', @Locale[$from..$to]); - # Select a slice one character on. - $from++; $to++; - $to = $#Locale if ($to > $#Locale); - $greater = join('', @Locale[$from..$to]); - ($yes, $no, $sign) = ($lesser lt $greater - ? (" ", "not ", 1) - : ("not ", " ", -1)); - # all these tests should FAIL (return 0). - @test = - ( - $no.' ($lesser lt $greater)', # 0 - $no.' ($lesser le $greater)', # 1 - 'not ($lesser ne $greater)', # 2 - ' ($lesser eq $greater)', # 3 - $yes.' ($lesser ge $greater)', # 4 - $yes.' ($lesser gt $greater)', # 5 - $yes.' ($greater lt $lesser )', # 6 - $yes.' ($greater le $lesser )', # 7 - 'not ($greater ne $lesser )', # 8 - ' ($greater eq $lesser )', # 9 - $no.' ($greater ge $lesser )', # 10 - $no.' ($greater gt $lesser )', # 11 - 'not (($lesser cmp $greater) == -$sign)' # 12 - ); - @test{@test} = 0 x @test; - $test = 0; - for my $ti (@test) { $test{$ti} = eval $ti ; $test ||= $test{$ti} } - if ($test) { - print "# failed 103 at:\n"; - print "# lesser = '$lesser'\n"; - print "# greater = '$greater'\n"; - print "# lesser cmp greater = ", $lesser cmp $greater, "\n"; - print "# greater cmp lesser = ", $greater cmp $lesser, "\n"; - print "# (greater) from = $from, to = $to\n"; - for my $ti (@test) { - printf("# %-40s %-4s", $ti, - $test{$ti} ? 'FAIL' : 'ok'); - if ($ti =~ /\(\.*(\$.+ +cmp +\$[^\)]+)\.*\)/) { - printf("(%s == %4d)", $1, eval $1); - } - print "\n"; - } - warn "The locale definition on your system may have errors.\n"; - last; +if ($didwarn) { + my @s; + + foreach my $l (@Locale) { + my $p = 0; + foreach my $t (99..103) { + $p++ if $Problem{$t}{$l}; } + push @s, $l if $p == 0; } + + my $s = join(" ", @s); + $s =~ s/(.{50,60}) /$1\n#\t/g; + + warn + "# The following locales\n#\n", + "#\t", $s, "\n#\n", + "# tested okay.\n#\n", } # eof @@ -641,65 +641,53 @@ perl_init_i18nl10n(int printwarn) else setlocale_failure = TRUE; } - if (!setlocale_failure) -#endif /* LC_ALL */ - { + if (!setlocale_failure) { #ifdef USE_LOCALE_CTYPE - if (! (curctype = setlocale(LC_CTYPE, - (!done && (lang || PerlEnv_getenv("LC_CTYPE"))) + if (! (curctype = + setlocale(LC_CTYPE, + (!done && (lang || PerlEnv_getenv("LC_CTYPE"))) ? "" : Nullch))) setlocale_failure = TRUE; #endif /* USE_LOCALE_CTYPE */ #ifdef USE_LOCALE_COLLATE - if (! (curcoll = setlocale(LC_COLLATE, - (!done && (lang || PerlEnv_getenv("LC_COLLATE"))) + if (! (curcoll = + setlocale(LC_COLLATE, + (!done && (lang || PerlEnv_getenv("LC_COLLATE"))) ? "" : Nullch))) setlocale_failure = TRUE; #endif /* USE_LOCALE_COLLATE */ #ifdef USE_LOCALE_NUMERIC - if (! (curnum = setlocale(LC_NUMERIC, - (!done && (lang || PerlEnv_getenv("LC_NUMERIC"))) + if (! (curnum = + setlocale(LC_NUMERIC, + (!done && (lang || PerlEnv_getenv("LC_NUMERIC"))) ? "" : Nullch))) setlocale_failure = TRUE; #endif /* USE_LOCALE_NUMERIC */ } -#else /* !LOCALE_ENVIRON_REQUIRED */ +#endif /* LC_ALL */ -#ifdef LC_ALL +#endif /* !LOCALE_ENVIRON_REQUIRED */ +#ifdef LC_ALL if (! setlocale(LC_ALL, "")) setlocale_failure = TRUE; - else { -#ifdef USE_LOCALE_CTYPE - curctype = setlocale(LC_CTYPE, Nullch); -#endif /* USE_LOCALE_CTYPE */ -#ifdef USE_LOCALE_COLLATE - curcoll = setlocale(LC_COLLATE, Nullch); -#endif /* USE_LOCALE_COLLATE */ -#ifdef USE_LOCALE_NUMERIC - curnum = setlocale(LC_NUMERIC, Nullch); -#endif /* USE_LOCALE_NUMERIC */ - } - -#else /* !LC_ALL */ +#endif /* LC_ALL */ + if (!setlocale_failure) { #ifdef USE_LOCALE_CTYPE - if (! (curctype = setlocale(LC_CTYPE, ""))) - setlocale_failure = TRUE; + if (! (curctype = setlocale(LC_CTYPE, ""))) + setlocale_failure = TRUE; #endif /* USE_LOCALE_CTYPE */ #ifdef USE_LOCALE_COLLATE - if (! (curcoll = setlocale(LC_COLLATE, ""))) - setlocale_failure = TRUE; + if (! (curcoll = setlocale(LC_COLLATE, ""))) + setlocale_failure = TRUE; #endif /* USE_LOCALE_COLLATE */ #ifdef USE_LOCALE_NUMERIC - if (! (curnum = setlocale(LC_NUMERIC, ""))) - setlocale_failure = TRUE; + if (! (curnum = setlocale(LC_NUMERIC, ""))) + setlocale_failure = TRUE; #endif /* USE_LOCALE_NUMERIC */ - -#endif /* LC_ALL */ - -#endif /* !LOCALE_ENVIRON_REQUIRED */ + } if (setlocale_failure) { char *p; |