This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
4c3f99f6c8aef9936595682df8a4d64b072e8fba
[perl5.git] / lib / Unicode / UCD.t
1 #!perl -w
2 BEGIN {
3     $::IS_ASCII = (ord("A") == 65) ? 1 : 0;
4     $::IS_EBCDIC = (ord("A") == 193) ? 1 : 0;
5     chdir 't' if -d 't';
6     @INC = '../lib';
7     require Config; import Config;
8     if ($Config{'extensions'} !~ /\bStorable\b/) {
9         print "1..0 # Skip: Storable was not built; Unicode::UCD uses Storable\n";
10         exit 0;
11     }
12 }
13
14 my @warnings;
15 local $SIG{__WARN__} = sub { push @warnings, @_  };
16
17 use strict;
18 use Test::More;
19
20 use Unicode::UCD qw(charinfo charprop charprops_all);
21
22 my $input_record_separator = 7; # Make sure Unicode::UCD isn't affected by
23 $/ = $input_record_separator;   # setting this.
24
25 my $charinfo;
26
27 is(charinfo(0x110000), undef, "Verify charinfo() of non-unicode is undef");
28 is(charprop(0x110000, 'age'), "Unassigned", "Verify charprop(age) of non-unicode is Unassigned");
29 is(charprop(0x110000, 'in'), "Unassigned", "Verify charprop(in), a bipartite Perl extension, works");
30 is(charprop(0x110000, 'Any'), undef, "Verify charprop of non-bipartite Perl extension returns undef");
31
32 my $cp = 0;
33 $charinfo = charinfo($cp);    # Null is often problematic, so test it.
34
35 is($charinfo->{code},           "0000",
36                         "Next tests are for charinfo and charprop; first NULL");
37 is($charinfo->{name},           "<control>");
38 is(charprop($cp, "name"),       "");
39
40 # This gets a sl-type property returning a flattened list
41 is(charprop($cp, "name_alias"), "NULL: control,NUL: abbreviation");
42
43 is($charinfo->{category},       "Cc");
44 is(charprop($cp, "category"),   "Control");
45 is($charinfo->{combining},      "0");
46 is(charprop($cp, "ccc"),        "Not_Reordered");
47 is($charinfo->{bidi},           "BN");
48 is(charprop($cp, "bc"),         "Boundary_Neutral");
49 is($charinfo->{decomposition},  "");
50 is(charprop($cp, "dm"),         "\0");
51 is($charinfo->{decimal},        "");
52 is($charinfo->{digit},          "");
53 is($charinfo->{numeric},        "");
54 is(charprop($cp, "nv"),         "NaN");
55 is($charinfo->{mirrored},       "N");
56 is(charprop($cp, "bidim"),      "No");
57 is($charinfo->{unicode10},      "NULL");
58 is(charprop($cp, "na1"),        "NULL");
59 is($charinfo->{comment},        "");
60 is(charprop($cp, "isc"),        "");
61 is($charinfo->{upper},          "");
62 is(charprop($cp, "uc"),         "\0");
63 is($charinfo->{lower},          "");
64 is(charprop($cp, "lc"),         "\0");
65 is($charinfo->{title},          "");
66 is(charprop($cp, "tc"),         "\0");
67 is($charinfo->{block},          "Basic Latin");
68 is(charprop($cp, "block"),      "Basic_Latin");
69 is($charinfo->{script},         "Common");
70 is(charprop($cp, "script"),     "Common");
71
72 $cp = utf8::unicode_to_native(0x41);
73 my $A_code = sprintf("%04X", ord("A"));
74 my $a_code = sprintf("%04X", ord("a"));
75 $charinfo = charinfo($cp);
76
77 is($charinfo->{code},           $A_code, "LATIN CAPITAL LETTER A");
78 is($charinfo->{name},           "LATIN CAPITAL LETTER A");
79 is(charprop($cp, 'name'),       "LATIN CAPITAL LETTER A");
80 is($charinfo->{category},       "Lu");
81 is(charprop($cp, 'gc'),         "Uppercase_Letter");
82 is($charinfo->{combining},      "0");
83 is(charprop($cp, 'ccc'),        "Not_Reordered");
84 is($charinfo->{bidi},           "L");
85 is(charprop($cp, 'bc'),         "Left_To_Right");
86 is($charinfo->{decomposition},  "");
87 is(charprop($cp, 'dm'),         "A");
88 is($charinfo->{decimal},        "");
89 is($charinfo->{digit},          "");
90 is($charinfo->{numeric},        "");
91 is(charprop($cp, 'nv'),        "NaN");
92 is($charinfo->{mirrored},       "N");
93 is(charprop($cp, 'bidim'),      "No");
94 is($charinfo->{unicode10},      "");
95 is(charprop($cp, 'na1'),        "");
96 is($charinfo->{comment},        "");
97 is(charprop($cp, 'isc'),        "");
98 is($charinfo->{upper},          "");
99 is(charprop($cp, 'uc'),         "A");
100 is($charinfo->{lower},          $a_code);
101 is(charprop($cp, 'lc'),         "a");
102 is($charinfo->{title},          "");
103 is(charprop($cp, 'tc'),         "A");
104 is($charinfo->{block},          "Basic Latin");
105 is(charprop($cp, 'block'),      "Basic_Latin");
106 is($charinfo->{script},         "Latin");
107 is(charprop($cp, 'script'),     "Latin");
108
109 $cp = 0x100;
110 $charinfo = charinfo($cp);
111
112 is($charinfo->{code},           "0100", "LATIN CAPITAL LETTER A WITH MACRON");
113 is($charinfo->{name},           "LATIN CAPITAL LETTER A WITH MACRON");
114 is(charprop($cp, 'name'),       "LATIN CAPITAL LETTER A WITH MACRON");
115 is($charinfo->{category},       "Lu");
116 is(charprop($cp, 'gc'),         "Uppercase_Letter");
117 is($charinfo->{combining},      "0");
118 is(charprop($cp, 'ccc'),        "Not_Reordered");
119 is($charinfo->{bidi},           "L");
120 is(charprop($cp, 'bc'),         "Left_To_Right");
121 is($charinfo->{decomposition},  "$A_code 0304");
122 is(charprop($cp, 'dm'),         "A\x{0304}");
123 is($charinfo->{decimal},        "");
124 is($charinfo->{digit},          "");
125 is($charinfo->{numeric},        "");
126 is(charprop($cp, 'nv'),         "NaN");
127 is($charinfo->{mirrored},       "N");
128 is(charprop($cp, 'bidim'),      "No");
129 is($charinfo->{unicode10},      "LATIN CAPITAL LETTER A MACRON");
130 is(charprop($cp, 'na1'),        "LATIN CAPITAL LETTER A MACRON");
131 is($charinfo->{comment},        "");
132 is(charprop($cp, 'isc'),        "");
133 is($charinfo->{upper},          "");
134 is(charprop($cp, 'uc'),         "\x{100}");
135 is($charinfo->{lower},          "0101");
136 is(charprop($cp, 'lc'),         "\x{101}");
137 is($charinfo->{title},          "");
138 is(charprop($cp, 'tc'),         "\x{100}");
139 is($charinfo->{block},          "Latin Extended-A");
140 is(charprop($cp, 'block'),      "Latin_Extended_A");
141 is($charinfo->{script},         "Latin");
142 is(charprop($cp, 'script'),     "Latin");
143
144 $cp = 0x590;               # 0x0590 is in the Hebrew block but unused.
145 $charinfo = charinfo($cp);
146
147 is($charinfo->{code},           undef,  "0x0590 - unused Hebrew");
148 is($charinfo->{name},           undef);
149 is(charprop($cp, 'name'),       "");
150 is($charinfo->{category},       undef);
151 is(charprop($cp, 'gc'),         "Unassigned");
152 is($charinfo->{combining},      undef);
153 is(charprop($cp, 'ccc'),        "Not_Reordered");
154 is($charinfo->{bidi},           undef);
155 is(charprop($cp, 'bc'),         "Right_To_Left");
156 is($charinfo->{decomposition},  undef);
157 is(charprop($cp, 'dm'),         "\x{590}");
158 is($charinfo->{decimal},        undef);
159 is($charinfo->{digit},          undef);
160 is($charinfo->{numeric},        undef);
161 is(charprop($cp, 'nv'),         "NaN");
162 is($charinfo->{mirrored},       undef);
163 is(charprop($cp, 'bidim'),      "No");
164 is($charinfo->{unicode10},      undef);
165 is(charprop($cp, 'na1'),        "");
166 is($charinfo->{comment},        undef);
167 is(charprop($cp, 'isc'),        "");
168 is($charinfo->{upper},          undef);
169 is(charprop($cp, 'uc'),         "\x{590}");
170 is($charinfo->{lower},          undef);
171 is(charprop($cp, 'lc'),         "\x{590}");
172 is($charinfo->{title},          undef);
173 is(charprop($cp, 'tc'),         "\x{590}");
174 is($charinfo->{block},          undef);
175 is(charprop($cp, 'block'),      "Hebrew");
176 is($charinfo->{script},         undef);
177 is(charprop($cp, 'script'),     "Unknown");
178
179 # 0x05d0 is in the Hebrew block and used.
180
181 $cp = 0x5d0;
182 $charinfo = charinfo($cp);
183
184 is($charinfo->{code},           "05D0", "05D0 - used Hebrew");
185 is($charinfo->{name},           "HEBREW LETTER ALEF");
186 is(charprop($cp, 'name'),       "HEBREW LETTER ALEF");
187 is($charinfo->{category},       "Lo");
188 is(charprop($cp, 'gc'),         "Other_Letter");
189 is($charinfo->{combining},      "0");
190 is(charprop($cp, 'ccc'),        "Not_Reordered");
191 is($charinfo->{bidi},           "R");
192 is(charprop($cp, 'bc'),         "Right_To_Left");
193 is($charinfo->{decomposition},  "");
194 is(charprop($cp, 'dm'),         "\x{5d0}");
195 is($charinfo->{decimal},        "");
196 is($charinfo->{digit},          "");
197 is($charinfo->{numeric},        "");
198 is(charprop($cp, 'nv'),         "NaN");
199 is($charinfo->{mirrored},       "N");
200 is(charprop($cp, 'bidim'),      "No");
201 is($charinfo->{unicode10},      "");
202 is(charprop($cp, 'na1'),        "");
203 is($charinfo->{comment},        "");
204 is(charprop($cp, 'isc'),        "");
205 is($charinfo->{upper},          "");
206 is(charprop($cp, 'uc'),         "\x{5d0}");
207 is($charinfo->{lower},          "");
208 is(charprop($cp, 'lc'),         "\x{5d0}");
209 is($charinfo->{title},          "");
210 is(charprop($cp, 'tc'),         "\x{5d0}");
211 is($charinfo->{block},          "Hebrew");
212 is(charprop($cp, 'block'),      "Hebrew");
213 is($charinfo->{script},         "Hebrew");
214 is(charprop($cp, 'script'),     "Hebrew");
215
216 # An open syllable in Hangul.
217
218 $cp = 0xAC00;
219 $charinfo = charinfo($cp);
220
221 is($charinfo->{code},           "AC00", "HANGUL SYLLABLE U+AC00");
222 is($charinfo->{name},           "HANGUL SYLLABLE GA");
223 is(charprop($cp, 'name'),       "HANGUL SYLLABLE GA");
224 is($charinfo->{category},       "Lo");
225 is(charprop($cp, 'gc'),         "Other_Letter");
226 is($charinfo->{combining},      "0");
227 is(charprop($cp, 'ccc'),        "Not_Reordered");
228 is($charinfo->{bidi},           "L");
229 is(charprop($cp, 'bc'),         "Left_To_Right");
230 is($charinfo->{decomposition},  "1100 1161");
231 is(charprop($cp, 'dm'),         "\x{1100}\x{1161}");
232 is($charinfo->{decimal},        "");
233 is($charinfo->{digit},          "");
234 is($charinfo->{numeric},        "");
235 is(charprop($cp, 'nv'),         "NaN");
236 is($charinfo->{mirrored},       "N");
237 is(charprop($cp, 'bidim'),      "No");
238 is($charinfo->{unicode10},      "");
239 is(charprop($cp, 'na1'),        "");
240 is($charinfo->{comment},        "");
241 is(charprop($cp, 'isc'),        "");
242 is($charinfo->{upper},          "");
243 is(charprop($cp, 'uc'),         "\x{AC00}");
244 is($charinfo->{lower},          "");
245 is(charprop($cp, 'lc'),         "\x{AC00}");
246 is($charinfo->{title},          "");
247 is(charprop($cp, 'tc'),         "\x{AC00}");
248 is($charinfo->{block},          "Hangul Syllables");
249 is(charprop($cp, 'block'),      "Hangul_Syllables");
250 is($charinfo->{script},         "Hangul");
251 is(charprop($cp, 'script'),     "Hangul");
252
253 # A closed syllable in Hangul.
254
255 $cp = 0xAE00;
256 $charinfo = charinfo($cp);
257
258 is($charinfo->{code},           "AE00", "HANGUL SYLLABLE U+AE00");
259 is($charinfo->{name},           "HANGUL SYLLABLE GEUL");
260 is(charprop($cp, 'name'),       "HANGUL SYLLABLE GEUL");
261 is($charinfo->{category},       "Lo");
262 is(charprop($cp, 'gc'),         "Other_Letter");
263 is($charinfo->{combining},      "0");
264 is(charprop($cp, 'ccc'),        "Not_Reordered");
265 is($charinfo->{bidi},           "L");
266 is(charprop($cp, 'bc'),         "Left_To_Right");
267 is($charinfo->{decomposition},  "1100 1173 11AF");
268 is(charprop($cp, 'dm'),         "\x{1100}\x{1173}\x{11AF}");
269 is($charinfo->{decimal},        "");
270 is($charinfo->{digit},          "");
271 is($charinfo->{numeric},        "");
272 is(charprop($cp, 'nv'),         "NaN");
273 is($charinfo->{mirrored},       "N");
274 is(charprop($cp, 'bidim'),      "No");
275 is($charinfo->{unicode10},      "");
276 is(charprop($cp, 'na1'),        "");
277 is($charinfo->{comment},        "");
278 is(charprop($cp, 'isc'),        "");
279 is($charinfo->{upper},          "");
280 is(charprop($cp, 'uc'),         "\x{AE00}");
281 is($charinfo->{lower},          "");
282 is(charprop($cp, 'lc'),         "\x{AE00}");
283 is($charinfo->{title},          "");
284 is(charprop($cp, 'tc'),         "\x{AE00}");
285 is($charinfo->{block},          "Hangul Syllables");
286 is(charprop($cp, 'block'),      "Hangul_Syllables");
287 is($charinfo->{script},         "Hangul");
288 is(charprop($cp, 'script'),     "Hangul");
289
290 $cp = 0x1D400;
291 $charinfo = charinfo($cp);
292
293 is($charinfo->{code},           "1D400", "MATHEMATICAL BOLD CAPITAL A");
294 is($charinfo->{name},           "MATHEMATICAL BOLD CAPITAL A");
295 is(charprop($cp, 'name'),       "MATHEMATICAL BOLD CAPITAL A");
296 is($charinfo->{category},       "Lu");
297 is(charprop($cp, 'gc'),         "Uppercase_Letter");
298 is($charinfo->{combining},      "0");
299 is(charprop($cp, 'ccc'),        "Not_Reordered");
300 is($charinfo->{bidi},           "L");
301 is(charprop($cp, 'bc'),         "Left_To_Right");
302 is($charinfo->{decomposition},  "<font> $A_code");
303 is(charprop($cp, 'dm'),         "A");
304 is($charinfo->{decimal},        "");
305 is($charinfo->{digit},          "");
306 is($charinfo->{numeric},        "");
307 is(charprop($cp, 'nv'),         "NaN");
308 is($charinfo->{mirrored},       "N");
309 is(charprop($cp, 'bidim'),      "No");
310 is($charinfo->{unicode10},      "");
311 is(charprop($cp, 'na1'),        "");
312 is($charinfo->{comment},        "");
313 is(charprop($cp, 'isc'),        "");
314 is($charinfo->{upper},          "");
315 is(charprop($cp, 'uc'),         "\x{1D400}");
316 is($charinfo->{lower},          "");
317 is(charprop($cp, 'lc'),         "\x{1D400}");
318 is($charinfo->{title},          "");
319 is(charprop($cp, 'tc'),         "\x{1D400}");
320 is($charinfo->{block},          "Mathematical Alphanumeric Symbols");
321 is(charprop($cp, 'block'),      "Mathematical_Alphanumeric_Symbols");
322 is($charinfo->{script},         "Common");
323 is(charprop($cp, 'script'),     "Common");
324
325 $cp = 0x9FBA;                   #Bug 58428
326 $charinfo = charinfo(0x9FBA);
327
328 is($charinfo->{code},           "9FBA", "U+9FBA");
329 is($charinfo->{name},           "CJK UNIFIED IDEOGRAPH-9FBA");
330 is(charprop($cp, 'name'),       "CJK UNIFIED IDEOGRAPH-9FBA");
331 is($charinfo->{category},       "Lo");
332 is(charprop($cp, 'gc'),         "Other_Letter");
333 is($charinfo->{combining},      "0");
334 is(charprop($cp, 'ccc'),        "Not_Reordered");
335 is($charinfo->{bidi},           "L");
336 is(charprop($cp, 'bc'),         "Left_To_Right");
337 is($charinfo->{decomposition},  "");
338 is(charprop($cp, 'dm'),         "\x{9FBA}");
339 is($charinfo->{decimal},        "");
340 is($charinfo->{digit},          "");
341 is($charinfo->{numeric},        "");
342 is(charprop($cp, 'nv'),         "NaN");
343 is($charinfo->{mirrored},       "N");
344 is(charprop($cp, 'bidim'),      "No");
345 is($charinfo->{unicode10},      "");
346 is(charprop($cp, 'na1'),        "");
347 is($charinfo->{comment},        "");
348 is(charprop($cp, 'isc'),        "");
349 is($charinfo->{upper},          "");
350 is(charprop($cp, 'uc'),         "\x{9FBA}");
351 is($charinfo->{lower},          "");
352 is(charprop($cp, 'lc'),         "\x{9FBA}");
353 is($charinfo->{title},          "");
354 is(charprop($cp, 'tc'),         "\x{9FBA}");
355 is($charinfo->{block},          "CJK Unified Ideographs");
356 is(charprop($cp, 'block'),      "CJK_Unified_Ideographs");
357 is($charinfo->{script},         "Han");
358 is(charprop($cp, 'script'),     "Han");
359
360 use Unicode::UCD qw(charblock charscript);
361
362 # 0x0590 is in the Hebrew block but unused.
363
364 is(charblock(0x590),          "Hebrew", "0x0590 - Hebrew unused charblock");
365 is(charscript(0x590),         "Unknown",    "0x0590 - Hebrew unused charscript");
366 is(charblock(0x1FFFF),        "No_Block", "0x1FFFF - unused charblock");
367
368 my $fraction_3_4_code = sprintf("%04X", utf8::unicode_to_native(0xbe));
369 $cp = $fraction_3_4_code;
370 $charinfo = charinfo($fraction_3_4_code);
371
372 is($charinfo->{code},           $fraction_3_4_code, "VULGAR FRACTION THREE QUARTERS");
373 is($charinfo->{name},           "VULGAR FRACTION THREE QUARTERS");
374 is(charprop($cp, 'name'),       "VULGAR FRACTION THREE QUARTERS");
375 is($charinfo->{category},       "No");
376 is(charprop($cp, 'gc'),         "Other_Number");
377 is($charinfo->{combining},      "0");
378 is(charprop($cp, 'ccc'),        "Not_Reordered");
379 is($charinfo->{bidi},           "ON");
380 is(charprop($cp, 'bc'),         "Other_Neutral");
381 is($charinfo->{decomposition},  "<fraction> "
382                                 . sprintf("%04X", ord "3")
383                                 . " 2044 "
384                                 . sprintf("%04X", ord "4"));
385 is(charprop($cp, 'dm'),         "3\x{2044}4");
386 is($charinfo->{decimal},        "");
387 is($charinfo->{digit},          "");
388 is($charinfo->{numeric},        "3/4");
389 is(charprop($cp, 'nv'),        "0.75");
390 is($charinfo->{mirrored},       "N");
391 is(charprop($cp, 'bidim'),      "No");
392 is($charinfo->{unicode10},      "FRACTION THREE QUARTERS");
393 is(charprop($cp, 'na1'),        "FRACTION THREE QUARTERS");
394 is($charinfo->{comment},        "");
395 is(charprop($cp, 'isc'),        "");
396 is($charinfo->{upper},          "");
397 is(charprop($cp, 'uc'),         chr hex $cp);
398 is($charinfo->{lower},          "");
399 is(charprop($cp, 'lc'),         chr hex $cp);
400 is($charinfo->{title},          "");
401 is(charprop($cp, 'tc'),         chr hex $cp);
402 is($charinfo->{block},          "Latin-1 Supplement");
403 is(charprop($cp, 'block'),      "Latin_1_Supplement");
404 is($charinfo->{script},         "Common");
405 is(charprop($cp, 'script'),     "Common");
406
407 # This is to test a case where both simple and full lowercases exist and
408 # differ
409 $cp = 0x130;
410 $charinfo = charinfo($cp);
411 my $I_code = sprintf("%04X", ord("I"));
412 my $i_code = sprintf("%04X", ord("i"));
413
414 is($charinfo->{code},           "0130", "LATIN CAPITAL LETTER I WITH DOT ABOVE");
415 is($charinfo->{name},           "LATIN CAPITAL LETTER I WITH DOT ABOVE");
416 is(charprop($cp, 'name'),       "LATIN CAPITAL LETTER I WITH DOT ABOVE");
417 is($charinfo->{category},       "Lu");
418 is(charprop($cp, 'gc'),         "Uppercase_Letter");
419 is($charinfo->{combining},      "0");
420 is(charprop($cp, 'ccc'),        "Not_Reordered");
421 is($charinfo->{bidi},           "L");
422 is(charprop($cp, 'bc'),         "Left_To_Right");
423 is($charinfo->{decomposition},  "$I_code 0307");
424 is(charprop($cp, 'dm'),         "I\x{0307}");
425 is($charinfo->{decimal},        "");
426 is($charinfo->{digit},          "");
427 is($charinfo->{numeric},        "");
428 is(charprop($cp, 'nv'),         "NaN");
429 is($charinfo->{mirrored},       "N");
430 is(charprop($cp, 'bidim'),      "No");
431 is($charinfo->{unicode10},      "LATIN CAPITAL LETTER I DOT");
432 is(charprop($cp, 'na1'),        "LATIN CAPITAL LETTER I DOT");
433 is($charinfo->{comment},        "");
434 is(charprop($cp, 'isc'),        "");
435 is($charinfo->{upper},          "");
436 is(charprop($cp, 'uc'),         "\x{130}");
437 is($charinfo->{lower},          $i_code);
438 is(charprop($cp, 'lc'),         "i\x{307}");
439 is($charinfo->{title},          "");
440 is(charprop($cp, 'tc'),         "\x{130}");
441 is($charinfo->{block},          "Latin Extended-A");
442 is(charprop($cp, 'block'),      "Latin_Extended_A");
443 is($charinfo->{script},         "Latin");
444 is(charprop($cp, 'script'),     "Latin");
445
446 # This is to test a case where both simple and full uppercases exist and
447 # differ
448 $cp = 0x1F80;
449 $charinfo = charinfo($cp);
450
451 is($charinfo->{code},           "1F80", "GREEK SMALL LETTER ALPHA WITH PSILI AND YPOGEGRAMMENI");
452 is($charinfo->{name},           "GREEK SMALL LETTER ALPHA WITH PSILI AND YPOGEGRAMMENI");
453 is(charprop($cp, "name"),       "GREEK SMALL LETTER ALPHA WITH PSILI AND YPOGEGRAMMENI");
454 is($charinfo->{category},       "Ll");
455 is(charprop($cp, "gc"),         "Lowercase_Letter");
456 is($charinfo->{combining},      "0");
457 is(charprop($cp, "ccc"),        "Not_Reordered");
458 is($charinfo->{bidi},           "L");
459 is(charprop($cp, "bc"),         "Left_To_Right");
460 is($charinfo->{decomposition},  "1F00 0345");
461 is(charprop($cp, "dm"),         "\x{1F00}\x{0345}");
462 is($charinfo->{decimal},        "");
463 is($charinfo->{digit},          "");
464 is($charinfo->{numeric},        "");
465 is(charprop($cp, "nv"),         "NaN");
466 is($charinfo->{mirrored},       "N");
467 is(charprop($cp, "bidim"),      "No");
468 is($charinfo->{unicode10},      "");
469 is(charprop($cp, "na1"),        "");
470 is($charinfo->{comment},        "");
471 is(charprop($cp, "isc"),        "");
472 is($charinfo->{upper},          "1F88");
473 is(charprop($cp, "uc"),         "\x{1F08}\x{0399}");
474 is(charprop($cp, "suc"),        "\x{1F88}");
475 is($charinfo->{lower},          "");
476 is(charprop($cp, "lc"),         "\x{1F80}");
477 is($charinfo->{title},          "1F88");
478 is(charprop($cp, "tc"),         "\x{1F88}");
479 is($charinfo->{block},          "Greek Extended");
480 is(charprop($cp, "block"),      "Greek_Extended");
481 is($charinfo->{script},         "Greek");
482 is(charprop($cp, "script"),     "Greek");
483
484 is(charprop(ord("A"), "foo"),    undef,
485                         "Verify charprop of unknown property returns <undef>");
486
487 # These were created from inspection of the code to exercise the branches
488 is(charprop(ord("("), "bpb"),    ")",
489             "Verify charprop figures out that s-type properties can be char");
490 is(charprop(ord("9"), "nv"),     9,
491                             "Verify charprop can adjust an ar-type property");
492 is(charprop(utf8::unicode_to_native(0xAD), "NFKC_Casefold"), "",
493                     "Verify charprop can handle an \"\" in ae-type property");
494
495 my $mark_props_ref = charprops_all(0x300);
496 is($mark_props_ref->{'Bidi_Class'}, "Nonspacing_Mark",
497                                     "Next tests are charprops_all of 0x300");
498 is($mark_props_ref->{'Bidi_Mirrored'}, "No");
499 is($mark_props_ref->{'Canonical_Combining_Class'}, "Above");
500 is($mark_props_ref->{'Case_Folding'}, "\x{300}");
501 is($mark_props_ref->{'Decomposition_Mapping'}, "\x{300}");
502 is($mark_props_ref->{'Decomposition_Type'}, "None");
503 is($mark_props_ref->{'General_Category'}, "Nonspacing_Mark");
504 is($mark_props_ref->{'ISO_Comment'}, "");
505 is($mark_props_ref->{'Lowercase_Mapping'}, "\x{300}");
506 is($mark_props_ref->{'Name'}, "COMBINING GRAVE ACCENT");
507 is($mark_props_ref->{'Numeric_Type'}, "None");
508 is($mark_props_ref->{'Numeric_Value'}, "NaN");
509 is($mark_props_ref->{'Simple_Case_Folding'}, "\x{300}");
510 is($mark_props_ref->{'Simple_Lowercase_Mapping'}, "\x{300}");
511 is($mark_props_ref->{'Simple_Titlecase_Mapping'}, "\x{300}");
512 is($mark_props_ref->{'Simple_Uppercase_Mapping'}, "\x{300}");
513 is($mark_props_ref->{'Titlecase_Mapping'}, "\x{300}");
514 is($mark_props_ref->{'Unicode_1_Name'}, "NON-SPACING GRAVE");
515 is($mark_props_ref->{'Uppercase_Mapping'}, "\x{300}");
516
517 use Unicode::UCD qw(charblocks charscripts);
518
519 my $charblocks = charblocks();
520
521 ok(exists $charblocks->{Thai}, 'Thai charblock exists');
522 is($charblocks->{Thai}->[0]->[0], hex('0e00'));
523 ok(!exists $charblocks->{PigLatin}, 'PigLatin charblock does not exist');
524
525 my $charscripts = charscripts();
526
527 ok(exists $charscripts->{Armenian}, 'Armenian charscript exists');
528 is($charscripts->{Armenian}->[0]->[0], hex('0531'));
529 ok(!exists $charscripts->{PigLatin}, 'PigLatin charscript does not exist');
530
531 my $charscript;
532
533 $charscript = charscript("12ab");
534 is($charscript, 'Ethiopic', 'Ethiopic charscript');
535
536 $charscript = charscript("0x12ab");
537 is($charscript, 'Ethiopic');
538
539 $charscript = charscript("U+12ab");
540 is($charscript, 'Ethiopic');
541
542 my $ranges;
543
544 $ranges = charscript('Ogham');
545 is($ranges->[0]->[0], hex('1680'), 'Ogham charscript');
546 is($ranges->[0]->[1], hex('169C'));
547
548 use Unicode::UCD qw(charinrange);
549
550 $ranges = charscript('Cherokee');
551 ok(!charinrange($ranges, "139f"), 'Cherokee charscript');
552 ok( charinrange($ranges, "13a0"));
553 ok( charinrange($ranges, "13f4"));
554 ok(!charinrange($ranges, "13ff"));
555
556 use Unicode::UCD qw(general_categories);
557
558 my $gc = general_categories();
559
560 ok(exists $gc->{L}, 'has L');
561 is($gc->{L}, 'Letter', 'L is Letter');
562 is($gc->{Lu}, 'UppercaseLetter', 'Lu is UppercaseLetter');
563
564 use Unicode::UCD qw(bidi_types);
565
566 my $bt = bidi_types();
567
568 ok(exists $bt->{L}, 'has L');
569 is($bt->{L}, 'Left-to-Right', 'L is Left-to-Right');
570 is($bt->{AL}, 'Right-to-Left Arabic', 'AL is Right-to-Left Arabic');
571
572 # If this fails, then maybe one should look at the Unicode changes to see
573 # what else might need to be updated.
574 is(Unicode::UCD::UnicodeVersion, '8.0.0', 'UnicodeVersion');
575
576 use Unicode::UCD qw(compexcl);
577
578 ok(!compexcl(0x0100), 'compexcl');
579 ok(!compexcl(0xD801), 'compexcl of surrogate');
580 ok(!compexcl(0x110000), 'compexcl of non-Unicode code point');
581 ok( compexcl(0x0958));
582
583 use Unicode::UCD qw(casefold);
584
585 my $casefold;
586
587 $casefold = casefold(utf8::unicode_to_native(0x41));
588
589 is($casefold->{code}, $A_code, 'casefold native(0x41) code');
590 is($casefold->{status}, 'C', 'casefold native(0x41) status');
591 is($casefold->{mapping}, $a_code, 'casefold native(0x41) mapping');
592 is($casefold->{full}, $a_code, 'casefold native(0x41) full');
593 is($casefold->{simple}, $a_code, 'casefold native(0x41) simple');
594 is($casefold->{turkic}, "", 'casefold native(0x41) turkic');
595
596 $casefold = casefold(utf8::unicode_to_native(0xdf));
597 my $sharp_s_code = sprintf("%04X", utf8::unicode_to_native(0xdf));
598 my $S_code = sprintf("%04X", ord "S");
599 my $s_code = sprintf("%04X", ord "s");
600
601 is($casefold->{code}, $sharp_s_code, 'casefold native(0xDF) code');
602 is($casefold->{status}, 'F', 'casefold native(0xDF) status');
603 is($casefold->{mapping}, "$s_code $s_code", 'casefold native(0xDF) mapping');
604 is($casefold->{full}, "$s_code $s_code", 'casefold native(0xDF) full');
605 is($casefold->{simple}, "", 'casefold native(0xDF) simple');
606 is($casefold->{turkic}, "", 'casefold native(0xDF) turkic');
607
608 # Do different tests depending on if version < 3.2, or not.
609 my $v_unicode_version = pack "C*", split /\./, Unicode::UCD::UnicodeVersion();
610 if ($v_unicode_version eq v3.0.1) {
611         # In this release, there was no special Turkic values.
612         # Both 0x130 and 0x131 folded to 'i'.
613
614         $casefold = casefold(0x130);
615
616         is($casefold->{code}, '0130', 'casefold 0x130 code');
617         is($casefold->{status}, 'C' , 'casefold 0x130 status');
618         is($casefold->{mapping}, $i_code, 'casefold 0x130 mapping');
619         is($casefold->{full}, $i_code, 'casefold 0x130 full');
620         is($casefold->{simple}, $i_code, 'casefold 0x130 simple');
621         is($casefold->{turkic}, "", 'casefold 0x130 turkic');
622
623         $casefold = casefold(0x131);
624
625         is($casefold->{code}, '0131', 'casefold 0x131 code');
626         is($casefold->{status}, 'C' , 'casefold 0x131 status');
627         is($casefold->{mapping}, $i_code, 'casefold 0x131 mapping');
628         is($casefold->{full}, $i_code, 'casefold 0x131 full');
629         is($casefold->{simple}, $i_code, 'casefold 0x131 simple');
630         is($casefold->{turkic}, "", 'casefold 0x131 turkic');
631 }
632 elsif ($v_unicode_version lt v3.2.0) {
633         $casefold = casefold(0x130);
634
635         is($casefold->{code}, '0130', 'casefold 0x130 code');
636         is($casefold->{status}, 'I' , 'casefold 0x130 status');
637         is($casefold->{mapping}, $i_code, 'casefold 0x130 mapping');
638         is($casefold->{full}, $i_code, 'casefold 0x130 full');
639         is($casefold->{simple}, $i_code, 'casefold 0x130 simple');
640         is($casefold->{turkic}, $i_code, 'casefold 0x130 turkic');
641
642         $casefold = casefold(0x131);
643
644         is($casefold->{code}, '0131', 'casefold 0x131 code');
645         is($casefold->{status}, 'I' , 'casefold 0x131 status');
646         is($casefold->{mapping}, $i_code, 'casefold 0x131 mapping');
647         is($casefold->{full}, $i_code, 'casefold 0x131 full');
648         is($casefold->{simple}, $i_code, 'casefold 0x131 simple');
649         is($casefold->{turkic}, $i_code, 'casefold 0x131 turkic');
650 } else {
651         $casefold = casefold(utf8::unicode_to_native(0x49));
652
653         is($casefold->{code}, $I_code, 'casefold native(0x49) code');
654         is($casefold->{status}, 'C' , 'casefold native(0x49) status');
655         is($casefold->{mapping}, $i_code, 'casefold native(0x49) mapping');
656         is($casefold->{full}, $i_code, 'casefold native(0x49) full');
657         is($casefold->{simple}, $i_code, 'casefold native(0x49) simple');
658         is($casefold->{turkic}, "0131", 'casefold native(0x49) turkic');
659
660         $casefold = casefold(0x130);
661
662         is($casefold->{code}, '0130', 'casefold 0x130 code');
663         is($casefold->{status}, 'F' , 'casefold 0x130 status');
664         is($casefold->{mapping}, "$i_code 0307", 'casefold 0x130 mapping');
665         is($casefold->{full}, "$i_code 0307", 'casefold 0x130 full');
666         is($casefold->{simple}, "", 'casefold 0x130 simple');
667         is($casefold->{turkic}, $i_code, 'casefold 0x130 turkic');
668 }
669
670 $casefold = casefold(0x1F88);
671
672 is($casefold->{code}, '1F88', 'casefold 0x1F88 code');
673 is($casefold->{status}, 'S' , 'casefold 0x1F88 status');
674 is($casefold->{mapping}, '1F80', 'casefold 0x1F88 mapping');
675 is($casefold->{full}, '1F00 03B9', 'casefold 0x1F88 full');
676 is($casefold->{simple}, '1F80', 'casefold 0x1F88 simple');
677 is($casefold->{turkic}, "", 'casefold 0x1F88 turkic');
678
679 ok(!casefold(utf8::unicode_to_native(0x20)));
680
681 use Unicode::UCD qw(casespec);
682
683 my $casespec;
684
685 ok(!casespec(utf8::unicode_to_native(0x41)));
686
687 $casespec = casespec(utf8::unicode_to_native(0xdf));
688
689 ok($casespec->{code} eq $sharp_s_code &&
690    $casespec->{lower} eq $sharp_s_code  &&
691    $casespec->{title} eq "$S_code $s_code"  &&
692    $casespec->{upper} eq "$S_code $S_code" &&
693    !defined $casespec->{condition}, 'casespec native(0xDF)');
694
695 $casespec = casespec(0x307);
696
697 ok($casespec->{az}->{code} eq '0307' &&
698    !defined $casespec->{az}->{lower} &&
699    $casespec->{az}->{title} eq '0307'  &&
700    $casespec->{az}->{upper} eq '0307' &&
701    $casespec->{az}->{condition} eq 'az After_I',
702    'casespec 0x307');
703
704 # perl #7305 UnicodeCD::compexcl is weird
705
706 for (1) {my $a=compexcl $_}
707 ok(1, 'compexcl read-only $_: perl #7305');
708 map {compexcl $_} %{{1=>2}};
709 ok(1, 'compexcl read-only hash: perl #7305');
710
711 is(Unicode::UCD::_getcode('123'),     123, "_getcode(123)");
712 is(Unicode::UCD::_getcode('0123'),  0x123, "_getcode(0123)");
713 is(Unicode::UCD::_getcode('0x123'), 0x123, "_getcode(0x123)");
714 is(Unicode::UCD::_getcode('0X123'), 0x123, "_getcode(0X123)");
715 is(Unicode::UCD::_getcode('U+123'), 0x123, "_getcode(U+123)");
716 is(Unicode::UCD::_getcode('u+123'), 0x123, "_getcode(u+123)");
717 is(Unicode::UCD::_getcode('U+1234'),   0x1234, "_getcode(U+1234)");
718 is(Unicode::UCD::_getcode('U+12345'), 0x12345, "_getcode(U+12345)");
719 is(Unicode::UCD::_getcode('123x'),    undef, "_getcode(123x)");
720 is(Unicode::UCD::_getcode('x123'),    undef, "_getcode(x123)");
721 is(Unicode::UCD::_getcode('0x123x'),  undef, "_getcode(x123)");
722 is(Unicode::UCD::_getcode('U+123x'),  undef, "_getcode(x123)");
723
724 {
725     my $r1 = charscript('Latin');
726     if (ok(defined $r1, "Found Latin script")) {
727         my $n1 = @$r1;
728         is($n1, 31, "number of ranges in Latin script (Unicode 7.0.0)") if $::IS_ASCII;
729         shift @$r1 while @$r1;
730         my $r2 = charscript('Latin');
731         is(@$r2, $n1, "modifying results should not mess up internal caches");
732     }
733 }
734
735 {
736         is(charinfo(0xdeadbeef), undef, "[perl #23273] warnings in Unicode::UCD");
737 }
738
739 use Unicode::UCD qw(namedseq);
740
741 is(namedseq("KATAKANA LETTER AINU P"), "\x{31F7}\x{309A}", "namedseq");
742 is(namedseq("KATAKANA LETTER AINU Q"), undef);
743 is(namedseq(), undef);
744 is(namedseq(qw(foo bar)), undef);
745 my @ns = namedseq("KATAKANA LETTER AINU P");
746 is(scalar @ns, 2);
747 is($ns[0], 0x31F7);
748 is($ns[1], 0x309A);
749 my %ns = namedseq();
750 is($ns{"KATAKANA LETTER AINU P"}, "\x{31F7}\x{309A}");
751 @ns = namedseq(42);
752 is(@ns, 0);
753
754 use Unicode::UCD qw(num);
755 use charnames ":full";
756
757 is(num("0"), 0, 'Verify num("0") == 0');
758 is(num("98765"), 98765, 'Verify num("98765") == 98765');
759 ok(! defined num("98765\N{FULLWIDTH DIGIT FOUR}"), 'Verify num("98765\N{FULLWIDTH DIGIT FOUR}") isnt defined');
760 is(num("\N{NEW TAI LUE DIGIT TWO}"), 2, 'Verify num("\N{NEW TAI LUE DIGIT TWO}") == 2');
761 is(num("\N{NEW TAI LUE DIGIT ONE}"), 1, 'Verify num("\N{NEW TAI LUE DIGIT ONE}") == 1');
762 is(num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE DIGIT ONE}"), 21, 'Verify num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE DIGIT ONE}") == 21');
763 ok(! defined num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE THAM DIGIT ONE}"), 'Verify num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE THAM DIGIT ONE}") isnt defined');
764 is(num("\N{CHAM DIGIT ZERO}\N{CHAM DIGIT THREE}"), 3, 'Verify num("\N{CHAM DIGIT ZERO}\N{CHAM DIGIT THREE}") == 3');
765 ok(! defined num("\N{CHAM DIGIT ZERO}\N{JAVANESE DIGIT NINE}"), 'Verify num("\N{CHAM DIGIT ZERO}\N{JAVANESE DIGIT NINE}") isnt defined');
766 is(num("\N{SUPERSCRIPT TWO}"), 2, 'Verify num("\N{SUPERSCRIPT TWO} == 2');
767 is(num("\N{ETHIOPIC NUMBER TEN THOUSAND}"), 10000, 'Verify num("\N{ETHIOPIC NUMBER TEN THOUSAND}") == 10000');
768 is(num("\N{NORTH INDIC FRACTION ONE HALF}"), .5, 'Verify num("\N{NORTH INDIC FRACTION ONE HALF}") == .5');
769 is(num("\N{U+12448}"), 9, 'Verify num("\N{U+12448}") == 9');
770 is(num("\N{U+5146}"), 1000000000000, 'Verify num("\N{U+5146}") == 1000000000000');
771
772 # Create a user-defined property
773 sub InKana {<<'END'}
774 3040    309F
775 30A0    30FF
776 END
777
778 use Unicode::UCD qw(prop_aliases);
779
780 is(prop_aliases(undef), undef, "prop_aliases(undef) returns <undef>");
781 is(prop_aliases("unknown property"), undef,
782                 "prop_aliases(<unknown property>) returns <undef>");
783 is(prop_aliases("InKana"), undef,
784                 "prop_aliases(<user-defined property>) returns <undef>");
785 is(prop_aliases("Perl_Decomposition_Mapping"), undef, "prop_aliases('Perl_Decomposition_Mapping') returns <undef> since internal-Perl-only");
786 is(prop_aliases("Perl_Charnames"), undef,
787     "prop_aliases('Perl_Charnames') returns <undef> since internal-Perl-only");
788 is(prop_aliases("isgc"), undef,
789     "prop_aliases('isgc') returns <undef> since is not covered Perl extension");
790 is(prop_aliases("Is_Is_Any"), undef,
791                 "prop_aliases('Is_Is_Any') returns <undef> since two is's");
792 is(prop_aliases("ccc=vr"), undef,
793                           "prop_aliases('ccc=vr') doesn't generate a warning");
794
795 require 'utf8_heavy.pl';
796 require "unicore/Heavy.pl";
797
798 # Keys are lists of properties. Values are defined if have been tested.
799 my %props;
800
801 # To test for loose matching, add in the characters that are ignored there.
802 my $extra_chars = "-_ ";
803
804 # The one internal property we accept
805 $props{'Perl_Decimal_Digit'} = 1;
806 my @list = prop_aliases("perldecimaldigit");
807 is_deeply(\@list,
808           [ "Perl_Decimal_Digit",
809             "Perl_Decimal_Digit"
810           ], "prop_aliases('perldecimaldigit') returns Perl_Decimal_Digit as both short and full names");
811
812 # Get the official Unicode property name synonyms and test them.
813
814 SKIP: {
815 skip "PropertyAliases.txt is not in this Unicode version", 1 if $v_unicode_version lt v3.2.0;
816 open my $props, "<", "../lib/unicore/PropertyAliases.txt"
817                 or die "Can't open Unicode PropertyAliases.txt";
818 local $/ = "\n";
819 while (<$props>) {
820     s/\s*#.*//;           # Remove comments
821     next if /^\s* $/x;    # Ignore empty and comment lines
822
823     chomp;
824     local $/ = $input_record_separator;
825     my $count = 0;  # 0th field in line is short name; 1th is long name
826     my $short_name;
827     my $full_name;
828     my @names_via_short;
829     foreach my $alias (split /\s*;\s*/) {    # Fields are separated by
830                                              # semi-colons
831         # Add in the characters that are supposed to be ignored, to test loose
832         # matching, which the tested function does on all inputs.
833         my $mod_name = "$extra_chars$alias";
834
835         my $loose = &utf8::_loose_name(lc $alias);
836
837         # Indicate we have tested this.
838         $props{$loose} = 1;
839
840         my @all_names = prop_aliases($mod_name);
841         if (grep { $_ eq $loose } @Unicode::UCD::suppressed_properties) {
842             is(@all_names, 0, "prop_aliases('$mod_name') returns undef since $alias is not installed");
843             next;
844         }
845         elsif (! @all_names) {
846             fail("prop_aliases('$mod_name')");
847             diag("'$alias' is unknown to prop_aliases()");
848             next;
849         }
850
851         if ($count == 0) {  # Is short name
852
853             @names_via_short = prop_aliases($mod_name);
854
855             # If the 0th test fails, no sense in continuing with the others
856             last unless is($names_via_short[0], $alias,
857                     "prop_aliases: '$alias' is the short name for '$mod_name'");
858             $short_name = $alias;
859         }
860         elsif ($count == 1) {   # Is full name
861
862             # Some properties have the same short and full name; no sense
863             # repeating the test if the same.
864             if ($alias ne $short_name) {
865                 my @names_via_full = prop_aliases($mod_name);
866                 is_deeply(\@names_via_full, \@names_via_short, "prop_aliases() returns the same list for both '$short_name' and '$mod_name'");
867             }
868
869             # Tests scalar context
870             is(prop_aliases($short_name), $alias,
871                 "prop_aliases: '$alias' is the long name for '$short_name'");
872         }
873         else {  # Is another alias
874             is_deeply(\@all_names, \@names_via_short, "prop_aliases() returns the same list for both '$short_name' and '$mod_name'");
875             ok((grep { $_ =~ /^$alias$/i } @all_names),
876                 "prop_aliases: '$alias' is listed as an alias for '$mod_name'");
877         }
878
879         $count++;
880     }
881 }
882 } # End of SKIP block
883
884 # Now test anything we can find that wasn't covered by the tests of the
885 # official properties.  We have no way of knowing if mktables omitted a Perl
886 # extension or not, but we do the best we can from its generated lists
887
888 foreach my $alias (sort keys %utf8::loose_to_file_of) {
889     next if $alias =~ /=/;
890     my $lc_name = lc $alias;
891     my $loose = &utf8::_loose_name($lc_name);
892     next if exists $props{$loose};  # Skip if already tested
893     $props{$loose} = 1;
894     my $mod_name = "$extra_chars$alias";    # Tests loose matching
895     my @aliases = prop_aliases($mod_name);
896     my $found_it = grep { &utf8::_loose_name(lc $_) eq $lc_name } @aliases;
897     if ($found_it) {
898         pass("prop_aliases: '$lc_name' is listed as an alias for '$mod_name'");
899     }
900     elsif ($lc_name =~ /l[_&]$/) {
901
902         # These two names are special in that they don't appear in the
903         # returned list because they are discouraged from use.  Verify
904         # that they return the same list as a non-discouraged version.
905         my @LC = prop_aliases('Is_LC');
906         is_deeply(\@aliases, \@LC, "prop_aliases: '$lc_name' returns the same list as 'Is_LC'");
907     }
908     else {
909         my $stripped = $lc_name =~ s/^is//;
910
911         # Could be that the input includes a prefix 'is', which is rarely
912         # returned as an alias, so having successfully stripped it off above,
913         # try again.
914         if ($stripped) {
915             $found_it = grep { &utf8::_loose_name(lc $_) eq $lc_name } @aliases;
916         }
917
918         # If that didn't work, it could be that it's a block, which is always
919         # returned with a leading 'In_' to avoid ambiguity.  Try comparing
920         # with that stripped off.
921         if (! $found_it) {
922             $found_it = grep { &utf8::_loose_name(s/^In_(.*)/\L$1/r) eq $lc_name }
923                               @aliases;
924             # Could check that is a real block, but tests for invmap will
925             # likely pickup any errors, since this will be tested there.
926             $lc_name = "in$lc_name" if $found_it;   # Change for message below
927         }
928         my $message = "prop_aliases: '$lc_name' is listed as an alias for '$mod_name'";
929         ($found_it) ? pass($message) : fail($message);
930     }
931 }
932
933 # Some of the Perl extensions should always be built; make sure they have the
934 # correct full name, etc.
935 for my $prop (qw(Alnum Blank Cntrl Digit Graph Print Word XDigit)) {
936     my @expected = ( $prop, "XPosix$prop" );
937     my @got = prop_aliases($prop);
938     splice @got, 2;
939     is_deeply(\@got, \@expected, "Got expected aliases for $prop");
940 }
941
942 my $done_equals = 0;
943 foreach my $alias (keys %utf8::stricter_to_file_of) {
944     if ($alias =~ /=/) {    # Only test one case where there is an equals
945         next if $done_equals;
946         $done_equals = 1;
947     }
948     my $lc_name = lc $alias;
949     my @list = prop_aliases($alias);
950     if ($alias =~ /^_/) {
951         is(@list, 0, "prop_aliases: '$lc_name' returns an empty list since it is internal_only");
952     }
953     elsif ($alias =~ /=/) {
954         is(@list, 0, "prop_aliases: '$lc_name' returns an empty list since is illegal property name");
955     }
956     else {
957         ok((grep { lc $_ eq $lc_name } @list),
958                 "prop_aliases: '$lc_name' is listed as an alias for '$alias'");
959     }
960 }
961
962 use Unicode::UCD qw(prop_values prop_value_aliases);
963
964 is(prop_value_aliases("unknown property", "unknown value"), undef,
965     "prop_value_aliases(<unknown property>, <unknown value>) returns <undef>");
966 is(prop_value_aliases(undef, undef), undef,
967                            "prop_value_aliases(undef, undef) returns <undef>");
968 is((prop_value_aliases("na", "A")), "A", "test that prop_value_aliases returns its input for properties that don't have synonyms");
969 is(prop_value_aliases("isgc", "C"), undef, "prop_value_aliases('isgc', 'C') returns <undef> since is not covered Perl extension");
970 is(prop_value_aliases("gc", "isC"), undef, "prop_value_aliases('gc', 'isC') returns <undef> since is not covered Perl extension");
971 is(prop_value_aliases("Any", "None"), undef, "prop_value_aliases('Any', 'None') returns <undef> since is Perl extension and 'None' is not valid");
972 is(prop_value_aliases("lc", "A"), "A", "prop_value_aliases('lc', 'A') returns its input, as docs say it does");
973
974 # We have no way of knowing if mktables omitted a Perl extension that it
975 # shouldn't have, but we can check if it omitted an official Unicode property
976 # name synonym.  And for those, we can check if the short and full names are
977 # correct.
978
979 my %pva_tested;   # List of things already tested.
980
981 SKIP: {
982 skip "PropValueAliases.txt is not in this Unicode version", 1 if $v_unicode_version lt v3.2.0;
983 open my $propvalues, "<", "../lib/unicore/PropValueAliases.txt"
984      or die "Can't open Unicode PropValueAliases.txt";
985 local $/ = "\n";
986
987 # Each examined line in the file is for a single value for a property.  We
988 # accumulate all the values for each property using these two variables.
989 my $prev_prop = "";
990 my @this_prop_values;
991
992 while (<$propvalues>) {
993     s/\s*#.*//;           # Remove comments
994     next if /^\s* $/x;    # Ignore empty and comment lines
995     chomp;
996     local $/ = $input_record_separator;
997
998     # Fix typo in official input file
999     s/CCC133/CCC132/g if $v_unicode_version eq v6.1.0;
1000
1001     my @fields = split /\s*;\s*/; # Fields are separated by semi-colons
1002     my $prop = shift @fields;   # 0th field is the property,
1003
1004     # When changing properties, we examine the accumulated values for the old
1005     # one to see if our function that returns them matches.
1006     if ($prev_prop ne $prop) {
1007         if ($prev_prop ne "") { # Skip for the first time through
1008             my @ucd_function_values = prop_values($prev_prop);
1009             @ucd_function_values = () unless @ucd_function_values;
1010
1011             # This perl extension doesn't appear in the official file
1012             push @this_prop_values, "Non_Canon" if $prev_prop eq 'dt';
1013
1014             my @file_values = undef;
1015             @file_values = sort { lc($a =~ s/_//gr) cmp lc($b =~ s/_//gr) }
1016                                    @this_prop_values if @this_prop_values;
1017             is_deeply(\@ucd_function_values, \@file_values,
1018               "prop_values('$prev_prop') returns correct list of values");
1019         }
1020         $prev_prop = $prop;
1021         undef @this_prop_values;
1022     }
1023
1024     my $count = 0;  # 0th field in line (after shifting off the property) is
1025                     # short name; 1th is long name
1026     my $short_name;
1027     my @names_via_short;    # Saves the values between iterations
1028
1029     # The property on the lhs of the = is always loosely matched.  Add in
1030     # characters that are ignored under loose matching to test that
1031     my $mod_prop = "$extra_chars$prop";
1032
1033     if ($fields[0] eq 'n/a') {  # See comments in input file, essentially
1034                                 # means full name and short name are identical
1035         $fields[0] = $fields[1];
1036     }
1037     elsif ($fields[0] ne $fields[1]
1038            && &utf8::_loose_name(lc $fields[0])
1039                eq &utf8::_loose_name(lc $fields[1])
1040            && $fields[1] !~ /[[:upper:]]/)
1041     {
1042         # Also, there is a bug in the file in which "n/a" is omitted, and
1043         # the two fields are identical except for case, and the full name
1044         # is all lower case.  Copy the "short" name unto the full one to
1045         # give it some upper case.
1046
1047         $fields[1] = $fields[0];
1048     }
1049
1050     # The ccc property in the file is special; has an extra numeric field
1051     # (0th), which should go at the end, since we use the next two fields as
1052     # the short and full names, respectively.  See comments in input file.
1053     splice (@fields, 0, 0, splice(@fields, 1, 2)) if $prop eq 'ccc';
1054
1055     my $loose_prop = &utf8::_loose_name(lc $prop);
1056     my $suppressed = grep { $_ eq $loose_prop }
1057                           @Unicode::UCD::suppressed_properties;
1058     push @this_prop_values, $fields[0] unless $suppressed;
1059     foreach my $value (@fields) {
1060         if ($suppressed) {
1061             is(prop_value_aliases($prop, $value), undef, "prop_value_aliases('$prop', '$value') returns undef for suppressed property $prop");
1062             next;
1063         }
1064         elsif (grep { $_ eq ("$loose_prop=" . &utf8::_loose_name(lc $value)) } @Unicode::UCD::suppressed_properties) {
1065             is(prop_value_aliases($prop, $value), undef, "prop_value_aliases('$prop', '$value') returns undef for suppressed property $prop=$value");
1066             next;
1067         }
1068
1069         # Add in test for loose matching.
1070         my $mod_value = "$extra_chars$value";
1071
1072         # If the value is a number, optionally negative, including a floating
1073         # point or rational numer, it should be only strictly matched, so the
1074         # loose matching should fail.
1075         if ($value =~ / ^ -? \d+ (?: [\/.] \d+ )? $ /x) {
1076             is(prop_value_aliases($mod_prop, $mod_value), undef, "prop_value_aliases('$mod_prop', '$mod_value') returns undef because '$mod_value' should be strictly matched");
1077
1078             # And reset so below tests just the strict matching.
1079             $mod_value = $value;
1080         }
1081
1082         if ($count == 0) {
1083
1084             @names_via_short = prop_value_aliases($mod_prop, $mod_value);
1085
1086             # If the 0th test fails, no sense in continuing with the others
1087             last unless is($names_via_short[0], $value, "prop_value_aliases: In '$prop', '$value' is the short name for '$mod_value'");
1088             $short_name = $value;
1089         }
1090         elsif ($count == 1) {
1091
1092             # Some properties have the same short and full name; no sense
1093             # repeating the test if the same.
1094             if ($value ne $short_name) {
1095                 my @names_via_full =
1096                             prop_value_aliases($mod_prop, $mod_value);
1097                 is_deeply(\@names_via_full, \@names_via_short, "In '$prop', prop_value_aliases() returns the same list for both '$short_name' and '$mod_value'");
1098             }
1099
1100             # Tests scalar context
1101             is(prop_value_aliases($prop, $short_name), $value, "'$value' is the long name for prop_value_aliases('$prop', '$short_name')");
1102         }
1103         else {
1104             my @all_names = prop_value_aliases($mod_prop, $mod_value);
1105             is_deeply(\@all_names, \@names_via_short, "In '$prop', prop_value_aliases() returns the same list for both '$short_name' and '$mod_value'");
1106             ok((grep { &utf8::_loose_name(lc $_) eq &utf8::_loose_name(lc $value) } prop_value_aliases($prop, $short_name)), "'$value' is listed as an alias for prop_value_aliases('$prop', '$short_name')");
1107         }
1108
1109         $pva_tested{&utf8::_loose_name(lc $prop) . "=" . &utf8::_loose_name(lc $value)} = 1;
1110         $count++;
1111     }
1112 }
1113 }   # End of SKIP block
1114
1115 # And test as best we can, the non-official pva's that mktables generates.
1116 foreach my $hash (\%utf8::loose_to_file_of, \%utf8::stricter_to_file_of) {
1117     foreach my $test (sort keys %$hash) {
1118         next if exists $pva_tested{$test};  # Skip if already tested
1119
1120         my ($prop, $value) = split "=", $test;
1121         next unless defined $value; # prop_value_aliases() requires an input
1122                                     # 'value'
1123         my $mod_value;
1124         if ($hash == \%utf8::loose_to_file_of) {
1125
1126             # Add extra characters to test loose-match rhs value
1127             $mod_value = "$extra_chars$value";
1128         }
1129         else { # Here value is strictly matched.
1130
1131             # Extra elements are added by mktables to this hash so that
1132             # something like "age=6.0" has a synonym of "age=6".  It's not
1133             # clear to me (khw) if we should be encouraging those synonyms, so
1134             # don't test for them.
1135             next if $value !~ /\D/ && exists $hash->{"$prop=$value.0"};
1136
1137             # Verify that loose matching fails when only strict is called for.
1138             next unless is(prop_value_aliases($prop, "$extra_chars$value"), undef,
1139                         "prop_value_aliases('$prop', '$extra_chars$value') returns undef since '$value' should be strictly matched"),
1140
1141             # Strict matching does allow for underscores between digits.  Test
1142             # for that.
1143             $mod_value = $value;
1144             while ($mod_value =~ s/(\d)(\d)/$1_$2/g) {}
1145         }
1146
1147         # The lhs property is always loosely matched, so add in extra
1148         # characters to test that.
1149         my $mod_prop = "$extra_chars$prop";
1150
1151         if ($prop eq 'gc' && $value =~ /l[_&]$/) {
1152             # These two names are special in that they don't appear in the
1153             # returned list because they are discouraged from use.  Verify
1154             # that they return the same list as a non-discouraged version.
1155             my @LC = prop_value_aliases('gc', 'lc');
1156             my @l_ = prop_value_aliases($mod_prop, $mod_value);
1157             is_deeply(\@l_, \@LC, "prop_value_aliases('$mod_prop', '$mod_value) returns the same list as prop_value_aliases('gc', 'lc')");
1158         }
1159         else {
1160             ok((grep { &utf8::_loose_name(lc $_) eq &utf8::_loose_name(lc $value) }
1161                 prop_value_aliases($mod_prop, $mod_value)),
1162                 "'$value' is listed as an alias for prop_value_aliases('$mod_prop', '$mod_value')");
1163         }
1164     }
1165 }
1166
1167 undef %pva_tested;
1168
1169 no warnings 'once'; # We use some values once from 'required' modules.
1170
1171 use Unicode::UCD qw(prop_invlist prop_invmap MAX_CP);
1172
1173 # There were some problems with caching interfering with prop_invlist() vs
1174 # prop_invmap() on binary properties, and also between the 3 properties where
1175 # Perl used the same 'To' name as another property (see utf8_heavy.pl).
1176 # So, before testing all of prop_invlist(),
1177 #   1)  call prop_invmap() to try both orders of these name issues.  This uses
1178 #       up two of the 3 properties;  the third will be left so that invlist()
1179 #       on it gets called before invmap()
1180 #   2)  call prop_invmap() on a generic binary property, ahead of invlist().
1181 # This should test that the caching works in both directions.
1182
1183 # These properties are not stable between Unicode versions, but the first few
1184 # elements are; just look at the first element to see if are getting the
1185 # distinction right.  The general inversion map testing below will test the
1186 # whole thing.
1187
1188 my $prop;
1189 my ($invlist_ref, $invmap_ref, $format, $missing);
1190 if ($::IS_ASCII) { # On EBCDIC, other things will come first, and can vary
1191                 # according to code page
1192     $prop = "uc";
1193     ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
1194     is($format, 'al', "prop_invmap() format of '$prop' is 'al'");
1195     is($missing, '0', "prop_invmap() missing of '$prop' is '0'");
1196     is($invlist_ref->[1], 0x61, "prop_invmap('$prop') list[1] is 0x61");
1197     is($invmap_ref->[1], 0x41, "prop_invmap('$prop') map[1] is 0x41");
1198
1199     $prop = "upper";
1200     ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
1201     is($format, 's', "prop_invmap() format of '$prop' is 's");
1202     is($missing, 'N', "prop_invmap() missing of '$prop' is 'N'");
1203     is($invlist_ref->[1], 0x41, "prop_invmap('$prop') list[1] is 0x41");
1204     is($invmap_ref->[1], 'Y', "prop_invmap('$prop') map[1] is 'Y'");
1205
1206     $prop = "lower";
1207     ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
1208     is($format, 's', "prop_invmap() format of '$prop' is 's'");
1209     is($missing, 'N', "prop_invmap() missing of '$prop' is 'N'");
1210     is($invlist_ref->[1], 0x61, "prop_invmap('$prop') list[1] is 0x61");
1211     is($invmap_ref->[1], 'Y', "prop_invmap('$prop') map[1] is 'Y'");
1212
1213     $prop = "lc";
1214     ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
1215     is($format, 'al', "prop_invmap() format of '$prop' is 'al'");
1216     is($missing, '0', "prop_invmap() missing of '$prop' is '0'");
1217     is($invlist_ref->[1], 0x41, "prop_invmap('$prop') list[1] is 0x41");
1218     is($invmap_ref->[1], 0x61, "prop_invmap('$prop') map[1] is 0x61");
1219 }
1220
1221 # This property is stable and small, so can test all of it
1222 $prop = "ASCII_Hex_Digit";
1223 ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
1224 is($format, 's', "prop_invmap() format of '$prop' is 's'");
1225 is($missing, 'N', "prop_invmap() missing of '$prop' is 'N'");
1226 if ($::IS_ASCII) {
1227     is_deeply($invlist_ref, [ 0x0000, 0x0030, 0x003A,
1228                               0x0041, 0x0047,
1229                               0x0061, 0x0067, 0x110000
1230                             ],
1231           "prop_invmap('$prop') code point list is correct");
1232 }
1233 elsif ($::IS_EBCDIC) {
1234     is_deeply($invlist_ref, [
1235             utf8::unicode_to_native(0x0000),
1236             utf8::unicode_to_native(0x0061), utf8::unicode_to_native(0x0066) + 1,
1237             utf8::unicode_to_native(0x0041), utf8::unicode_to_native(0x0046) + 1,
1238             utf8::unicode_to_native(0x0030), utf8::unicode_to_native(0x0039) + 1,
1239             utf8::unicode_to_native(0x110000)
1240           ],
1241           "prop_invmap('$prop') code point list is correct");
1242 }
1243 is_deeply($invmap_ref, [ 'N', 'Y', 'N', 'Y', 'N', 'Y', 'N', 'N' ] ,
1244           "prop_invmap('$prop') map list is correct");
1245
1246 is(prop_invlist("Unknown property"), undef, "prop_invlist(<Unknown property>) returns undef");
1247 is(prop_invlist(undef), undef, "prop_invlist(undef) returns undef");
1248 is(prop_invlist("Any"), 2, "prop_invlist('Any') returns the number of elements in scalar context");
1249 my @invlist = prop_invlist("Is_Any");
1250 is_deeply(\@invlist, [ 0, 0x110000 ], "prop_invlist works on 'Is_' prefixes");
1251 is(prop_invlist("Is_Is_Any"), undef, "prop_invlist('Is_Is_Any') returns <undef> since two is's");
1252
1253 use Storable qw(dclone);
1254
1255 is(prop_invlist("InKana"), undef, "prop_invlist(<user-defined property returns undef>)");
1256
1257 # The way both the tests for invlist and invmap work is that they take the
1258 # lists returned by the functions and construct from them what the original
1259 # file should look like, which are then compared with the file.  If they are
1260 # identical, the test passes.  What this tests isn't that the results are
1261 # correct, but that invlist and invmap haven't introduced errors beyond what
1262 # are there in the files.  As a small hedge against that, test some
1263 # prop_invlist() tables fully with the known correct result.  We choose
1264 # ASCII_Hex_Digit again, as it is stable.
1265 if ($::IS_ASCII) {
1266     @invlist = prop_invlist("AHex");
1267     is_deeply(\@invlist, [ 0x0030, 0x003A, 0x0041,
1268                                  0x0047, 0x0061, 0x0067 ],
1269           "prop_invlist('AHex') is exactly the expected set of points");
1270     @invlist = prop_invlist("AHex=f");
1271     is_deeply(\@invlist, [ 0x0000, 0x0030, 0x003A, 0x0041,
1272                                  0x0047, 0x0061, 0x0067 ],
1273           "prop_invlist('AHex=f') is exactly the expected set of points");
1274 }
1275 elsif ($::IS_EBCDIC) { # Relies on the ranges 0-9, a-f, and A-F each being
1276                     # contiguous
1277     @invlist = prop_invlist("AHex");
1278     is_deeply(\@invlist, [
1279             utf8::unicode_to_native(0x0061), utf8::unicode_to_native(0x0066) + 1,
1280             utf8::unicode_to_native(0x0041), utf8::unicode_to_native(0x0046) + 1,
1281             utf8::unicode_to_native(0x0030), utf8::unicode_to_native(0x0039) + 1,
1282        ],
1283        "prop_invlist('AHex') is exactly the expected set of points");
1284     @invlist = prop_invlist("AHex=f");
1285     is_deeply(\@invlist, [
1286             utf8::unicode_to_native(0x0000),
1287             utf8::unicode_to_native(0x0061),
1288             utf8::unicode_to_native(0x0066) + 1,
1289             utf8::unicode_to_native(0x0041),
1290             utf8::unicode_to_native(0x0046) + 1,
1291             utf8::unicode_to_native(0x0030),
1292             utf8::unicode_to_native(0x0039) + 1,
1293        ],
1294        "prop_invlist('AHex=f') is exactly the expected set of points");
1295 }
1296
1297 sub fail_with_diff ($$$$) {
1298     # For use below to output better messages
1299     my ($prop, $official, $constructed, $tested_function_name) = @_;
1300
1301     is($constructed, $official, "$tested_function_name('$prop')");
1302     diag("Comment out lines " . (__LINE__ - 1) . " through " . (__LINE__ + 1) . " in '$0' on Un*x-like systems to see just the differences.  Uses the 'diff' first in your \$PATH");
1303     return;
1304
1305     fail("$tested_function_name('$prop')");
1306
1307     require File::Temp;
1308     my $off = File::Temp->new();
1309     local $/ = "\n";
1310     chomp $official;
1311     print $off $official, "\n";
1312     close $off || die "Can't close official";
1313
1314     chomp $constructed;
1315     my $gend = File::Temp->new();
1316     print $gend $constructed, "\n";
1317     close $gend || die "Can't close gend";
1318
1319     my $diff = File::Temp->new();
1320     system("diff $off $gend > $diff");
1321
1322     open my $fh, "<", $diff || die "Can't open $diff";
1323     my @diffs = <$fh>;
1324     diag("In the diff output below '<' marks lines from the filesystem tables;\n'>' are from $tested_function_name()");
1325     diag(@diffs);
1326 }
1327
1328 my %tested_invlist;
1329
1330 # Look at everything we think that mktables tells us exists, both loose and
1331 # strict
1332 foreach my $set_of_tables (\%utf8::stricter_to_file_of, \%utf8::loose_to_file_of)
1333 {
1334     foreach my $table (sort keys %$set_of_tables) {
1335
1336         my $mod_table;
1337         my ($prop_only, $value) = split "=", $table;
1338         if (defined $value) {
1339
1340             # If this is to be loose matched, add in characters to test that.
1341             if ($set_of_tables == \%utf8::loose_to_file_of) {
1342                 $value = "$extra_chars$value";
1343             }
1344             else {  # Strict match
1345
1346                 # Verify that loose matching fails when only strict is called
1347                 # for.
1348                 next unless is(prop_invlist("$prop_only=$extra_chars$value"), undef, "prop_invlist('$prop_only=$extra_chars$value') returns undef since should be strictly matched");
1349
1350                 # Strict matching does allow for underscores between digits.
1351                 # Test for that.
1352                 while ($value =~ s/(\d)(\d)/$1_$2/g) {}
1353             }
1354
1355             # The property portion in compound form specifications always
1356             # matches loosely
1357             $mod_table = "$extra_chars$prop_only = $value";
1358         }
1359         else {  # Single-form.
1360
1361             # Like above, use loose if required, and insert underscores
1362             # between digits if strict.
1363             if ($set_of_tables == \%utf8::loose_to_file_of) {
1364                 $mod_table = "$extra_chars$table";
1365             }
1366             else {
1367                 $mod_table = $table;
1368                 while ($mod_table =~ s/(\d)(\d)/$1_$2/g) {}
1369             }
1370         }
1371
1372         my @tested = prop_invlist($mod_table);
1373         if ($table =~ /^_/) {
1374             is(@tested, 0, "prop_invlist('$mod_table') returns an empty list since is internal-only");
1375             next;
1376         }
1377
1378         # If we have already tested a property that uses the same file, this
1379         # list should be identical to the one that was tested, and can bypass
1380         # everything else.
1381         my $file = $set_of_tables->{$table};
1382         if (exists $tested_invlist{$file}) {
1383             is_deeply(\@tested, $tested_invlist{$file}, "prop_invlist('$mod_table') gave same results as its name synonym");
1384             next;
1385         }
1386         $tested_invlist{$file} = dclone \@tested;
1387
1388         # A '!' in the file name means that it is to be inverted.
1389         my $invert = $file =~ s/!//;
1390         my $official;
1391
1392         # If the file's directory is '#', it is a special case where the
1393         # contents are in-lined with semi-colons meaning new-lines, instead of
1394         # it being an actual file to read.  The file is an index in to the
1395         # array of the definitions
1396         if ($file =~ s!^#/!!) {
1397             $official = $utf8::inline_definitions[$file];
1398         }
1399         else {
1400             $official = do "unicore/lib/$file.pl";
1401         }
1402
1403         # Get rid of any trailing space and comments in the file.
1404         $official =~ s/\s*(#.*)?$//mg;
1405         local $/ = "\n";
1406         chomp $official;
1407         $/ = $input_record_separator;
1408
1409         # If we are to test against an inverted file, it is easier to invert
1410         # our array than the file.
1411         if ($invert) {
1412             if (@tested && $tested[0] == 0) {
1413                 shift @tested;
1414             } else {
1415                 unshift @tested, 0;
1416             }
1417         }
1418
1419         # Now construct a string from the list that should match the file.
1420         # The file is inversion list format code points, like this:
1421         # V1216
1422         # 65      # [26]
1423         # 91
1424         # 192     # [23]
1425         # ...
1426         # The V indicates it's an inversion list, and is followed immediately
1427         # by the number of elements (lines) that follow giving its contents.
1428         # The list has even numbered elements (0th, 2nd, ...) start ranges
1429         # that are in the list, and odd ones that aren't in the list.
1430         # Therefore the odd numbered ones are one beyond the end of the
1431         # previous range, but otherwise don't get reflected in the file.
1432         my $tested =  join "\n", ("V" . scalar @tested), @tested;
1433         local $/ = "\n";
1434         chomp $tested;
1435         $/ = $input_record_separator;
1436         if ($tested ne $official) {
1437             fail_with_diff($mod_table, $official, $tested, "prop_invlist");
1438             next;
1439         }
1440
1441         pass("prop_invlist('$mod_table')");
1442     }
1443 }
1444
1445 # Now test prop_invmap().
1446
1447 @list = prop_invmap("Unknown property");
1448 is (@list, 0, "prop_invmap(<Unknown property>) returns an empty list");
1449 @list = prop_invmap(undef);
1450 is (@list, 0, "prop_invmap(undef) returns an empty list");
1451 ok (! eval "prop_invmap('gc')" && $@ ne "",
1452                                 "prop_invmap('gc') dies in scalar context");
1453 @list = prop_invmap("_X_Begin");
1454 is (@list, 0, "prop_invmap(<internal property>) returns an empty list");
1455 @list = prop_invmap("InKana");
1456 is(@list, 0, "prop_invmap(<user-defined property returns undef>)");
1457 @list = prop_invmap("Perl_Decomposition_Mapping"), undef,
1458 is(@list, 0, "prop_invmap('Perl_Decomposition_Mapping') returns <undef> since internal-Perl-only");
1459 @list = prop_invmap("Perl_Charnames"), undef,
1460 is(@list, 0, "prop_invmap('Perl_Charnames') returns <undef> since internal-Perl-only");
1461 @list = prop_invmap("Is_Is_Any");
1462 is(@list, 0, "prop_invmap('Is_Is_Any') returns <undef> since two is's");
1463
1464 # The files for these properties are not used by Perl, but are retained for
1465 # backwards compatibility with applications that read them directly, with
1466 # comments in them that their use is deprecated.  Until such time as we remove
1467 # them completely, we test that they exist, are correct, and that their
1468 # formats haven't changed.  This hash contains the info needed to test them as
1469 # if they were regular properties.  'replaced_by' gives the equivalent
1470 # property now used by Perl.
1471 my %legacy_props = (
1472             Legacy_Case_Folding =>        { replaced_by => 'cf',
1473                                             file => 'To/Fold',
1474                                             swash_name => 'ToFold'
1475                                           },
1476             Legacy_Lowercase_Mapping =>   { replaced_by => 'lc',
1477                                             file => 'To/Lower',
1478                                             swash_name => 'ToLower'
1479                                           },
1480             Legacy_Titlecase_Mapping =>   { replaced_by => 'tc',
1481                                             file => 'To/Title',
1482                                             swash_name => 'ToTitle'
1483                                           },
1484             Legacy_Uppercase_Mapping =>   { replaced_by => 'uc',
1485                                             file => 'To/Upper',
1486                                             swash_name => 'ToUpper'
1487                                           },
1488             Legacy_Perl_Decimal_Digit =>  { replaced_by => 'Perl_Decimal_Digit',
1489                                             file => 'To/Digit',
1490                                             swash_name => 'ToDigit'
1491                                            },
1492         );
1493
1494 foreach my $legacy_prop (keys %legacy_props) {
1495     @list = prop_invmap($legacy_prop);
1496     is(@list, 0, "'$legacy_prop' is unknown to prop_invmap");
1497 }
1498
1499 # The files for these properties shouldn't have their formats changed in case
1500 # applications use them (though such use is deprecated).
1501 my @legacy_file_format = (keys %legacy_props,
1502                           qw( Bidi_Mirroring_Glyph
1503                               NFKC_Casefold
1504                            )
1505                           );
1506
1507 # The set of properties to test on has already been compiled into %props by
1508 # the prop_aliases() tests.
1509
1510 my %tested_invmaps;
1511
1512 # Like prop_invlist(), prop_invmap() is tested by comparing the results
1513 # returned by the function with the tables that mktables generates.  Some of
1514 # these tables are directly stored as files on disk, in either the unicore or
1515 # unicore/To directories, and most should be listed in the mktables generated
1516 # hash %utf8::loose_property_to_file_of, with a few additional ones that this
1517 # handles specially.  For these, the files are read in directly, massaged, and
1518 # compared with what invmap() returns.  The SPECIALS hash in some of these
1519 # files overrides values in the main part of the file.
1520 #
1521 # The other properties are tested indirectly by generating all the possible
1522 # inversion lists for the property, and seeing if those match the inversion
1523 # lists returned by prop_invlist(), which has already been tested.
1524
1525 PROPERTY:
1526 foreach my $prop (sort(keys %props), sort keys %legacy_props) {
1527     my $is_legacy = 0;
1528     my $loose_prop = &utf8::_loose_name(lc $prop);
1529     my $suppressed = grep { $_ eq $loose_prop }
1530                           @Unicode::UCD::suppressed_properties;
1531
1532     my $actual_lookup_prop;
1533     my $display_prop;        # The property name that is displayed, as opposed
1534                              # to the one that is actually used.
1535
1536     # Find the short and full names that this property goes by
1537     my ($name, $full_name) = prop_aliases($prop);
1538     if (! $name) {
1539
1540         # Here, Perl doesn't know about this property.  It could be a
1541         # suppressed one, or a legacy one.
1542         if (grep { $prop eq $_ } keys %legacy_props) {
1543
1544             # For legacy properties, we look up the modern equivalent
1545             # property instead; later massaging the results to look like the
1546             # known format of the legacy property.  We add info about the
1547             # legacy property to the data structures for the rest of the
1548             # properties; this is to avoid more special cases for the legacies
1549             # in the code below
1550             $full_name = $name = $prop;
1551             $actual_lookup_prop = $legacy_props{$prop}->{'replaced_by'};
1552             my $base_file = $legacy_props{$prop}->{'file'};
1553
1554             # This legacy property is otherwise unknown to Perl; so shouldn't
1555             # have any information about it already.
1556             ok(! exists $utf8::loose_property_to_file_of{$loose_prop},
1557                "There isn't a hash entry for file lookup of $prop");
1558             $utf8::loose_property_to_file_of{$loose_prop} = $base_file;
1559
1560             ok(! exists $utf8::file_to_swash_name{$loose_prop},
1561                "There isn't a hash entry for swash lookup of $prop");
1562             $utf8::file_to_swash_name{$base_file}
1563                                         = $legacy_props{$prop}->{'swash_name'};
1564             $display_prop = $prop;
1565             $is_legacy = 1;
1566         }
1567         else {
1568             if (! $suppressed) {
1569                 fail("prop_invmap('$prop')");
1570                 diag("is unknown to prop_aliases(), and we need it in order to test prop_invmap");
1571             }
1572         next PROPERTY;
1573         }
1574     }
1575
1576     # Normalize the short name, as it is stored in the hashes under the
1577     # normalized version.
1578     $name = &utf8::_loose_name(lc $name);
1579
1580     # Add in the characters that are supposed to be ignored to test loose
1581     # matching, which the tested function applies to all properties
1582     $display_prop = "$extra_chars$prop" unless $display_prop;
1583     $actual_lookup_prop = $display_prop unless $actual_lookup_prop;
1584
1585     my ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($actual_lookup_prop);
1586     my $return_ref = [ $invlist_ref, $invmap_ref, $format, $missing ];
1587
1588
1589     # The legacy property files all are expanded out so that each range is 1
1590     # element long.  That isn't true of the modern equivalent we use to check
1591     # those files for correctness against.  So take the output of the proxy
1592     # and expand it to match the legacy file.
1593     if ($is_legacy) {
1594         my @expanded_list;
1595         my @expanded_map;
1596         for my $i (0 .. @$invlist_ref - 1 - 1) {
1597             if (ref $invmap_ref->[$i] || $invmap_ref->[$i] eq $missing) {
1598
1599                 # No adjustments should be done for the default mapping and
1600                 # the multi-char ones.
1601                 push @expanded_list, $invlist_ref->[$i];
1602                 push @expanded_map, $invmap_ref->[$i];
1603             }
1604             else {
1605
1606                 # Expand the range into separate elements for each item.
1607                 my $offset = 0;
1608                 for my $j ($invlist_ref->[$i] .. $invlist_ref->[$i+1] -1) {
1609                     push @expanded_list, $j;
1610                     push @expanded_map, $invmap_ref->[$i] + $offset;
1611
1612                     # The 'ae' format is for Legacy_Perl_Decimal_Digit; the
1613                     # other 4 are kept with leading zeros in the file, so
1614                     # convert to that.
1615                     $expanded_map[-1] = sprintf("%04X", $expanded_map[-1])
1616                                                             if $format ne 'ae';
1617                     $offset++;
1618                 }
1619             }
1620         }
1621
1622         # Final element is taken as is.  The map should always be to the
1623         # default value, so don't do a sprintf like we did above.
1624         push @expanded_list, $invlist_ref->[-1];
1625         push @expanded_map, $invmap_ref->[-1];
1626
1627         $invlist_ref = \@expanded_list;
1628         $invmap_ref = \@expanded_map;
1629     }
1630
1631     # If have already tested this property under a different name, merely
1632     # compare the return from now with the saved one from before.
1633     if (exists $tested_invmaps{$name}) {
1634         is_deeply($return_ref, $tested_invmaps{$name}, "prop_invmap('$display_prop') gave same results as its synonym, '$name'");
1635         next PROPERTY;
1636     }
1637     $tested_invmaps{$name} = dclone $return_ref;
1638
1639     # If prop_invmap() returned nothing, is ok iff is a property whose file is
1640     # not generated.
1641     if ($suppressed) {
1642         if (defined $format) {
1643             fail("prop_invmap('$display_prop')");
1644             diag("did not return undef for suppressed property $prop");
1645         }
1646         next PROPERTY;
1647     }
1648     elsif (!defined $format) {
1649         fail("prop_invmap('$display_prop')");
1650         diag("'$prop' is unknown to prop_invmap()");
1651         next PROPERTY;
1652     }
1653
1654     # The two parallel arrays must have the same number of elements.
1655     if (@$invlist_ref != @$invmap_ref) {
1656         fail("prop_invmap('$display_prop')");
1657         diag("invlist has "
1658              . scalar @$invlist_ref
1659              . " while invmap has "
1660              . scalar @$invmap_ref
1661              . " elements");
1662         next PROPERTY;
1663     }
1664
1665     # The last element must be for the above-Unicode code points, and must be
1666     # for the default value.
1667     if ($invlist_ref->[-1] != 0x110000) {
1668         fail("prop_invmap('$display_prop')");
1669         diag("The last inversion list element is not 0x110000");
1670         next PROPERTY;
1671     }
1672
1673     my $upper_limit_subtract;
1674
1675     # prop_invmap() adds an extra element not present in the disk files for
1676     # the above-Unicode code points.  For almost all properties, that will be
1677     # to $missing.  In that case we don't look further at it when comparing
1678     # with the disk files.
1679     if ($invmap_ref->[-1] eq $missing) {
1680         $upper_limit_subtract = 1;
1681     }
1682     elsif ($invmap_ref->[-1] eq 'Y' && ! grep { $_ !~ /[YN]/ } @$invmap_ref) {
1683
1684         # But that's not true for a few binary properties like 'Unassigned'
1685         # that are Perl extensions (in this case for Gc=Unassigned) which
1686         # match above-Unicode code points (hence the 'Y' in the test above).
1687         # For properties where it isn't $missing, we're going to want to look
1688         # at the whole thing when comparing with the disk file.
1689         $upper_limit_subtract = 0;
1690
1691         # In those properties like 'Unassigned, the final element should be
1692         # just a repetition of the next-to-last element, and won't be in the
1693         # disk file, so remove it for the comparison.  Otherwise, we will
1694         # compare the whole of the array with the whole of the disk file.
1695         if ($invlist_ref->[-2] <= 0x10FFFF && $invmap_ref->[-2] eq 'Y') {
1696             pop @$invlist_ref;
1697             pop @$invmap_ref;
1698         }
1699     }
1700     else {
1701         fail("prop_invmap('$display_prop')");
1702         diag("The last inversion list element is '$invmap_ref->[-1]', and should be '$missing'");
1703         next PROPERTY;
1704     }
1705
1706     if ($name eq 'bmg') {   # This one has an atypical $missing
1707         if ($missing ne "") {
1708             fail("prop_invmap('$display_prop')");
1709             diag("The missings should be \"\"; got '$missing'");
1710             next PROPERTY;
1711         }
1712     }
1713     elsif ($format =~ /^ a (?!r) /x) {
1714         if ($full_name eq 'Perl_Decimal_Digit') {
1715             if ($missing ne "") {
1716                 fail("prop_invmap('$display_prop')");
1717                 diag("The missings should be \"\"; got '$missing'");
1718                 next PROPERTY;
1719             }
1720         }
1721         elsif ($missing ne "0" && ! grep { $prop eq $_ } keys %legacy_props) {
1722             fail("prop_invmap('$display_prop')");
1723             diag("The missings should be '0'; got '$missing'");
1724             next PROPERTY;
1725         }
1726     }
1727     elsif ($missing =~ /[<>]/) {
1728         fail("prop_invmap('$display_prop')");
1729         diag("The missings should NOT be something with <...>'");
1730         next PROPERTY;
1731
1732         # I don't want to hard code in what all the missings should be, so
1733         # those don't get fully tested.
1734     }
1735
1736     # Certain properties don't have their own files, but must be constructed
1737     # using proxies.
1738     my $proxy_prop = $name;
1739     if ($full_name eq 'Present_In') {
1740         $proxy_prop = "age";    # The maps for these two props are identical
1741     }
1742     elsif ($full_name eq 'Simple_Case_Folding'
1743            || $full_name =~ /Simple_ (.) .*? case_Mapping  /x)
1744     {
1745         if ($full_name eq 'Simple_Case_Folding') {
1746             $proxy_prop = 'cf';
1747         }
1748         else {
1749             # We captured the U, L, or T, leading to uc, lc, or tc.
1750             $proxy_prop = lc $1 . "c";
1751         }
1752         if ($format ne "a") {
1753             fail("prop_invmap('$display_prop')");
1754             diag("The format should be 'a'; got '$format'");
1755             next PROPERTY;
1756         }
1757     }
1758
1759     if ($format !~ / ^ (?: a [der]? | ale? | n | sl? ) $ /x) {
1760         fail("prop_invmap('$display_prop')");
1761         diag("Unknown format '$format'");
1762         next PROPERTY;
1763     }
1764
1765     my $base_file;
1766     my $official;
1767
1768     # Handle the properties that have full disk files for them (except the
1769     # Name property which is structurally enough different that it is handled
1770     # separately below.)
1771     if ($name ne 'na'
1772         && ($name eq 'blk'
1773             || defined
1774                     ($base_file = $utf8::loose_property_to_file_of{$proxy_prop})
1775             || exists $utf8::loose_to_file_of{$proxy_prop}
1776             || $name eq "dm"))
1777     {
1778         # In the above, blk is done unconditionally, as we need to test that
1779         # the old-style block names are returned, even if mktables has
1780         # generated a file for the new-style; the test for dm comes afterward,
1781         # so that if a file has been generated for it explicitly, we use that
1782         # file (which is valid, unlike blk) instead of the combo
1783         # Decomposition.pl files.
1784         my $file;
1785         my $is_binary = 0;
1786         if ($name eq 'blk') {
1787
1788             # The blk property is special.  The original file with old block
1789             # names is retained, and the default (on ASCII platforms) is to
1790             # not write out a new-name file.  What we do is get the old names
1791             # into a data structure, and from that create what the new file
1792             # would look like.  $base_file is needed to be defined, just to
1793             # avoid a message below.
1794             $base_file = "This is a dummy name";
1795             my $blocks_ref = charblocks();
1796
1797             if ($::IS_EBCDIC) {
1798                 # On EBCDIC, the first two blocks can each contain multiple
1799                 # ranges.  We create a new version with each of these
1800                 # flattened, so have one level.  ($index is used as a dummy
1801                 # key.)
1802                 my %new_blocks;
1803                 my $index = 0;
1804                 foreach my $block (values %$blocks_ref) {
1805                     foreach my $range (@$block) {
1806                         $new_blocks{$index++}[0] = $range;
1807                     }
1808                 }
1809                 $blocks_ref = \%new_blocks;
1810             }
1811             $official = "";
1812             for my $range (sort { $a->[0][0] <=> $b->[0][0] }
1813                            values %$blocks_ref)
1814             {
1815                 # Translate the charblocks() data structure to what the file
1816                 # would look like.  (The sub range is for EBCDIC platforms
1817                 # where Latin1 and ASCII are intermixed.)
1818                 if ($range->[0][0] == $range->[0][1]) {
1819                     $official .= sprintf("%X\t\t%s\n",
1820                                          $range->[0][0],
1821                                          $range->[0][2]);
1822                 }
1823                 else {
1824                     $official .= sprintf("%X\t%X\t%s\n",
1825                                          $range->[0][0],
1826                                          $range->[0][1],
1827                                          $range->[0][2]);
1828                 }
1829             }
1830         }
1831         else {
1832             $base_file = "Decomposition" if $format eq 'ad';
1833
1834             # Above leaves $base_file undefined only if it came from the hash
1835             # below.  This should happen only when it is a binary property
1836             # (and are accessing via a single-form name, like 'In_Latin1'),
1837             # and so it is stored in a different directory than the To ones.
1838             # XXX Currently, the only cases where it is complemented are the
1839             # ones that have no code points.  And it works out for these that
1840             # 1) complementing them, and then 2) adding or subtracting the
1841             # initial 0 and final 110000 cancel each other out.  But further
1842             # work would be needed in the unlikely event that an inverted
1843             # property comes along without these characteristics
1844             if (!defined $base_file) {
1845                 $base_file = $utf8::loose_to_file_of{$proxy_prop};
1846                 $is_binary = ($base_file =~ s/!//) ? -1 : 1;
1847                 $base_file = "lib/$base_file" unless $base_file =~ m!^#/!;
1848             }
1849
1850             # Read in the file.  If the file's directory is '#', it is a
1851             # special case where the contents are in-lined with semi-colons
1852             # meaning new-lines, instead of it being an actual file to read.
1853             if ($base_file =~ s!^#/!!) {
1854                 $official = $utf8::inline_definitions[$base_file];
1855             }
1856             else {
1857                 $official = do "unicore/$base_file.pl";
1858             }
1859
1860             # Get rid of any trailing space and comments in the file.
1861             $official =~ s/\s*(#.*)?$//mg;
1862
1863             if ($format eq 'ad') {
1864                 my @official = split /\n/, $official;
1865                 $official = "";
1866                 foreach my $line (@official) {
1867                     my ($start, $end, $value)
1868                                     = $line =~ / ^ (.+?) \t (.*?) \t (.+?)
1869                                                 \s* ( \# .* )? $ /x;
1870                     # Decomposition.pl also has the <compatible> types in it,
1871                     # which should be removed.
1872                     $value =~ s/<.*?> //;
1873                     $official .= "$start\t\t$value\n";
1874
1875                     # If this is a multi-char range, we turn it into as many
1876                     # single character ranges as necessary.  This makes things
1877                     # easier below.
1878                     if ($end ne "") {
1879                         for my $i (hex($start) + 1 .. hex $end) {
1880                             $official .= sprintf "%X\t\t%s\n", $i, $value;
1881                         }
1882                     }
1883                 }
1884             }
1885         }
1886         local $/ = "\n";
1887         chomp $official;
1888         $/ = $input_record_separator;
1889
1890         # Get the format for the file, and if there are any special elements,
1891         # get a reference to them.
1892         my $swash_name = $utf8::file_to_swash_name{$base_file};
1893         my $specials_ref;
1894         my $file_format;    # The 'format' given inside the file
1895         if ($swash_name) {
1896             $specials_ref = $utf8::SwashInfo{$swash_name}{'specials_name'};
1897             if ($specials_ref) {
1898
1899                 # Convert from the name to the actual reference.
1900                 no strict 'refs';
1901                 $specials_ref = \%{$specials_ref};
1902             }
1903
1904             $file_format = $utf8::SwashInfo{$swash_name}{'format'};
1905         }
1906
1907         # Leading zeros used to be used with the values in the files that give,
1908         # ranges, but these have been mostly stripped off, except for some
1909         # files whose formats should not change in any way.
1910         my $file_range_format = (grep { $full_name eq $_ } @legacy_file_format)
1911                               ? "%04X"
1912                               : "%X";
1913         # Currently this property still has leading zeroes in the mapped-to
1914         # values, but otherwise, those values follow the same rules as the
1915         # ranges.
1916         my $file_map_format = ($full_name eq 'Decomposition_Mapping')
1917                               ? "%04X"
1918                               : $file_range_format;
1919
1920         # Certain of the proxy properties have to be adjusted to match the
1921         # real ones.
1922         if ($full_name
1923                  =~ /^(Legacy_)?(Case_Folding|(Lower|Title|Upper)case_Mapping)/)
1924         {
1925
1926             # Here we have either
1927             #   1) Case_Folding; or
1928             #   2) a proxy that is a full mapping, which means that what the
1929             #      real property is is the equivalent simple mapping.
1930             # In both cases, the file will have a standard list containing
1931             # simple mappings (to a single code point), and a specials hash
1932             # which contains all the mappings that are to multiple code
1933             # points.  First, extract a list containing all the file's simple
1934             # mappings.
1935             my @list;
1936             for (split "\n", $official) {
1937                 my ($start, $end, $value) = / ^ (.+?) \t (.*?) \t (.+?)
1938                                                 \s* ( \# .* )? $ /x;
1939                 $end = $start if $end eq "";
1940                 push @list, [ hex $start, hex $end, hex $value ];
1941             }
1942
1943             # For these mappings, the file contains all the simple mappings,
1944             # including the ones that are overridden by the specials.  These
1945             # need to be removed as the list is for just the full ones.
1946
1947             # Go through any special mappings one by one.  The keys are the
1948             # UTF-8 representation of code points.
1949             my $i = 0;
1950             foreach my $utf8_cp (sort keys %$specials_ref) {
1951                 my $cp = $utf8_cp;
1952                 utf8::decode($cp);
1953                 $cp = ord $cp;
1954
1955                 # Find the spot in the @list of simple mappings that this
1956                 # special applies to; uses a linear search.
1957                 while ($i < @list -1 ) {
1958                     last if  $cp <= $list[$i][1];
1959                     $i++;
1960                 }
1961
1962                 # Here $i is such that it points to the first range which ends
1963                 # at or above cp, and hence is the only range that could
1964                 # possibly contain it.
1965
1966                 # If not in this range, no range contains it: nothing to
1967                 # remove.
1968                 next if $cp < $list[$i][0];
1969
1970                 # Otherwise, remove the existing entry.  If it is the first
1971                 # element of the range...
1972                 if ($cp == $list[$i][0]) {
1973
1974                     # ... and there are other elements in the range, just
1975                     # shorten the range to exclude this code point.
1976                     if ($list[$i][1] > $list[$i][0]) {
1977                         $list[$i][0]++;
1978                     }
1979
1980                     # ... but if it is the only element in the range, remove
1981                     # it entirely.
1982                     else {
1983                         splice @list, $i, 1;
1984                     }
1985                 }
1986                 else { # Is somewhere in the middle of the range
1987                     # Split the range into two, excluding this one in the
1988                     # middle
1989                     splice @list, $i, 1,
1990                            [ $list[$i][0], $cp - 1, $list[$i][2] ],
1991                            [ $cp + 1, $list[$i][1], $list[$i][2] ];
1992                 }
1993             }
1994
1995             # Here, have gone through all the specials, modifying @list as
1996             # needed.  Turn it back into what the file should look like.
1997             $official = "";
1998             for my $element (@list) {
1999                 $official .= "\n" if $official;
2000                 if ($element->[1] == $element->[0]) {
2001                     $official
2002                         .= sprintf "$file_range_format\t\t$file_map_format",
2003                                     $element->[0],        $element->[2];
2004                 }
2005                 else {
2006                     $official .= sprintf "$file_range_format\t$file_range_format\t$file_map_format",
2007                                          $element->[0],
2008                                          $element->[1],
2009                                          $element->[2];
2010                 }
2011             }
2012         }
2013         elsif ($full_name
2014             =~ / ^ Simple_(Case_Folding|(Lower|Title|Upper)case_Mapping) $ /x)
2015         {
2016
2017             # These properties have everything in the regular array, and the
2018             # specials are superfluous.
2019             undef $specials_ref;
2020         }
2021         elsif ($format !~ /^a/ && defined $file_format && $file_format eq 'x') {
2022
2023             # For these properties the file is output using hex notation for the
2024             # map.  Convert from hex to decimal.
2025             my @lines = split "\n", $official;
2026             foreach my $line (@lines) {
2027                 my ($lower, $upper, $map) = split "\t", $line;
2028                 $line = "$lower\t$upper\t" . hex $map;
2029             }
2030             $official = join "\n", @lines;
2031         }
2032
2033         # Here, in $official, we have what the file looks like, or should like
2034         # if we've had to fix it up.  Now take the invmap() output and reverse
2035         # engineer from that what the file should look like.  Each iteration
2036         # appends the next line to the running string.
2037         my $tested_map = "";
2038
2039         # For use with files for binary properties only, which are stored in
2040         # inversion list format.  This counts the number of data lines in the
2041         # file.
2042         my $binary_count = 0;
2043
2044         # Create a copy of the file's specials hash.  (It has been undef'd if
2045         # we know it isn't relevant to this property, so if it exists, it's an
2046         # error or is relevant).  As we go along, we delete from that copy.
2047         # If a delete fails, or something is left over after we are done,
2048         # it's an error
2049         my %specials = %$specials_ref if $specials_ref;
2050
2051         # The extra -$upper_limit_subtract is because the final element may
2052         # have been tested above to be for anything above Unicode, in which
2053         # case the file may not go that high.
2054         for (my $i = 0; $i < @$invlist_ref - $upper_limit_subtract; $i++) {
2055
2056             # If the map element is a reference, have to stringify it (but
2057             # don't do so if the format doesn't allow references, so that an
2058             # improper format will generate an error.
2059             if (ref $invmap_ref->[$i]
2060                 && ($format eq 'ad' || $format =~ /^ . l /x))
2061             {
2062                 # The stringification depends on the format.
2063                 if ($format eq 'sl') {
2064
2065                     # At the time of this writing, there are two types of 'sl'
2066                     # format  One, in Name_Alias, has multiple separate
2067                     # entries for each code point; the other, in
2068                     # Script_Extension, is space separated.  Assume the latter
2069                     # for non-Name_Alias.
2070                     if ($full_name ne 'Name_Alias') {
2071                         $invmap_ref->[$i] = join " ", @{$invmap_ref->[$i]};
2072                     }
2073                     else {
2074                         # For Name_Alias, we emulate the file.  Entries with
2075                         # just one value don't need any changes, but we
2076                         # convert the list entries into a series of lines for
2077                         # the file, starting with the first name.  The
2078                         # succeeding entries are on separate lines, with the
2079                         # code point repeated for each one and then two tabs,
2080                         # then the value.  Code at the end of the loop will
2081                         # set up the first line with its code point and two
2082                         # tabs before the value, just as it does for every
2083                         # other property; thus the special handling of the
2084                         # first line.
2085                         if (ref $invmap_ref->[$i]) {
2086                             my $hex_cp = sprintf("%X", $invlist_ref->[$i]);
2087                             my $concatenated = $invmap_ref->[$i][0];
2088                             for (my $j = 1; $j < @{$invmap_ref->[$i]}; $j++) {
2089                                 $concatenated .= "\n$hex_cp\t\t"
2090                                               .  $invmap_ref->[$i][$j];
2091                             }
2092                             $invmap_ref->[$i] = $concatenated;
2093                         }
2094                     }
2095                 }
2096                 elsif ($format =~ / ^ al e? $/x) {
2097
2098                     # For an al property, the stringified result should be in
2099                     # the specials hash.  The key is the utf8 bytes of the
2100                     # code point, and the value is its map as a utf-8 string.
2101                     my $value;
2102                     my $key = chr $invlist_ref->[$i];
2103                     utf8::encode($key);
2104                     if (! defined ($value = delete $specials{$key})) {
2105                         fail("prop_invmap('$display_prop')");
2106                         diag(sprintf "There was no specials element for %04X", $invlist_ref->[$i]);
2107                         next PROPERTY;
2108                     }
2109                     my $packed = pack "W*", @{$invmap_ref->[$i]};
2110                     utf8::upgrade($packed);
2111                     if ($value ne $packed) {
2112                         fail("prop_invmap('$display_prop')");
2113                         diag(sprintf "For %04X, expected the mapping to be "
2114                          . "'$packed', but got '$value'", $invlist_ref->[$i]);
2115                         next PROPERTY;
2116                     }
2117
2118                     # As this doesn't get tested when we later compare with
2119                     # the actual file, it could be out of order and we
2120                     # wouldn't know it.
2121                     if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
2122                         || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
2123                     {
2124                         fail("prop_invmap('$display_prop')");
2125                         diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
2126                         next PROPERTY;
2127                     }
2128                     next;
2129                 }
2130                 elsif ($format eq 'ad') {
2131
2132                     # The decomposition mapping file has the code points as
2133                     # a string of space-separated hex constants.
2134                     $invmap_ref->[$i] = join " ", map { sprintf "%04X", $_ }
2135                                                            @{$invmap_ref->[$i]};
2136                 }
2137                 else {
2138                     fail("prop_invmap('$display_prop')");
2139                     diag("Can't handle format '$format'");
2140                     next PROPERTY;
2141                 }
2142             } # Otherwise, the map is to a simple scalar
2143             elsif (defined $file_format && $file_format eq 'ax') {
2144                 # These maps are in hex
2145                 $invmap_ref->[$i] = sprintf("%X", $invmap_ref->[$i]);
2146             }
2147             elsif ($format eq 'ad' || $format eq 'ale') {
2148
2149                 # The numerics in the returned map are stored as adjusted
2150                 # decimal integers.  The defaults are 0, and don't appear in
2151                 # $official, and are excluded later, but the elements must be
2152                 # converted back to their hex values before comparing with
2153                 # $official, as these files, for backwards compatibility, are
2154                 # not stored as adjusted.  (There currently is only one ale
2155                 # property, nfkccf.  If that changed this would also have to.)
2156                 if ($invmap_ref->[$i] =~ / ^ -? \d+ $ /x
2157                     && $invmap_ref->[$i] != 0)
2158                 {
2159                     my $next = $invmap_ref->[$i] + 1;
2160                     $invmap_ref->[$i] = sprintf($file_map_format,
2161                                                 $invmap_ref->[$i]);
2162
2163                     # If there are other elements in this range they need to
2164                     # be adjusted; they must individually be re-mapped.  Do
2165                     # this by splicing in a new element into the list and the
2166                     # map containing the remainder of the range.  Next time
2167                     # through we will look at that (possibly splicing again
2168                     # until the whole range is processed).
2169                     if ($invlist_ref->[$i+1] > $invlist_ref->[$i] + 1) {
2170                         splice @$invlist_ref, $i+1, 0,
2171                                 $invlist_ref->[$i] + 1;
2172                         splice @$invmap_ref, $i+1, 0, $next;
2173                     }
2174                 }
2175                 if ($format eq 'ale' && $invmap_ref->[$i] eq "") {
2176
2177                     # ale properties have maps to the empty string that also
2178                     # should be in the specials hash, with the key the utf8
2179                     # bytes representing the code point, and the map just empty.
2180                     my $value;
2181                     my $key = chr $invlist_ref->[$i];
2182                     utf8::encode($key);
2183                     if (! defined ($value = delete $specials{$key})) {
2184                         fail("prop_invmap('$display_prop')");
2185                         diag(sprintf "There was no specials element for %04X", $invlist_ref->[$i]);
2186                         next PROPERTY;
2187                     }
2188                     if ($value ne "") {
2189                         fail("prop_invmap('$display_prop')");
2190                         diag(sprintf "For %04X, expected the mapping to be \"\", but got '$value'", $invlist_ref->[$i]);
2191                         next PROPERTY;
2192                     }
2193
2194                     # As this doesn't get tested when we later compare with
2195                     # the actual file, it could be out of order and we
2196                     # wouldn't know it.
2197                     if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
2198                         || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
2199                     {
2200                         fail("prop_invmap('$display_prop')");
2201                         diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
2202                         next PROPERTY;
2203                     }
2204                     next;
2205                 }
2206             }
2207             elsif ($is_binary) { # These binary files don't have an explicit Y
2208                 $invmap_ref->[$i] =~ s/Y//;
2209             }
2210
2211             # The file doesn't include entries that map to $missing, so don't
2212             # include it in the built-up string.  But make sure that it is in
2213             # the correct order in the input.
2214             if ($invmap_ref->[$i] eq $missing) {
2215                 if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
2216                     || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
2217                 {
2218                     fail("prop_invmap('$display_prop')");
2219                     diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
2220                     next PROPERTY;
2221                 }
2222                 next;
2223             }
2224
2225             # The ad property has one entry which isn't in the file.
2226             # Ignore it, but make sure it is in order.
2227             if ($format eq 'ad'
2228                 && $invmap_ref->[$i] eq '<hangul syllable>'
2229                 && $invlist_ref->[$i] == 0xAC00)
2230             {
2231                 if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
2232                     || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
2233                 {
2234                     fail("prop_invmap('$display_prop')");
2235                     diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
2236                     next PROPERTY;
2237                 }
2238                 next;
2239             }
2240
2241             # Finally have figured out what the map column in the file should
2242             # be.  Append the line to the running string.
2243             my $start = $invlist_ref->[$i];
2244             my $end = (defined $invlist_ref->[$i+1])
2245                       ? $invlist_ref->[$i+1] - 1
2246                       : $Unicode::UCD::MAX_CP;
2247             if ($is_binary) {
2248
2249                 # Files for binary properties are in inversion list format,
2250                 # without ranges.
2251                 $tested_map .= "$start\n";
2252                 $binary_count++;
2253
2254                 # If the final value is infinity, no line for it exists.
2255                 if ($end < $Unicode::UCD::MAX_CP) {
2256                     $tested_map .= ($end + 1) . "\n";
2257                     $binary_count++;
2258                 }
2259             }
2260             else {
2261                 $end = ($start == $end) ? "" : sprintf($file_range_format, $end);
2262                 if ($invmap_ref->[$i] ne "") {
2263                     $tested_map .= sprintf "$file_range_format\t%s\t%s\n",
2264                                             $start, $end, $invmap_ref->[$i];
2265                 }
2266                 elsif ($end ne "") {
2267                     $tested_map .= sprintf "$file_range_format\t%s\n",
2268                                             $start,             $end;
2269                 }
2270                 else {
2271                     $tested_map .= sprintf "$file_range_format\n", $start;
2272                 }
2273             }
2274         } # End of looping over all elements.
2275
2276         # Binary property files begin with a line count line.
2277         $tested_map = "V$binary_count\n$tested_map" if $binary_count;
2278
2279         # Here are done with generating what the file should look like
2280
2281         local $/ = "\n";
2282         chomp $tested_map;
2283         $/ = $input_record_separator;
2284
2285         # And compare.
2286         if ($tested_map ne $official) {
2287             fail_with_diff($display_prop, $official, $tested_map, "prop_invmap");
2288             next PROPERTY;
2289         }
2290
2291         # There shouldn't be any specials unaccounted for.
2292         if (keys %specials) {
2293             fail("prop_invmap('$display_prop')");
2294             diag("Unexpected specials: " . join ", ", keys %specials);
2295             next PROPERTY;
2296         }
2297     }
2298     elsif ($format eq 'n') {
2299
2300         # Handle the Name property similar to the above.  But the file is
2301         # sufficiently different that it is more convenient to make a special
2302         # case for it.  It is a combination of the Name, Unicode1_Name, and
2303         # Name_Alias properties, and named sequences.  We need to remove all
2304         # but the Name in order to do the comparison.
2305
2306         if ($missing ne "") {
2307             fail("prop_invmap('$display_prop')");
2308             diag("The missings should be \"\"; got \"missing\"");
2309             next PROPERTY;
2310         }
2311
2312         $official = do "unicore/Name.pl";
2313
2314         # Get rid of the named sequences portion of the file.  These don't
2315         # have a tab before the first blank on a line.
2316         $official =~ s/ ^ [^\t]+ \  .*? \n //xmg;
2317
2318         # And get rid of the controls.  These are named in the file, but
2319         # shouldn't be in the property.  This gets rid of the two ranges in
2320         # one fell swoop, and also all the Unicode1_Name values that may not
2321         # be in Name_Alias.
2322         if ($::IS_ASCII) {
2323             $official =~ s/ 00000 \t .* 0001F .*? \n//xs;
2324             $official =~ s/ 0007F \t .* 0009F .*? \n//xs;
2325         }
2326         elsif ($::IS_EBCDIC) { # Won't work for POSIX-BC
2327             $official =~ s/ 00000 \t .* 0003F .*? \n//xs;
2328             $official =~ s/ 000FF \t .* 000FF .*? \n//xs;
2329         }
2330
2331         # And remove the aliases.  We read in the Name_Alias property, and go
2332         # through them one by one.
2333         my ($aliases_code_points, $aliases_maps, undef, undef)
2334                                                 = &prop_invmap('Name_Alias');
2335         for (my $i = 0; $i < @$aliases_code_points; $i++) {
2336             my $code_point = $aliases_code_points->[$i];
2337
2338             # Already removed these above.
2339             next if $code_point <= 0x1F
2340                     || ($code_point >= 0x7F && $code_point <= 0x9F);
2341
2342             my $hex_code_point = sprintf "%05X", $code_point;
2343
2344             # Convert to a list if not already to make the following loop
2345             # control uniform.
2346             $aliases_maps->[$i] = [ $aliases_maps->[$i] ]
2347                                                 if ! ref $aliases_maps->[$i];
2348
2349             # Remove each alias for this code point from the file
2350             foreach my $alias (@{$aliases_maps->[$i]}) {
2351
2352                 # Remove the alias type from the entry, retaining just the name.
2353                 $alias =~ s/:.*//;
2354
2355                 $alias = quotemeta($alias);
2356                 $official =~ s/$hex_code_point \t $alias \n //x;
2357             }
2358         }
2359         local $/ = "\n";
2360         chomp $official;
2361         $/ = $input_record_separator;
2362
2363         # Here have adjusted the file.  We also have to adjust the returned
2364         # inversion map by checking and deleting all the lines in it that
2365         # won't be in the file.  These are the lines that have generated
2366         # things, like <hangul syllable>.
2367         my $tested_map = "";        # Current running string
2368         my @code_point_in_names =
2369                                @Unicode::UCD::code_points_ending_in_code_point;
2370
2371         for my $i (0 .. @$invlist_ref - 1 - $upper_limit_subtract) {
2372             my $start = $invlist_ref->[$i];
2373             my $end = $invlist_ref->[$i+1] - 1;
2374             if ($invmap_ref->[$i] eq $missing) {
2375                 if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
2376                     || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
2377                 {
2378                     fail("prop_invmap('$display_prop')");
2379                     diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
2380                     next PROPERTY;
2381                 }
2382                 next;
2383             }
2384             if ($invmap_ref->[$i] =~ / (.*) ( < .*? > )/x) {
2385                 my $name = $1;
2386                 my $type = $2;
2387                 if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
2388                     || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
2389                 {
2390                     fail("prop_invmap('$display_prop')");
2391                     diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
2392                     next PROPERTY;
2393                 }
2394                 if ($type eq "<hangul syllable>") {
2395                     if ($name ne "") {
2396                         fail("prop_invmap('$display_prop')");
2397                         diag("Unexpected text in $invmap_ref->[$i]");
2398                         next PROPERTY;
2399                     }
2400                     if ($start != 0xAC00) {
2401                         fail("prop_invmap('$display_prop')");
2402                         diag(sprintf("<hangul syllables> should begin at 0xAC00, got %04X", $start));
2403                         next PROPERTY;
2404                     }
2405                     if ($end != $start + 11172 - 1) {
2406                         fail("prop_invmap('$display_prop')");
2407                         diag(sprintf("<hangul syllables> should end at %04X, got %04X", $start + 11172 -1, $end));
2408                         next PROPERTY;
2409                     }
2410                 }
2411                 elsif ($type ne "<code point>") {
2412                     fail("prop_invmap('$display_prop')");
2413                     diag("Unexpected text '$type' in $invmap_ref->[$i]");
2414                     next PROPERTY;
2415                 }
2416                 else {
2417
2418                     # Look through the array of names that end in code points,
2419                     # and look for this start and end.  If not found is an
2420                     # error.  If found, delete it, and at the end, make sure
2421                     # have deleted everything.
2422                     for my $i (0 .. @code_point_in_names - 1) {
2423                         my $hash = $code_point_in_names[$i];
2424                         if ($hash->{'low'} == $start
2425                             && $hash->{'high'} == $end
2426                             && "$hash->{'name'}-" eq $name)
2427                         {
2428                             splice @code_point_in_names, $i, 1;
2429                             last;
2430                         }
2431                         else {
2432                             fail("prop_invmap('$display_prop')");
2433                             diag("Unexpected code-point-in-name line '$invmap_ref->[$i]'");
2434                             next PROPERTY;
2435                         }
2436                     }
2437                 }
2438
2439                 next;
2440             }
2441
2442             # Have adjusted the map, as needed.  Append to running string.
2443             $end = ($start == $end) ? "" : sprintf("%05X", $end);
2444             $tested_map .= sprintf "%05X\t%s\n", $start, $invmap_ref->[$i];
2445         }
2446
2447         # Finished creating the string from the inversion map.  Can compare
2448         # with what the file is.
2449         local $/ = "\n";
2450         chomp $tested_map;
2451         $/ = $input_record_separator;
2452         if ($tested_map ne $official) {
2453             fail_with_diff($display_prop, $official, $tested_map, "prop_invmap");
2454             next PROPERTY;
2455         }
2456         if (@code_point_in_names) {
2457             fail("prop_invmap('$display_prop')");
2458             use Data::Dumper;
2459             diag("Missing code-point-in-name line(s)" . Dumper \@code_point_in_names);
2460             next PROPERTY;
2461         }
2462     }
2463     elsif ($format eq 's') {
2464
2465         # Here the map is not more or less directly from a file stored on
2466         # disk.  We try a different tack.  These should all be properties that
2467         # have just a few possible values (most of them are  binary).  We go
2468         # through the map list, sorting each range into buckets, one for each
2469         # map value.  Thus for binary properties there will be a bucket for Y
2470         # and one for N.  The buckets are inversion lists.  We compare each
2471         # constructed inversion list with what we would get for it using
2472         # prop_invlist(), which has already been tested.  If they all match,
2473         # the whole map must have matched.
2474         my %maps;
2475         my $previous_map;
2476
2477         for my $i (0 .. @$invlist_ref - 1 - $upper_limit_subtract) {
2478             my $range_start = $invlist_ref->[$i];
2479
2480             # Because we are sorting into buckets, things could be
2481             # out-of-order here, and still be in the correct order in the
2482             # bucket, and hence wouldn't show up as an error; so have to
2483             # check.
2484             if (($i > 0 && $range_start <= $invlist_ref->[$i-1])
2485                 || $range_start >= $invlist_ref->[$i+1])
2486             {
2487                 fail("prop_invmap('$display_prop')");
2488                 diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
2489                 next PROPERTY;
2490             }
2491
2492             # This new range closes out the range started in the previous
2493             # iteration.
2494             push @{$maps{$previous_map}}, $range_start if defined $previous_map;
2495
2496             # And starts a range which will be closed in the next iteration.
2497             $previous_map = $invmap_ref->[$i];
2498             push @{$maps{$previous_map}}, $range_start;
2499         }
2500
2501         # The range we just started hasn't been closed, and we didn't look at
2502         # the final element of the loop.  If that range is for the default
2503         # value, it shouldn't be closed, as it is to extend to infinity.  But
2504         # otherwise, it should end at the final Unicode code point, and the
2505         # list that maps to the default value should have another element that
2506         # does go to infinity for every above Unicode code point.
2507
2508         if (@$invlist_ref > 1) {
2509             my $penultimate_map = $invmap_ref->[-2];
2510             if ($penultimate_map ne $missing) {
2511
2512                 # The -1th element contains the first non-Unicode code point.
2513                 push @{$maps{$penultimate_map}}, $invlist_ref->[-1];
2514                 push @{$maps{$missing}}, $invlist_ref->[-1];
2515             }
2516         }
2517
2518         # Here, we have the buckets (inversion lists) all constructed.  Go
2519         # through each and verify that matches what prop_invlist() returns.
2520         # We could use is_deeply() for the comparison, but would get multiple
2521         # messages for each $prop.
2522         foreach my $map (sort keys %maps) {
2523             my @off_invlist = prop_invlist("$prop = $map");
2524             my $min = (@off_invlist >= @{$maps{$map}})
2525                        ? @off_invlist
2526                        : @{$maps{$map}};
2527             for my $i (0 .. $min- 1) {
2528                 if ($i > @off_invlist - 1) {
2529                     fail("prop_invmap('$display_prop')");
2530                     diag("There is no element [$i] for $prop=$map from prop_invlist(), while [$i] in the implicit one constructed from prop_invmap() is '$maps{$map}[$i]'");
2531                     next PROPERTY;
2532                 }
2533                 elsif ($i > @{$maps{$map}} - 1) {
2534                     fail("prop_invmap('$display_prop')");
2535                     diag("There is no element [$i] from the implicit $prop=$map constructed from prop_invmap(), while [$i] in the one from prop_invlist() is '$off_invlist[$i]'");
2536                     next PROPERTY;
2537                 }
2538                 elsif ($maps{$map}[$i] ne $off_invlist[$i]) {
2539                     fail("prop_invmap('$display_prop')");
2540                     diag("Element [$i] of the implicit $prop=$map constructed from prop_invmap() is '$maps{$map}[$i]', and the one from prop_invlist() is '$off_invlist[$i]'");
2541                     next PROPERTY;
2542                 }
2543             }
2544         }
2545     }
2546     else {  # Don't know this property nor format.
2547
2548         fail("prop_invmap('$display_prop')");
2549         diag("Unknown property '$display_prop' or format '$format'");
2550         next PROPERTY;
2551     }
2552
2553     pass("prop_invmap('$display_prop')");
2554 }
2555
2556 # A few tests of search_invlist
2557 use Unicode::UCD qw(search_invlist);
2558
2559 my ($scripts_ranges_ref, $scripts_map_ref) = prop_invmap("Script");
2560 my $index = search_invlist($scripts_ranges_ref, 0x390);
2561 is($scripts_map_ref->[$index], "Greek", "U+0390 is Greek");
2562 my @alpha_invlist = prop_invlist("Alpha");
2563 is(search_invlist(\@alpha_invlist, ord("\t")), undef, "search_invlist returns undef for code points before first one on the list");
2564
2565 ok($/ eq $input_record_separator,  "The record separator didn't get overridden");
2566
2567 if (! ok(@warnings == 0, "No warnings were generated")) {
2568     diag(join "\n", "The warnings are:", @warnings);
2569 }
2570
2571 done_testing();