• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /********************************************************************
2  * COPYRIGHT:
3  * Copyright (c) 1997-2013, International Business Machines Corporation and
4  * others. All Rights Reserved.
5  ********************************************************************/
6 /*****************************************************************************
7 *
8 * File CLOCTST.C
9 *
10 * Modification History:
11 *        Name                     Description
12 *     Madhu Katragadda            Ported for C API
13 ******************************************************************************
14 */
15 #include "cloctst.h"
16 #include <stdlib.h>
17 #include <stdio.h>
18 #include <string.h>
19 #include "cintltst.h"
20 #include "cstring.h"
21 #include "uparse.h"
22 #include "uresimp.h"
23 
24 #include "unicode/putil.h"
25 #include "unicode/ubrk.h"
26 #include "unicode/uchar.h"
27 #include "unicode/ucol.h"
28 #include "unicode/udat.h"
29 #include "unicode/uloc.h"
30 #include "unicode/umsg.h"
31 #include "unicode/ures.h"
32 #include "unicode/uset.h"
33 #include "unicode/ustring.h"
34 #include "unicode/utypes.h"
35 #include "unicode/ulocdata.h"
36 #include "unicode/uldnames.h"
37 #include "unicode/parseerr.h" /* may not be included with some uconfig switches */
38 #include "udbgutil.h"
39 #define LENGTHOF(array) (int32_t)(sizeof(array)/sizeof((array)[0]))
40 
41 static void TestNullDefault(void);
42 static void TestNonexistentLanguageExemplars(void);
43 static void TestLocDataErrorCodeChaining(void);
44 static void TestLanguageExemplarsFallbacks(void);
45 static void TestDisplayNameBrackets(void);
46 
47 static void TestUnicodeDefines(void);
48 
49 void PrintDataTable();
50 
51 /*---------------------------------------------------
52   table of valid data
53  --------------------------------------------------- */
54 #define LOCALE_SIZE 9
55 #define LOCALE_INFO_SIZE 28
56 
57 static const char* const rawData2[LOCALE_INFO_SIZE][LOCALE_SIZE] = {
58     /* language code */
59     {   "en",   "fr",   "ca",   "el",   "no",   "zh",   "de",   "es",  "ja"    },
60     /* script code */
61     {   "",     "",     "",     "",     "",     "", "", "", ""  },
62     /* country code */
63     {   "US",   "FR",   "ES",   "GR",   "NO",   "CN", "DE", "", "JP"    },
64     /* variant code */
65     {   "",     "",     "",     "",     "NY",   "", "", "", ""      },
66     /* full name */
67     {   "en_US",    "fr_FR",    "ca_ES",
68         "el_GR",    "no_NO_NY", "zh_Hans_CN",
69         "de_DE@collation=phonebook", "es@collation=traditional",  "ja_JP@calendar=japanese" },
70     /* ISO-3 language */
71     {   "eng",  "fra",  "cat",  "ell",  "nor",  "zho", "deu", "spa", "jpn"   },
72     /* ISO-3 country */
73     {   "USA",  "FRA",  "ESP",  "GRC",  "NOR",  "CHN", "DEU", "", "JPN"   },
74     /* LCID */
75     {   "409", "40c", "403", "408", "814",  "804", "10407", "40a", "411"     },
76 
77     /* display language (English) */
78     {   "English",  "French",   "Catalan", "Greek",    "Norwegian", "Chinese", "German", "Spanish", "Japanese"    },
79     /* display script code (English) */
80     {   "",     "",     "",     "",     "",     "Simplified Han", "", "", ""       },
81     /* display country (English) */
82     {   "United States",    "France",   "Spain",  "Greece",   "Norway", "China", "Germany", "", "Japan"       },
83     /* display variant (English) */
84     {   "",     "",     "",     "",     "NY",  "", "", "", ""       },
85     /* display name (English) */
86     {   "English (United States)", "French (France)", "Catalan (Spain)",
87         "Greek (Greece)", "Norwegian (Norway, NY)", "Chinese (Simplified, China)",
88         "German (Germany, Sort Order=Phonebook Sort Order)", "Spanish (Sort Order=Traditional Sort Order)", "Japanese (Japan, Calendar=Japanese Calendar)" },
89 
90     /* display language (French) */
91     {   "anglais",  "fran\\u00E7ais",   "catalan", "grec",    "norv\\u00E9gien",    "chinois", "allemand", "espagnol", "japonais"     },
92     /* display script code (French) */
93     {   "",     "",     "",     "",     "",     "sinogrammes simplifi\\u00e9s", "", "", ""         },
94     /* display country (French) */
95     {   "\\u00C9tats-Unis",    "France",   "Espagne",  "Gr\\u00E8ce",   "Norv\\u00E8ge",    "Chine", "Allemagne", "", "Japon"       },
96     /* display variant (French) */
97     {   "",     "",     "",     "",     "NY",   "", "", "", ""       },
98     /* display name (French) */
99     {   "anglais (\\u00C9tats-Unis)", "fran\\u00E7ais (France)", "catalan (Espagne)",
100         "grec (Gr\\u00E8ce)", "norv\\u00E9gien (Norv\\u00E8ge, NY)",  "chinois (simplifi\\u00e9, Chine)",
101         "allemand (Allemagne, ordre de tri=Ordre de l\\u2019annuaire)", "espagnol (ordre de tri=Ordre traditionnel)", "japonais (Japon, calendrier=Calendrier japonais)" },
102 
103     /* display language (Catalan) */
104     {   "angl\\u00E8s", "franc\\u00E8s", "catal\\u00E0", "grec",  "noruec", "xin\\u00E8s", "alemany", "espanyol", "japon\\u00E8s"    },
105     /* display script code (Catalan) */
106     {   "",     "",     "",     "",     "",     "han simplificat", "", "", ""         },
107     /* display country (Catalan) */
108     {   "Estats Units", "Fran\\u00E7a", "Espanya",  "Gr\\u00E8cia", "Noruega",  "Xina", "Alemanya", "", "Jap\\u00F3"    },
109     /* display variant (Catalan) */
110     {   "", "", "",                    "", "NY",    "", "", "", ""    },
111     /* display name (Catalan) */
112     {   "angl\\u00E8s (Estats Units)", "franc\\u00E8s (Fran\\u00E7a)", "catal\\u00E0 (Espanya)",
113     "grec (Gr\\u00E8cia)", "noruec (Noruega, NY)", "xin\\u00E8s (simplificat, Xina)",
114     "alemany (Alemanya, ordenaci\\u00F3=ordre de la guia telef\\u00F2nica)", "espanyol (ordenaci\\u00F3=ordre tradicional)", "japon\\u00E8s (Jap\\u00F3, calendari=calendari japon\\u00e8s)" },
115 
116     /* display language (Greek) */
117     {
118         "\\u0391\\u03b3\\u03b3\\u03bb\\u03b9\\u03ba\\u03ac",
119         "\\u0393\\u03b1\\u03bb\\u03bb\\u03b9\\u03ba\\u03ac",
120         "\\u039a\\u03b1\\u03c4\\u03b1\\u03bb\\u03b1\\u03bd\\u03b9\\u03ba\\u03ac",
121         "\\u0395\\u03bb\\u03bb\\u03b7\\u03bd\\u03b9\\u03ba\\u03ac",
122         "\\u039d\\u03bf\\u03c1\\u03b2\\u03b7\\u03b3\\u03b9\\u03ba\\u03ac",
123         "\\u039A\\u03B9\\u03BD\\u03B5\\u03B6\\u03B9\\u03BA\\u03AC",
124         "\\u0393\\u03B5\\u03C1\\u03BC\\u03B1\\u03BD\\u03B9\\u03BA\\u03AC",
125         "\\u0399\\u03C3\\u03C0\\u03B1\\u03BD\\u03B9\\u03BA\\u03AC",
126         "\\u0399\\u03B1\\u03C0\\u03C9\\u03BD\\u03B9\\u03BA\\u03AC"
127     },
128     /* display script code (Greek) */
129 
130     {   "",     "",     "",     "",     "", "\\u0391\\u03c0\\u03bb\\u03bf\\u03c0\\u03bf\\u03b9\\u03b7\\u03bc\\u03ad\\u03bd\\u03bf \\u03a7\\u03b1\\u03bd", "", "", "" },
131     /* display country (Greek) */
132     {
133         "\\u0397\\u03BD\\u03C9\\u03BC\\u03AD\\u03BD\\u03B5\\u03C2 \\u03A0\\u03BF\\u03BB\\u03B9\\u03C4\\u03B5\\u03AF\\u03B5\\u03C2",
134         "\\u0393\\u03b1\\u03bb\\u03bb\\u03af\\u03b1",
135         "\\u0399\\u03c3\\u03c0\\u03b1\\u03bd\\u03af\\u03b1",
136         "\\u0395\\u03bb\\u03bb\\u03ac\\u03b4\\u03b1",
137         "\\u039d\\u03bf\\u03c1\\u03b2\\u03b7\\u03b3\\u03af\\u03b1",
138         "\\u039A\\u03AF\\u03BD\\u03B1",
139         "\\u0393\\u03B5\\u03C1\\u03BC\\u03B1\\u03BD\\u03AF\\u03B1",
140         "",
141         "\\u0399\\u03B1\\u03C0\\u03C9\\u03BD\\u03AF\\u03B1"
142     },
143     /* display variant (Greek) */
144     {   "", "", "", "", "NY", "", "", "", ""    }, /* TODO: currently there is no translation for NY in Greek fix this test when we have it */
145     /* display name (Greek) */
146     {
147         "\\u0391\\u03b3\\u03b3\\u03bb\\u03b9\\u03ba\\u03ac (\\u0397\\u03BD\\u03C9\\u03BC\\u03AD\\u03BD\\u03B5\\u03C2 \\u03A0\\u03BF\\u03BB\\u03B9\\u03C4\\u03B5\\u03AF\\u03B5\\u03C2)",
148         "\\u0393\\u03b1\\u03bb\\u03bb\\u03b9\\u03ba\\u03ac (\\u0393\\u03b1\\u03bb\\u03bb\\u03af\\u03b1)",
149         "\\u039a\\u03b1\\u03c4\\u03b1\\u03bb\\u03b1\\u03bd\\u03b9\\u03ba\\u03ac (\\u0399\\u03c3\\u03c0\\u03b1\\u03bd\\u03af\\u03b1)",
150         "\\u0395\\u03bb\\u03bb\\u03b7\\u03bd\\u03b9\\u03ba\\u03ac (\\u0395\\u03bb\\u03bb\\u03ac\\u03b4\\u03b1)",
151         "\\u039d\\u03bf\\u03c1\\u03b2\\u03b7\\u03b3\\u03b9\\u03ba\\u03ac (\\u039d\\u03bf\\u03c1\\u03b2\\u03b7\\u03b3\\u03af\\u03b1, NY)",
152         "\\u039A\\u03B9\\u03BD\\u03B5\\u03B6\\u03B9\\u03BA\\u03AC (\\u0391\\u03c0\\u03bb\\u03bf\\u03c0\\u03bf\\u03b9\\u03b7\\u03bc\\u03ad\\u03bd\\u03bf, \\u039A\\u03AF\\u03BD\\u03B1)",
153         "\\u0393\\u03b5\\u03c1\\u03bc\\u03b1\\u03bd\\u03b9\\u03ba\\u03ac (\\u0393\\u03b5\\u03c1\\u03bc\\u03b1\\u03bd\\u03af\\u03b1, \\u03a3\\u03b5\\u03b9\\u03c1\\u03ac \\u03c4\\u03b1\\u03be\\u03b9\\u03bd\\u03cc\\u03bc\\u03b7\\u03c3\\u03b7\\u03c2=\\u03a3\\u03b5\\u03b9\\u03c1\\u03ac \\u03c4\\u03b1\\u03be\\u03b9\\u03bd\\u03cc\\u03bc\\u03b7\\u03c3\\u03b7\\u03c2 \\u03c4\\u03b7\\u03bb\\u03b5\\u03c6\\u03c9\\u03bd\\u03b9\\u03ba\\u03bf\\u03cd \\u03ba\\u03b1\\u03c4\\u03b1\\u03bb\\u03cc\\u03b3\\u03bf\\u03c5)",
154         "\\u0399\\u03c3\\u03c0\\u03b1\\u03bd\\u03b9\\u03ba\\u03ac (\\u03a3\\u03b5\\u03b9\\u03c1\\u03ac \\u03c4\\u03b1\\u03be\\u03b9\\u03bd\\u03cc\\u03bc\\u03b7\\u03c3\\u03b7\\u03c2=\\u03a0\\u03b1\\u03c1\\u03b1\\u03b4\\u03bf\\u03c3\\u03b9\\u03b1\\u03ba\\u03ae \\u03c3\\u03b5\\u03b9\\u03c1\\u03ac \\u03c4\\u03b1\\u03be\\u03b9\\u03bd\\u03cc\\u03bc\\u03b7\\u03c3\\u03b7\\u03c2)",
155         "\\u0399\\u03b1\\u03c0\\u03c9\\u03bd\\u03b9\\u03ba\\u03ac (\\u0399\\u03b1\\u03c0\\u03c9\\u03bd\\u03af\\u03b1, \\u0397\\u03bc\\u03b5\\u03c1\\u03bf\\u03bb\\u03cc\\u03b3\\u03b9\\u03bf=\\u0399\\u03b1\\u03c0\\u03c9\\u03bd\\u03b9\\u03ba\\u03cc \\u03b7\\u03bc\\u03b5\\u03c1\\u03bf\\u03bb\\u03cc\\u03b3\\u03b9\\u03bf)"
156     }
157 };
158 
159 static UChar*** dataTable=0;
160 enum {
161     ENGLISH = 0,
162     FRENCH = 1,
163     CATALAN = 2,
164     GREEK = 3,
165     NORWEGIAN = 4
166 };
167 
168 enum {
169     LANG = 0,
170     SCRIPT = 1,
171     CTRY = 2,
172     VAR = 3,
173     NAME = 4,
174     LANG3 = 5,
175     CTRY3 = 6,
176     LCID = 7,
177     DLANG_EN = 8,
178     DSCRIPT_EN = 9,
179     DCTRY_EN = 10,
180     DVAR_EN = 11,
181     DNAME_EN = 12,
182     DLANG_FR = 13,
183     DSCRIPT_FR = 14,
184     DCTRY_FR = 15,
185     DVAR_FR = 16,
186     DNAME_FR = 17,
187     DLANG_CA = 18,
188     DSCRIPT_CA = 19,
189     DCTRY_CA = 20,
190     DVAR_CA = 21,
191     DNAME_CA = 22,
192     DLANG_EL = 23,
193     DSCRIPT_EL = 24,
194     DCTRY_EL = 25,
195     DVAR_EL = 26,
196     DNAME_EL = 27
197 };
198 
199 #define TESTCASE(name) addTest(root, &name, "tsutil/cloctst/" #name)
200 
201 void addLocaleTest(TestNode** root);
202 
addLocaleTest(TestNode ** root)203 void addLocaleTest(TestNode** root)
204 {
205     TESTCASE(TestObsoleteNames); /* srl- move */
206     TESTCASE(TestBasicGetters);
207     TESTCASE(TestNullDefault);
208     TESTCASE(TestPrefixes);
209     TESTCASE(TestSimpleResourceInfo);
210     TESTCASE(TestDisplayNames);
211     TESTCASE(TestGetAvailableLocales);
212     TESTCASE(TestDataDirectory);
213 #if !UCONFIG_NO_FILE_IO && !UCONFIG_NO_LEGACY_CONVERSION
214     TESTCASE(TestISOFunctions);
215 #endif
216     TESTCASE(TestISO3Fallback);
217     TESTCASE(TestUninstalledISO3Names);
218     TESTCASE(TestSimpleDisplayNames);
219     TESTCASE(TestVariantParsing);
220     TESTCASE(TestKeywordVariants);
221     TESTCASE(TestKeywordVariantParsing);
222     TESTCASE(TestCanonicalization);
223     TESTCASE(TestKeywordSet);
224     TESTCASE(TestKeywordSetError);
225     TESTCASE(TestDisplayKeywords);
226     TESTCASE(TestDisplayKeywordValues);
227     TESTCASE(TestGetBaseName);
228 #if !UCONFIG_NO_FILE_IO
229     TESTCASE(TestGetLocale);
230 #endif
231     TESTCASE(TestDisplayNameWarning);
232     TESTCASE(TestNonexistentLanguageExemplars);
233     TESTCASE(TestLocDataErrorCodeChaining);
234     TESTCASE(TestLanguageExemplarsFallbacks);
235     TESTCASE(TestCalendar);
236     TESTCASE(TestDateFormat);
237     TESTCASE(TestCollation);
238     TESTCASE(TestULocale);
239     TESTCASE(TestUResourceBundle);
240     TESTCASE(TestDisplayName);
241     TESTCASE(TestAcceptLanguage);
242     TESTCASE(TestGetLocaleForLCID);
243     TESTCASE(TestOrientation);
244     TESTCASE(TestLikelySubtags);
245     TESTCASE(TestToLanguageTag);
246     TESTCASE(TestForLanguageTag);
247     TESTCASE(TestTrailingNull);
248     TESTCASE(TestUnicodeDefines);
249     TESTCASE(TestEnglishExemplarCharacters);
250     TESTCASE(TestDisplayNameBrackets);
251 }
252 
253 
254 /* testing uloc(), uloc_getName(), uloc_getLanguage(), uloc_getVariant(), uloc_getCountry() */
TestBasicGetters()255 static void TestBasicGetters() {
256     int32_t i;
257     int32_t cap;
258     UErrorCode status = U_ZERO_ERROR;
259     char *testLocale = 0;
260     char *temp = 0, *name = 0;
261     log_verbose("Testing Basic Getters\n");
262     for (i = 0; i < LOCALE_SIZE; i++) {
263         testLocale=(char*)malloc(sizeof(char) * (strlen(rawData2[NAME][i])+1));
264         strcpy(testLocale,rawData2[NAME][i]);
265 
266         log_verbose("Testing   %s  .....\n", testLocale);
267         cap=uloc_getLanguage(testLocale, NULL, 0, &status);
268         if(status==U_BUFFER_OVERFLOW_ERROR){
269             status=U_ZERO_ERROR;
270             temp=(char*)malloc(sizeof(char) * (cap+1));
271             uloc_getLanguage(testLocale, temp, cap+1, &status);
272         }
273         if(U_FAILURE(status)){
274             log_err("ERROR: in uloc_getLanguage  %s\n", myErrorName(status));
275         }
276         if (0 !=strcmp(temp,rawData2[LANG][i]))    {
277             log_err("  Language code mismatch: %s versus  %s\n", temp, rawData2[LANG][i]);
278         }
279 
280 
281         cap=uloc_getCountry(testLocale, temp, cap, &status);
282         if(status==U_BUFFER_OVERFLOW_ERROR){
283             status=U_ZERO_ERROR;
284             temp=(char*)realloc(temp, sizeof(char) * (cap+1));
285             uloc_getCountry(testLocale, temp, cap+1, &status);
286         }
287         if(U_FAILURE(status)){
288             log_err("ERROR: in uloc_getCountry  %s\n", myErrorName(status));
289         }
290         if (0 != strcmp(temp, rawData2[CTRY][i])) {
291             log_err(" Country code mismatch:  %s  versus   %s\n", temp, rawData2[CTRY][i]);
292 
293           }
294 
295         cap=uloc_getVariant(testLocale, temp, cap, &status);
296         if(status==U_BUFFER_OVERFLOW_ERROR){
297             status=U_ZERO_ERROR;
298             temp=(char*)realloc(temp, sizeof(char) * (cap+1));
299             uloc_getVariant(testLocale, temp, cap+1, &status);
300         }
301         if(U_FAILURE(status)){
302             log_err("ERROR: in uloc_getVariant  %s\n", myErrorName(status));
303         }
304         if (0 != strcmp(temp, rawData2[VAR][i])) {
305             log_err("Variant code mismatch:  %s  versus   %s\n", temp, rawData2[VAR][i]);
306         }
307 
308         cap=uloc_getName(testLocale, NULL, 0, &status);
309         if(status==U_BUFFER_OVERFLOW_ERROR){
310             status=U_ZERO_ERROR;
311             name=(char*)malloc(sizeof(char) * (cap+1));
312             uloc_getName(testLocale, name, cap+1, &status);
313         } else if(status==U_ZERO_ERROR) {
314           log_err("ERROR: in uloc_getName(%s,NULL,0,..), expected U_BUFFER_OVERFLOW_ERROR!\n", testLocale);
315         }
316         if(U_FAILURE(status)){
317             log_err("ERROR: in uloc_getName   %s\n", myErrorName(status));
318         }
319         if (0 != strcmp(name, rawData2[NAME][i])){
320             log_err(" Mismatch in getName:  %s  versus   %s\n", name, rawData2[NAME][i]);
321         }
322 
323         free(temp);
324         free(name);
325 
326         free(testLocale);
327     }
328 }
329 
TestNullDefault()330 static void TestNullDefault() {
331     UErrorCode status = U_ZERO_ERROR;
332     char original[ULOC_FULLNAME_CAPACITY];
333 
334     uprv_strcpy(original, uloc_getDefault());
335     uloc_setDefault("qq_BLA", &status);
336     if (uprv_strcmp(uloc_getDefault(), "qq_BLA") != 0) {
337         log_err(" Mismatch in uloc_setDefault:  qq_BLA  versus   %s\n", uloc_getDefault());
338     }
339     uloc_setDefault(NULL, &status);
340     if (uprv_strcmp(uloc_getDefault(), original) != 0) {
341         log_err(" uloc_setDefault(NULL, &status) didn't get the default locale back!\n");
342     }
343 
344     {
345     /* Test that set & get of default locale work, and that
346      * default locales are cached and reused, and not overwritten.
347      */
348         const char *n_en_US;
349         const char *n_fr_FR;
350         const char *n2_en_US;
351 
352         status = U_ZERO_ERROR;
353         uloc_setDefault("en_US", &status);
354         n_en_US = uloc_getDefault();
355         if (strcmp(n_en_US, "en_US") != 0) {
356             log_err("Wrong result from uloc_getDefault().  Expected \"en_US\", got \"%s\"\n", n_en_US);
357         }
358 
359         uloc_setDefault("fr_FR", &status);
360         n_fr_FR = uloc_getDefault();
361         if (strcmp(n_en_US, "en_US") != 0) {
362             log_err("uloc_setDefault altered previously default string."
363                 "Expected \"en_US\", got \"%s\"\n",  n_en_US);
364         }
365         if (strcmp(n_fr_FR, "fr_FR") != 0) {
366             log_err("Wrong result from uloc_getDefault().  Expected \"fr_FR\", got %s\n",  n_fr_FR);
367         }
368 
369         uloc_setDefault("en_US", &status);
370         n2_en_US = uloc_getDefault();
371         if (strcmp(n2_en_US, "en_US") != 0) {
372             log_err("Wrong result from uloc_getDefault().  Expected \"en_US\", got \"%s\"\n", n_en_US);
373         }
374         if (n2_en_US != n_en_US) {
375             log_err("Default locale cache failed to reuse en_US locale.\n");
376         }
377 
378         if (U_FAILURE(status)) {
379             log_err("Failure returned from uloc_setDefault - \"%s\"\n", u_errorName(status));
380         }
381 
382     }
383 
384 }
385 /* Test the i- and x- and @ and . functionality
386 */
387 
388 #define PREFIXBUFSIZ 128
389 
TestPrefixes()390 static void TestPrefixes() {
391     int row = 0;
392     int n;
393     const char *loc, *expected;
394 
395     static const char * const testData[][7] =
396     {
397         /* NULL canonicalize() column means "expect same as getName()" */
398         {"sv", "", "FI", "AL", "sv-fi-al", "sv_FI_AL", NULL},
399         {"en", "", "GB", "", "en-gb", "en_GB", NULL},
400         {"i-hakka", "", "MT", "XEMXIJA", "i-hakka_MT_XEMXIJA", "i-hakka_MT_XEMXIJA", NULL},
401         {"i-hakka", "", "CN", "", "i-hakka_CN", "i-hakka_CN", NULL},
402         {"i-hakka", "", "MX", "", "I-hakka_MX", "i-hakka_MX", NULL},
403         {"x-klingon", "", "US", "SANJOSE", "X-KLINGON_us_SANJOSE", "x-klingon_US_SANJOSE", NULL},
404 
405         {"zh", "Hans", "", "PINYIN", "zh-Hans-pinyin", "zh_Hans__PINYIN", "zh_Hans@collation=pinyin"},
406         {"hy", "", "", "AREVMDA", "hy_AREVMDA", "hy__AREVMDA", NULL},
407 
408         {"de", "", "", "1901", "de-1901", "de__1901", NULL},
409         {"mr", "", "", "", "mr.utf8", "mr.utf8", "mr"},
410         {"de", "", "TV", "", "de-tv.koi8r", "de_TV.koi8r", "de_TV"},
411         {"x-piglatin", "", "ML", "", "x-piglatin_ML.MBE", "x-piglatin_ML.MBE", "x-piglatin_ML"},  /* Multibyte English */
412         {"i-cherokee", "","US", "", "i-Cherokee_US.utf7", "i-cherokee_US.utf7", "i-cherokee_US"},
413         {"x-filfli", "", "MT", "FILFLA", "x-filfli_MT_FILFLA.gb-18030", "x-filfli_MT_FILFLA.gb-18030", "x-filfli_MT_FILFLA"},
414         {"no", "", "NO", "NY", "no-no-ny.utf32@B", "no_NO_NY.utf32@B", "no_NO_NY_B"},
415         {"no", "", "NO", "",  "no-no.utf32@B", "no_NO.utf32@B", "no_NO_B"},
416         {"no", "", "",   "NY", "no__ny", "no__NY", NULL},
417         {"no", "", "",   "", "no@ny", "no@ny", "no__NY"},
418         {"el", "Latn", "", "", "el-latn", "el_Latn", NULL},
419         {"en", "Cyrl", "RU", "", "en-cyrl-ru", "en_Cyrl_RU", NULL},
420         {"zh", "Hant", "TW", "STROKE", "zh-hant_TW_STROKE", "zh_Hant_TW_STROKE", "zh_Hant_TW@collation=stroke"},
421         {"qq", "Qqqq", "QQ", "QQ", "qq_Qqqq_QQ_QQ", "qq_Qqqq_QQ_QQ", NULL},
422         {"qq", "Qqqq", "", "QQ", "qq_Qqqq__QQ", "qq_Qqqq__QQ", NULL},
423         {"ab", "Cdef", "GH", "IJ", "ab_cdef_gh_ij", "ab_Cdef_GH_IJ", NULL}, /* total garbage */
424 
425         {NULL,NULL,NULL,NULL,NULL,NULL,NULL}
426     };
427 
428     static const char * const testTitles[] = {
429         "uloc_getLanguage()",
430         "uloc_getScript()",
431         "uloc_getCountry()",
432         "uloc_getVariant()",
433         "name",
434         "uloc_getName()",
435         "uloc_canonicalize()"
436     };
437 
438     char buf[PREFIXBUFSIZ];
439     int32_t len;
440     UErrorCode err;
441 
442 
443     for(row=0;testData[row][0] != NULL;row++) {
444         loc = testData[row][NAME];
445         log_verbose("Test #%d: %s\n", row, loc);
446 
447         err = U_ZERO_ERROR;
448         len=0;
449         buf[0]=0;
450         for(n=0;n<=(NAME+2);n++) {
451             if(n==NAME) continue;
452 
453             for(len=0;len<PREFIXBUFSIZ;len++) {
454                 buf[len] = '%'; /* Set a tripwire.. */
455             }
456             len = 0;
457 
458             switch(n) {
459             case LANG:
460                 len = uloc_getLanguage(loc, buf, PREFIXBUFSIZ, &err);
461                 break;
462 
463             case SCRIPT:
464                 len = uloc_getScript(loc, buf, PREFIXBUFSIZ, &err);
465                 break;
466 
467             case CTRY:
468                 len = uloc_getCountry(loc, buf, PREFIXBUFSIZ, &err);
469                 break;
470 
471             case VAR:
472                 len = uloc_getVariant(loc, buf, PREFIXBUFSIZ, &err);
473                 break;
474 
475             case NAME+1:
476                 len = uloc_getName(loc, buf, PREFIXBUFSIZ, &err);
477                 break;
478 
479             case NAME+2:
480                 len = uloc_canonicalize(loc, buf, PREFIXBUFSIZ, &err);
481                 break;
482 
483             default:
484                 strcpy(buf, "**??");
485                 len=4;
486             }
487 
488             if(U_FAILURE(err)) {
489                 log_err("#%d: %s on %s: err %s\n",
490                     row, testTitles[n], loc, u_errorName(err));
491             } else {
492                 log_verbose("#%d: %s on %s: -> [%s] (length %d)\n",
493                     row, testTitles[n], loc, buf, len);
494 
495                 if(len != (int32_t)strlen(buf)) {
496                     log_err("#%d: %s on %s: -> [%s] (length returned %d, actual %d!)\n",
497                         row, testTitles[n], loc, buf, len, strlen(buf)+1);
498 
499                 }
500 
501                 /* see if they smashed something */
502                 if(buf[len+1] != '%') {
503                     log_err("#%d: %s on %s: -> [%s] - wrote [%X] out ofbounds!\n",
504                         row, testTitles[n], loc, buf, buf[len+1]);
505                 }
506 
507                 expected = testData[row][n];
508                 if (expected == NULL && n == (NAME+2)) {
509                     /* NULL expected canonicalize() means "expect same as getName()" */
510                     expected = testData[row][NAME+1];
511                 }
512                 if(strcmp(buf, expected)) {
513                     log_err("#%d: %s on %s: -> [%s] (expected '%s'!)\n",
514                         row, testTitles[n], loc, buf, expected);
515 
516                 }
517             }
518         }
519     }
520 }
521 
522 
523 /* testing uloc_getISO3Language(), uloc_getISO3Country(),  */
TestSimpleResourceInfo()524 static void TestSimpleResourceInfo() {
525     int32_t i;
526     char* testLocale = 0;
527     UChar* expected = 0;
528 
529     const char* temp;
530     char            temp2[20];
531     testLocale=(char*)malloc(sizeof(char) * 1);
532     expected=(UChar*)malloc(sizeof(UChar) * 1);
533 
534     setUpDataTable();
535     log_verbose("Testing getISO3Language and getISO3Country\n");
536     for (i = 0; i < LOCALE_SIZE; i++) {
537 
538         testLocale=(char*)realloc(testLocale, sizeof(char) * (u_strlen(dataTable[NAME][i])+1));
539         u_austrcpy(testLocale, dataTable[NAME][i]);
540 
541         log_verbose("Testing   %s ......\n", testLocale);
542 
543         temp=uloc_getISO3Language(testLocale);
544         expected=(UChar*)realloc(expected, sizeof(UChar) * (strlen(temp) + 1));
545         u_uastrcpy(expected,temp);
546         if (0 != u_strcmp(expected, dataTable[LANG3][i])) {
547             log_err("  ISO-3 language code mismatch:  %s versus  %s\n",  austrdup(expected),
548                 austrdup(dataTable[LANG3][i]));
549         }
550 
551         temp=uloc_getISO3Country(testLocale);
552         expected=(UChar*)realloc(expected, sizeof(UChar) * (strlen(temp) + 1));
553         u_uastrcpy(expected,temp);
554         if (0 != u_strcmp(expected, dataTable[CTRY3][i])) {
555             log_err("  ISO-3 Country code mismatch:  %s versus  %s\n",  austrdup(expected),
556                 austrdup(dataTable[CTRY3][i]));
557         }
558         sprintf(temp2, "%x", (int)uloc_getLCID(testLocale));
559         if (strcmp(temp2, rawData2[LCID][i]) != 0) {
560             log_err("LCID mismatch: %s versus %s\n", temp2 , rawData2[LCID][i]);
561         }
562     }
563 
564     free(expected);
565     free(testLocale);
566     cleanUpDataTable();
567 }
568 
569 /* if len < 0, we convert until we hit UChar 0x0000, which is not output. will add trailing null
570  * if there's room but won't be included in result.  result < 0 indicates an error.
571  * Returns the number of chars written (not those that would be written if there's enough room.*/
UCharsToEscapedAscii(const UChar * utext,int32_t len,char * resultChars,int32_t buflen)572 static int32_t UCharsToEscapedAscii(const UChar* utext, int32_t len, char* resultChars, int32_t buflen) {
573     static const struct {
574         char escapedChar;
575         UChar sourceVal;
576     } ESCAPE_MAP[] = {
577         /*a*/ {'a', 0x07},
578         /*b*/ {'b', 0x08},
579         /*e*/ {'e', 0x1b},
580         /*f*/ {'f', 0x0c},
581         /*n*/ {'n', 0x0a},
582         /*r*/ {'r', 0x0d},
583         /*t*/ {'t', 0x09},
584         /*v*/ {'v', 0x0b}
585     };
586     static const int32_t ESCAPE_MAP_LENGTH = sizeof(ESCAPE_MAP)/sizeof(ESCAPE_MAP[0]);
587     static const char HEX_DIGITS[] = {
588         '0', '1', '2', '3', '4', '5', '6', '7',
589         '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'
590     };
591     int32_t i, j;
592     int32_t resultLen = 0;
593     const int32_t limit = len<0 ? buflen : len; /* buflen is long enough to hit the buffer limit */
594     const int32_t escapeLimit1 = buflen-2;
595     const int32_t escapeLimit2 = buflen-6;
596     UChar uc;
597 
598     if(utext==NULL || resultChars==NULL || buflen<0) {
599         return -1;
600     }
601 
602     for(i=0;i<limit && resultLen<buflen;++i) {
603         uc=utext[i];
604         if(len<0 && uc==0) {
605             break;
606         }
607         if(uc<0x20) {
608             for(j=0;j<ESCAPE_MAP_LENGTH && uc!=ESCAPE_MAP[j].sourceVal;j++) {
609             }
610             if(j<ESCAPE_MAP_LENGTH) {
611                 if(resultLen>escapeLimit1) {
612                     break;
613                 }
614                 resultChars[resultLen++]='\\';
615                 resultChars[resultLen++]=ESCAPE_MAP[j].escapedChar;
616                 continue;
617             }
618         } else if(uc<0x7f) {
619             u_austrncpy(resultChars + resultLen, &uc, 1);
620             resultLen++;
621             continue;
622         }
623 
624         if(resultLen>escapeLimit2) {
625             break;
626         }
627 
628         /* have to escape the uchar */
629         resultChars[resultLen++]='\\';
630         resultChars[resultLen++]='u';
631         resultChars[resultLen++]=HEX_DIGITS[(uc>>12)&0xff];
632         resultChars[resultLen++]=HEX_DIGITS[(uc>>8)&0xff];
633         resultChars[resultLen++]=HEX_DIGITS[(uc>>4)&0xff];
634         resultChars[resultLen++]=HEX_DIGITS[uc&0xff];
635     }
636 
637     if(resultLen<buflen) {
638         resultChars[resultLen] = 0;
639     }
640 
641     return resultLen;
642 }
643 
644 /*
645  * Jitterbug 2439 -- markus 20030425
646  *
647  * The lookup of display names must not fall back through the default
648  * locale because that yields useless results.
649  */
TestDisplayNames()650 static void TestDisplayNames()
651 {
652     UChar buffer[100];
653     UErrorCode errorCode=U_ZERO_ERROR;
654     int32_t length;
655     log_verbose("Testing getDisplayName for different locales\n");
656 
657     log_verbose("  In locale = en_US...\n");
658     doTestDisplayNames("en_US", DLANG_EN);
659     log_verbose("  In locale = fr_FR....\n");
660     doTestDisplayNames("fr_FR", DLANG_FR);
661     log_verbose("  In locale = ca_ES...\n");
662     doTestDisplayNames("ca_ES", DLANG_CA);
663     log_verbose("  In locale = gr_EL..\n");
664     doTestDisplayNames("el_GR", DLANG_EL);
665 
666     /* test that the default locale has a display name for its own language */
667     errorCode=U_ZERO_ERROR;
668     length=uloc_getDisplayLanguage(NULL, NULL, buffer, LENGTHOF(buffer), &errorCode);
669     if(U_FAILURE(errorCode) || (length<=3 && buffer[0]<=0x7f)) {
670         /* check <=3 to reject getting the language code as a display name */
671         log_data_err("unable to get a display string for the language of the default locale - %s (Are you missing data?)\n", u_errorName(errorCode));
672     }
673 
674     /* test that we get the language code itself for an unknown language, and a default warning */
675     errorCode=U_ZERO_ERROR;
676     length=uloc_getDisplayLanguage("qq", "rr", buffer, LENGTHOF(buffer), &errorCode);
677     if(errorCode!=U_USING_DEFAULT_WARNING || length!=2 || buffer[0]!=0x71 || buffer[1]!=0x71) {
678         log_err("error getting the display string for an unknown language - %s\n", u_errorName(errorCode));
679     }
680 
681     /* test that we get a default warning for a display name where one component is unknown (4255) */
682     errorCode=U_ZERO_ERROR;
683     length=uloc_getDisplayName("qq_US_POSIX", "en_US", buffer, LENGTHOF(buffer), &errorCode);
684     if(errorCode!=U_USING_DEFAULT_WARNING) {
685         log_err("error getting the display name for a locale with an unknown language - %s\n", u_errorName(errorCode));
686     }
687 
688     {
689         int32_t i;
690         static const char *aLocale = "es@collation=traditional;calendar=japanese";
691         static const char *testL[] = { "en_US",
692             "fr_FR",
693             "ca_ES",
694             "el_GR" };
695         static const char *expect[] = { "Spanish (Calendar=Japanese Calendar, Sort Order=Traditional Sort Order)", /* note sorted order of keywords */
696             "espagnol (calendrier=Calendrier japonais, ordre de tri=Ordre traditionnel)",
697             "espanyol (calendari=calendari japon\\u00e8s, ordenaci\\u00f3=ordre tradicional)",
698             "\\u0399\\u03c3\\u03c0\\u03b1\\u03bd\\u03b9\\u03ba\\u03ac (\\u0397\\u03bc\\u03b5\\u03c1\\u03bf\\u03bb\\u03cc\\u03b3\\u03b9\\u03bf=\\u0399\\u03b1\\u03c0\\u03c9\\u03bd\\u03b9\\u03ba\\u03cc \\u03b7\\u03bc\\u03b5\\u03c1\\u03bf\\u03bb\\u03cc\\u03b3\\u03b9\\u03bf, \\u03a3\\u03b5\\u03b9\\u03c1\\u03ac \\u03c4\\u03b1\\u03be\\u03b9\\u03bd\\u03cc\\u03bc\\u03b7\\u03c3\\u03b7\\u03c2=\\u03a0\\u03b1\\u03c1\\u03b1\\u03b4\\u03bf\\u03c3\\u03b9\\u03b1\\u03ba\\u03ae \\u03c3\\u03b5\\u03b9\\u03c1\\u03ac \\u03c4\\u03b1\\u03be\\u03b9\\u03bd\\u03cc\\u03bc\\u03b7\\u03c3\\u03b7\\u03c2)" };
699         UChar *expectBuffer;
700 
701         for(i=0;i<LENGTHOF(testL);i++) {
702             errorCode = U_ZERO_ERROR;
703             uloc_getDisplayName(aLocale, testL[i], buffer, LENGTHOF(buffer), &errorCode);
704             if(U_FAILURE(errorCode)) {
705                 log_err("FAIL in uloc_getDisplayName(%s,%s,..) -> %s\n", aLocale, testL[i], u_errorName(errorCode));
706             } else {
707                 expectBuffer = CharsToUChars(expect[i]);
708                 if(u_strcmp(buffer,expectBuffer)) {
709                     log_data_err("FAIL in uloc_getDisplayName(%s,%s,..) expected '%s' got '%s' (Are you missing data?)\n", aLocale, testL[i], expect[i], austrdup(buffer));
710                 } else {
711                     log_verbose("pass in uloc_getDisplayName(%s,%s,..) got '%s'\n", aLocale, testL[i], expect[i]);
712                 }
713                 free(expectBuffer);
714             }
715         }
716     }
717 
718     /* test that we properly preflight and return data when there's a non-default pattern,
719        see ticket #8262. */
720     {
721         int32_t i;
722         static const char *locale="az_Cyrl";
723         static const char *displayLocale="ja";
724         static const char *expectedChars =
725                 "\\u30a2\\u30bc\\u30eb\\u30d0\\u30a4\\u30b8\\u30e3\\u30f3\\u8a9e "
726                 "(\\u30ad\\u30ea\\u30eb\\u6587\\u5b57)";
727         UErrorCode ec=U_ZERO_ERROR;
728         UChar result[256];
729         int32_t len;
730         int32_t preflightLen=uloc_getDisplayName(locale, displayLocale, NULL, 0, &ec);
731         /* inconvenient semantics when preflighting, this condition is expected... */
732         if(ec==U_BUFFER_OVERFLOW_ERROR) {
733             ec=U_ZERO_ERROR;
734         }
735         len=uloc_getDisplayName(locale, displayLocale, result, LENGTHOF(result), &ec);
736         if(U_FAILURE(ec)) {
737             log_err("uloc_getDisplayName(%s, %s...) returned error: %s",
738                     locale, displayLocale, u_errorName(ec));
739         } else {
740             UChar *expected=CharsToUChars(expectedChars);
741             int32_t expectedLen=u_strlen(expected);
742 
743             if(len!=expectedLen) {
744                 log_data_err("uloc_getDisplayName(%s, %s...) returned string of length %d, expected length %d",
745                         locale, displayLocale, len, expectedLen);
746             } else if(preflightLen!=expectedLen) {
747                 log_err("uloc_getDisplayName(%s, %s...) returned preflight length %d, expected length %d",
748                         locale, displayLocale, preflightLen, expectedLen);
749             } else if(u_strncmp(result, expected, len)) {
750                 int32_t cap=len*6+1;  /* worst case + space for trailing null */
751                 char* resultChars=(char*)malloc(cap);
752                 int32_t resultCharsLen=UCharsToEscapedAscii(result, len, resultChars, cap);
753                 if(resultCharsLen<0 || resultCharsLen<cap-1) {
754                     log_err("uloc_getDisplayName(%s, %s...) mismatch", locale, displayLocale);
755                 } else {
756                     log_err("uloc_getDisplayName(%s, %s...) returned '%s' but expected '%s'",
757                             locale, displayLocale, resultChars, expectedChars);
758                 }
759                 free(resultChars);
760                 resultChars=NULL;
761             } else {
762                 /* test all buffer sizes */
763                 for(i=len+1;i>=0;--i) {
764                     len=uloc_getDisplayName(locale, displayLocale, result, i, &ec);
765                     if(ec==U_BUFFER_OVERFLOW_ERROR) {
766                         ec=U_ZERO_ERROR;
767                     }
768                     if(U_FAILURE(ec)) {
769                         log_err("using buffer of length %d returned error %s", i, u_errorName(ec));
770                         break;
771                     }
772                     if(len!=expectedLen) {
773                         log_err("with buffer of length %d, expected length %d but got %d", i, expectedLen, len);
774                         break;
775                     }
776                     /* There's no guarantee about what's in the buffer if we've overflowed, in particular,
777                      * we don't know that it's been filled, so no point in checking. */
778                 }
779             }
780 
781             free(expected);
782         }
783     }
784 }
785 
786 
787 /* test for uloc_getAvialable()  and uloc_countAvilable()*/
TestGetAvailableLocales()788 static void TestGetAvailableLocales()
789 {
790 
791     const char *locList;
792     int32_t locCount,i;
793 
794     log_verbose("Testing the no of avialable locales\n");
795     locCount=uloc_countAvailable();
796     if (locCount == 0)
797         log_data_err("countAvailable() returned an empty list!\n");
798 
799     /* use something sensible w/o hardcoding the count */
800     else if(locCount < 0){
801         log_data_err("countAvailable() returned a wrong value!= %d\n", locCount);
802     }
803     else{
804         log_info("Number of locales returned = %d\n", locCount);
805     }
806     for(i=0;i<locCount;i++){
807         locList=uloc_getAvailable(i);
808 
809         log_verbose(" %s\n", locList);
810     }
811 }
812 
813 /* test for u_getDataDirectory, u_setDataDirectory, uloc_getISO3Language */
TestDataDirectory()814 static void TestDataDirectory()
815 {
816 
817     char            oldDirectory[512];
818     const char     *temp,*testValue1,*testValue2,*testValue3;
819     const char path[40] ="d:\\icu\\source\\test\\intltest" U_FILE_SEP_STRING; /*give the required path */
820 
821     log_verbose("Testing getDataDirectory()\n");
822     temp = u_getDataDirectory();
823     strcpy(oldDirectory, temp);
824 
825     testValue1=uloc_getISO3Language("en_US");
826     log_verbose("first fetch of language retrieved  %s\n", testValue1);
827 
828     if (0 != strcmp(testValue1,"eng")){
829         log_err("Initial check of ISO3 language failed: expected \"eng\", got  %s \n", testValue1);
830     }
831 
832     /*defining the path for DataDirectory */
833     log_verbose("Testing setDataDirectory\n");
834     u_setDataDirectory( path );
835     if(strcmp(path, u_getDataDirectory())==0)
836         log_verbose("setDataDirectory working fine\n");
837     else
838         log_err("Error in setDataDirectory. Directory not set correctly - came back as [%s], expected [%s]\n", u_getDataDirectory(), path);
839 
840     testValue2=uloc_getISO3Language("en_US");
841     log_verbose("second fetch of language retrieved  %s \n", testValue2);
842 
843     u_setDataDirectory(oldDirectory);
844     testValue3=uloc_getISO3Language("en_US");
845     log_verbose("third fetch of language retrieved  %s \n", testValue3);
846 
847     if (0 != strcmp(testValue3,"eng")) {
848        log_err("get/setDataDirectory() failed: expected \"eng\", got \" %s  \" \n", testValue3);
849     }
850 }
851 
852 
853 
854 /*=========================================================== */
855 
856 static UChar _NUL=0;
857 
doTestDisplayNames(const char * displayLocale,int32_t compareIndex)858 static void doTestDisplayNames(const char* displayLocale, int32_t compareIndex)
859 {
860     UErrorCode status = U_ZERO_ERROR;
861     int32_t i;
862     int32_t maxresultsize;
863 
864     const char *testLocale;
865 
866 
867     UChar  *testLang  = 0;
868     UChar  *testScript  = 0;
869     UChar  *testCtry = 0;
870     UChar  *testVar = 0;
871     UChar  *testName = 0;
872 
873 
874     UChar*  expectedLang = 0;
875     UChar*  expectedScript = 0;
876     UChar*  expectedCtry = 0;
877     UChar*  expectedVar = 0;
878     UChar*  expectedName = 0;
879 
880 setUpDataTable();
881 
882     for(i=0;i<LOCALE_SIZE; ++i)
883     {
884         testLocale=rawData2[NAME][i];
885 
886         log_verbose("Testing.....  %s\n", testLocale);
887 
888         maxresultsize=0;
889         maxresultsize=uloc_getDisplayLanguage(testLocale, displayLocale, NULL, maxresultsize, &status);
890         if(status==U_BUFFER_OVERFLOW_ERROR)
891         {
892             status=U_ZERO_ERROR;
893             testLang=(UChar*)malloc(sizeof(UChar) * (maxresultsize+1));
894             uloc_getDisplayLanguage(testLocale, displayLocale, testLang, maxresultsize + 1, &status);
895         }
896         else
897         {
898             testLang=&_NUL;
899         }
900         if(U_FAILURE(status)){
901             log_err("Error in getDisplayLanguage()  %s\n", myErrorName(status));
902         }
903 
904         maxresultsize=0;
905         maxresultsize=uloc_getDisplayScript(testLocale, displayLocale, NULL, maxresultsize, &status);
906         if(status==U_BUFFER_OVERFLOW_ERROR)
907         {
908             status=U_ZERO_ERROR;
909             testScript=(UChar*)malloc(sizeof(UChar) * (maxresultsize+1));
910             uloc_getDisplayScript(testLocale, displayLocale, testScript, maxresultsize + 1, &status);
911         }
912         else
913         {
914             testScript=&_NUL;
915         }
916         if(U_FAILURE(status)){
917             log_err("Error in getDisplayScript()  %s\n", myErrorName(status));
918         }
919 
920         maxresultsize=0;
921         maxresultsize=uloc_getDisplayCountry(testLocale, displayLocale, NULL, maxresultsize, &status);
922         if(status==U_BUFFER_OVERFLOW_ERROR)
923         {
924             status=U_ZERO_ERROR;
925             testCtry=(UChar*)malloc(sizeof(UChar) * (maxresultsize+1));
926             uloc_getDisplayCountry(testLocale, displayLocale, testCtry, maxresultsize + 1, &status);
927         }
928         else
929         {
930             testCtry=&_NUL;
931         }
932         if(U_FAILURE(status)){
933             log_err("Error in getDisplayCountry()  %s\n", myErrorName(status));
934         }
935 
936         maxresultsize=0;
937         maxresultsize=uloc_getDisplayVariant(testLocale, displayLocale, NULL, maxresultsize, &status);
938         if(status==U_BUFFER_OVERFLOW_ERROR)
939         {
940             status=U_ZERO_ERROR;
941             testVar=(UChar*)malloc(sizeof(UChar) * (maxresultsize+1));
942             uloc_getDisplayVariant(testLocale, displayLocale, testVar, maxresultsize + 1, &status);
943         }
944         else
945         {
946             testVar=&_NUL;
947         }
948         if(U_FAILURE(status)){
949                 log_err("Error in getDisplayVariant()  %s\n", myErrorName(status));
950         }
951 
952         maxresultsize=0;
953         maxresultsize=uloc_getDisplayName(testLocale, displayLocale, NULL, maxresultsize, &status);
954         if(status==U_BUFFER_OVERFLOW_ERROR)
955         {
956             status=U_ZERO_ERROR;
957             testName=(UChar*)malloc(sizeof(UChar) * (maxresultsize+1));
958             uloc_getDisplayName(testLocale, displayLocale, testName, maxresultsize + 1, &status);
959         }
960         else
961         {
962             testName=&_NUL;
963         }
964         if(U_FAILURE(status)){
965             log_err("Error in getDisplayName()  %s\n", myErrorName(status));
966         }
967 
968         expectedLang=dataTable[compareIndex][i];
969         if(u_strlen(expectedLang)== 0)
970             expectedLang=dataTable[DLANG_EN][i];
971 
972         expectedScript=dataTable[compareIndex + 1][i];
973         if(u_strlen(expectedScript)== 0)
974             expectedScript=dataTable[DSCRIPT_EN][i];
975 
976         expectedCtry=dataTable[compareIndex + 2][i];
977         if(u_strlen(expectedCtry)== 0)
978             expectedCtry=dataTable[DCTRY_EN][i];
979 
980         expectedVar=dataTable[compareIndex + 3][i];
981         if(u_strlen(expectedVar)== 0)
982             expectedVar=dataTable[DVAR_EN][i];
983 
984         expectedName=dataTable[compareIndex + 4][i];
985         if(u_strlen(expectedName) == 0)
986             expectedName=dataTable[DNAME_EN][i];
987 
988         if (0 !=u_strcmp(testLang,expectedLang))  {
989             log_data_err(" Display Language mismatch: got %s expected %s displayLocale=%s (Are you missing data?)\n", austrdup(testLang), austrdup(expectedLang), displayLocale);
990         }
991 
992         if (0 != u_strcmp(testScript,expectedScript))   {
993             log_data_err(" Display Script mismatch: got %s expected %s displayLocale=%s (Are you missing data?)\n", austrdup(testScript), austrdup(expectedScript), displayLocale);
994         }
995 
996         if (0 != u_strcmp(testCtry,expectedCtry))   {
997             log_data_err(" Display Country mismatch: got %s expected %s displayLocale=%s (Are you missing data?)\n", austrdup(testCtry), austrdup(expectedCtry), displayLocale);
998         }
999 
1000         if (0 != u_strcmp(testVar,expectedVar))    {
1001             log_data_err(" Display Variant mismatch: got %s expected %s displayLocale=%s (Are you missing data?)\n", austrdup(testVar), austrdup(expectedVar), displayLocale);
1002         }
1003 
1004         if(0 != u_strcmp(testName, expectedName))    {
1005             log_data_err(" Display Name mismatch: got %s expected %s displayLocale=%s (Are you missing data?)\n", austrdup(testName), austrdup(expectedName), displayLocale);
1006         }
1007 
1008         if(testName!=&_NUL) {
1009             free(testName);
1010         }
1011         if(testLang!=&_NUL) {
1012             free(testLang);
1013         }
1014         if(testScript!=&_NUL) {
1015             free(testScript);
1016         }
1017         if(testCtry!=&_NUL) {
1018             free(testCtry);
1019         }
1020         if(testVar!=&_NUL) {
1021             free(testVar);
1022         }
1023     }
1024 cleanUpDataTable();
1025 }
1026 
1027 /*------------------------------
1028  * TestDisplayNameBrackets
1029  */
1030 
1031 typedef struct {
1032     const char * displayLocale;
1033     const char * namedRegion;
1034     const char * namedLocale;
1035     const char * regionName;
1036     const char * localeName;
1037 } DisplayNameBracketsItem;
1038 
1039 static const DisplayNameBracketsItem displayNameBracketsItems[] = {
1040     { "en", "CC", "en_CC",      "Cocos (Keeling) Islands",  "English (Cocos [Keeling] Islands)"  },
1041     { "en", "MM", "my_MM",      "Myanmar (Burma)",          "Burmese (Myanmar [Burma])"          },
1042     { "en", "MM", "my_Mymr_MM", "Myanmar (Burma)",          "Burmese (Myanmar, Myanmar [Burma])" },
1043     { "zh", "CC", "en_CC",      "\\u79D1\\u79D1\\u65AF\\uFF08\\u57FA\\u6797\\uFF09\\u7FA4\\u5C9B", "\\u82F1\\u6587\\uFF08\\u79D1\\u79D1\\u65AF\\uFF3B\\u57FA\\u6797\\uFF3D\\u7FA4\\u5C9B\\uFF09" },
1044     { "zh", "CG", "fr_CG",      "\\u521A\\u679C\\uFF08\\u5E03\\uFF09",                             "\\u6CD5\\u6587\\uFF08\\u521A\\u679C\\uFF3B\\u5E03\\uFF3D\\uFF09" },
1045     { NULL, NULL, NULL,         NULL,                       NULL                                 }
1046 };
1047 
1048 enum { kDisplayNameBracketsMax = 128 };
1049 
TestDisplayNameBrackets()1050 static void TestDisplayNameBrackets()
1051 {
1052     const DisplayNameBracketsItem * itemPtr = displayNameBracketsItems;
1053     for (; itemPtr->displayLocale != NULL; itemPtr++) {
1054         ULocaleDisplayNames * uldn;
1055         UErrorCode status;
1056         UChar expectRegionName[kDisplayNameBracketsMax];
1057         UChar expectLocaleName[kDisplayNameBracketsMax];
1058         UChar getName[kDisplayNameBracketsMax];
1059         int32_t ulen;
1060 
1061         (void) u_unescape(itemPtr->regionName, expectRegionName, kDisplayNameBracketsMax);
1062         (void) u_unescape(itemPtr->localeName, expectLocaleName, kDisplayNameBracketsMax);
1063 
1064         status = U_ZERO_ERROR;
1065         ulen = uloc_getDisplayCountry(itemPtr->namedLocale, itemPtr->displayLocale, getName, kDisplayNameBracketsMax, &status);
1066         if ( U_FAILURE(status) || u_strcmp(getName, expectRegionName) != 0 ) {
1067             log_data_err("uloc_getDisplayCountry for displayLocale %s and namedLocale %s returns unexpected name or status %s\n", itemPtr->displayLocale, itemPtr->namedLocale, myErrorName(status));
1068         }
1069 
1070         status = U_ZERO_ERROR;
1071         ulen = uloc_getDisplayName(itemPtr->namedLocale, itemPtr->displayLocale, getName, kDisplayNameBracketsMax, &status);
1072         if ( U_FAILURE(status) || u_strcmp(getName, expectLocaleName) != 0 ) {
1073             log_data_err("uloc_getDisplayName for displayLocale %s and namedLocale %s returns unexpected name or status %s\n", itemPtr->displayLocale, itemPtr->namedLocale, myErrorName(status));
1074         }
1075 
1076 #if !UCONFIG_NO_FORMATTING
1077         status = U_ZERO_ERROR;
1078         uldn = uldn_open(itemPtr->displayLocale, ULDN_STANDARD_NAMES, &status);
1079         if (U_SUCCESS(status)) {
1080             status = U_ZERO_ERROR;
1081             ulen = uldn_regionDisplayName(uldn, itemPtr->namedRegion, getName, kDisplayNameBracketsMax, &status);
1082             if ( U_FAILURE(status) || u_strcmp(getName, expectRegionName) != 0 ) {
1083                 log_data_err("uldn_regionDisplayName for displayLocale %s and namedRegion %s returns unexpected name or status %s\n", itemPtr->displayLocale, itemPtr->namedRegion, myErrorName(status));
1084             }
1085 
1086             status = U_ZERO_ERROR;
1087             ulen = uldn_localeDisplayName(uldn, itemPtr->namedLocale, getName, kDisplayNameBracketsMax, &status);
1088             if ( U_FAILURE(status) || u_strcmp(getName, expectLocaleName) != 0 ) {
1089                 log_data_err("uldn_localeDisplayName for displayLocale %s and namedLocale %s returns unexpected name or status %s\n", itemPtr->displayLocale, itemPtr->namedLocale, myErrorName(status));
1090             }
1091 
1092             uldn_close(uldn);
1093         } else {
1094             log_data_err("uldn_open fails for displayLocale %s, status=%s\n", itemPtr->displayLocale, u_errorName(status));
1095         }
1096 #endif
1097     }
1098 }
1099 
1100 /*------------------------------
1101  * TestISOFunctions
1102  */
1103 
1104 #if !UCONFIG_NO_FILE_IO && !UCONFIG_NO_LEGACY_CONVERSION
1105 /* test for uloc_getISOLanguages, uloc_getISOCountries */
TestISOFunctions()1106 static void TestISOFunctions()
1107 {
1108     const char* const* str=uloc_getISOLanguages();
1109     const char* const* str1=uloc_getISOCountries();
1110     const char* test;
1111     const char *key = NULL;
1112     int32_t count = 0, skipped = 0;
1113     int32_t expect;
1114     UResourceBundle *res;
1115     UResourceBundle *subRes;
1116     UErrorCode status = U_ZERO_ERROR;
1117 
1118     /*  test getISOLanguages*/
1119     /*str=uloc_getISOLanguages(); */
1120     log_verbose("Testing ISO Languages: \n");
1121 
1122     /* use structLocale - this data is no longer in root */
1123     res = ures_openDirect(loadTestData(&status), "structLocale", &status);
1124     subRes = ures_getByKey(res, "Languages", NULL, &status);
1125     if (U_FAILURE(status)) {
1126         log_data_err("There is an error in structLocale's ures_getByKey(\"Languages\"), status=%s\n", u_errorName(status));
1127         return;
1128     }
1129 
1130     expect = ures_getSize(subRes);
1131     for(count = 0; *(str+count) != 0; count++)
1132     {
1133         key = NULL;
1134         test = *(str+count);
1135         status = U_ZERO_ERROR;
1136 
1137         do {
1138             /* Skip over language tags. This API only returns language codes. */
1139             skipped += (key != NULL);
1140             ures_getNextString(subRes, NULL, &key, &status);
1141         }
1142         while (key != NULL && strchr(key, '_'));
1143 
1144         if(key == NULL)
1145             break;
1146         /* TODO: Consider removing sh, which is deprecated */
1147         if(strcmp(key,"root") == 0 || strcmp(key,"Fallback") == 0 || strcmp(key,"sh") == 0) {
1148             ures_getNextString(subRes, NULL, &key, &status);
1149             skipped++;
1150         }
1151 #if U_CHARSET_FAMILY==U_ASCII_FAMILY
1152         /* This code only works on ASCII machines where the keys are stored in ASCII order */
1153         if(strcmp(test,key)) {
1154             /* The first difference usually implies the place where things get out of sync */
1155             log_err("FAIL Language diff at offset %d, \"%s\" != \"%s\"\n", count, test, key);
1156         }
1157 #endif
1158 
1159         if(!strcmp(test,"in"))
1160             log_err("FAIL getISOLanguages() has obsolete language code %s\n", test);
1161         if(!strcmp(test,"iw"))
1162             log_err("FAIL getISOLanguages() has obsolete language code %s\n", test);
1163         if(!strcmp(test,"ji"))
1164             log_err("FAIL getISOLanguages() has obsolete language code %s\n", test);
1165         if(!strcmp(test,"jw"))
1166             log_err("FAIL getISOLanguages() has obsolete language code %s\n", test);
1167         if(!strcmp(test,"sh"))
1168             log_err("FAIL getISOLanguages() has obsolete language code %s\n", test);
1169     }
1170 
1171     expect -= skipped; /* Ignore the skipped resources from structLocale */
1172 
1173     if(count!=expect) {
1174         log_err("There is an error in getISOLanguages, got %d, expected %d (as per structLocale)\n", count, expect);
1175     }
1176 
1177     subRes = ures_getByKey(res, "Countries", subRes, &status);
1178     log_verbose("Testing ISO Countries");
1179     skipped = 0;
1180     expect = ures_getSize(subRes) - 1; /* Skip ZZ */
1181     for(count = 0; *(str1+count) != 0; count++)
1182     {
1183         key = NULL;
1184         test = *(str1+count);
1185         do {
1186             /* Skip over numeric UN tags. This API only returns ISO-3166 codes. */
1187             skipped += (key != NULL);
1188             ures_getNextString(subRes, NULL, &key, &status);
1189         }
1190         while (key != NULL && strlen(key) != 2);
1191 
1192         if(key == NULL)
1193             break;
1194         /* TODO: Consider removing CS, which is deprecated */
1195         while(strcmp(key,"QO") == 0 || strcmp(key,"QU") == 0 || strcmp(key,"CS") == 0) {
1196             ures_getNextString(subRes, NULL, &key, &status);
1197             skipped++;
1198         }
1199 #if U_CHARSET_FAMILY==U_ASCII_FAMILY
1200         /* This code only works on ASCII machines where the keys are stored in ASCII order */
1201         if(strcmp(test,key)) {
1202             /* The first difference usually implies the place where things get out of sync */
1203             log_err("FAIL Country diff at offset %d, \"%s\" != \"%s\"\n", count, test, key);
1204         }
1205 #endif
1206         if(!strcmp(test,"FX"))
1207             log_err("FAIL getISOCountries() has obsolete country code %s\n", test);
1208         if(!strcmp(test,"YU"))
1209             log_err("FAIL getISOCountries() has obsolete country code %s\n", test);
1210         if(!strcmp(test,"ZR"))
1211             log_err("FAIL getISOCountries() has obsolete country code %s\n", test);
1212     }
1213 
1214     ures_getNextString(subRes, NULL, &key, &status);
1215     if (strcmp(key, "ZZ") != 0) {
1216         log_err("ZZ was expected to be the last entry in structLocale, but got %s\n", key);
1217     }
1218 #if U_CHARSET_FAMILY==U_EBCDIC_FAMILY
1219     /* On EBCDIC machines, the numbers are sorted last. Account for those in the skipped value too. */
1220     key = NULL;
1221     do {
1222         /* Skip over numeric UN tags. uloc_getISOCountries only returns ISO-3166 codes. */
1223         skipped += (key != NULL);
1224         ures_getNextString(subRes, NULL, &key, &status);
1225     }
1226     while (U_SUCCESS(status) && key != NULL && strlen(key) != 2);
1227 #endif
1228     expect -= skipped; /* Ignore the skipped resources from structLocale */
1229     if(count!=expect)
1230     {
1231         log_err("There is an error in getISOCountries, got %d, expected %d \n", count, expect);
1232     }
1233     ures_close(subRes);
1234     ures_close(res);
1235 }
1236 #endif
1237 
setUpDataTable()1238 static void setUpDataTable()
1239 {
1240     int32_t i,j;
1241     dataTable = (UChar***)(calloc(sizeof(UChar**),LOCALE_INFO_SIZE));
1242 
1243     for (i = 0; i < LOCALE_INFO_SIZE; i++) {
1244         dataTable[i] = (UChar**)(calloc(sizeof(UChar*),LOCALE_SIZE));
1245         for (j = 0; j < LOCALE_SIZE; j++){
1246             dataTable[i][j] = CharsToUChars(rawData2[i][j]);
1247         }
1248     }
1249 }
1250 
cleanUpDataTable()1251 static void cleanUpDataTable()
1252 {
1253     int32_t i,j;
1254     if(dataTable != NULL) {
1255         for (i=0; i<LOCALE_INFO_SIZE; i++) {
1256             for(j = 0; j < LOCALE_SIZE; j++) {
1257                 free(dataTable[i][j]);
1258             }
1259             free(dataTable[i]);
1260         }
1261         free(dataTable);
1262     }
1263     dataTable = NULL;
1264 }
1265 
1266 /**
1267  * @bug 4011756 4011380
1268  */
TestISO3Fallback()1269 static void TestISO3Fallback()
1270 {
1271     const char* test="xx_YY";
1272 
1273     const char * result;
1274 
1275     result = uloc_getISO3Language(test);
1276 
1277     /* Conform to C API usage  */
1278 
1279     if (!result || (result[0] != 0))
1280        log_err("getISO3Language() on xx_YY returned %s instead of \"\"");
1281 
1282     result = uloc_getISO3Country(test);
1283 
1284     if (!result || (result[0] != 0))
1285         log_err("getISO3Country() on xx_YY returned %s instead of \"\"");
1286 }
1287 
1288 /**
1289  * @bug 4118587
1290  */
TestSimpleDisplayNames()1291 static void TestSimpleDisplayNames()
1292 {
1293   /*
1294      This test is different from TestDisplayNames because TestDisplayNames checks
1295      fallback behavior, combination of language and country names to form locale
1296      names, and other stuff like that.  This test just checks specific language
1297      and country codes to make sure we have the correct names for them.
1298   */
1299     char languageCodes[] [4] = { "he", "id", "iu", "ug", "yi", "za", "419" };
1300     const char* languageNames [] = { "Hebrew", "Indonesian", "Inuktitut", "Uyghur", "Yiddish",
1301                                "Zhuang", "419" };
1302     const char* inLocale [] = { "en_US", "zh_Hant"};
1303     UErrorCode status=U_ZERO_ERROR;
1304 
1305     int32_t i;
1306     int32_t localeIndex = 0;
1307     for (i = 0; i < 7; i++) {
1308         UChar *testLang=0;
1309         UChar *expectedLang=0;
1310         int size=0;
1311 
1312         if (i == 6) {
1313             localeIndex = 1; /* Use the second locale for the rest of the test. */
1314         }
1315 
1316         size=uloc_getDisplayLanguage(languageCodes[i], inLocale[localeIndex], NULL, size, &status);
1317         if(status==U_BUFFER_OVERFLOW_ERROR) {
1318             status=U_ZERO_ERROR;
1319             testLang=(UChar*)malloc(sizeof(UChar) * (size + 1));
1320             uloc_getDisplayLanguage(languageCodes[i], inLocale[localeIndex], testLang, size + 1, &status);
1321         }
1322         expectedLang=(UChar*)malloc(sizeof(UChar) * (strlen(languageNames[i])+1));
1323         u_uastrcpy(expectedLang, languageNames[i]);
1324         if (u_strcmp(testLang, expectedLang) != 0)
1325             log_data_err("Got wrong display name for %s : Expected \"%s\", got \"%s\".\n",
1326                     languageCodes[i], languageNames[i], austrdup(testLang));
1327         free(testLang);
1328         free(expectedLang);
1329     }
1330 
1331 }
1332 
1333 /**
1334  * @bug 4118595
1335  */
TestUninstalledISO3Names()1336 static void TestUninstalledISO3Names()
1337 {
1338   /* This test checks to make sure getISO3Language and getISO3Country work right
1339      even for locales that are not installed. */
1340     static const char iso2Languages [][4] = {     "am", "ba", "fy", "mr", "rn",
1341                                         "ss", "tw", "zu" };
1342     static const char iso3Languages [][5] = {     "amh", "bak", "fry", "mar", "run",
1343                                         "ssw", "twi", "zul" };
1344     static const char iso2Countries [][6] = {     "am_AF", "ba_BW", "fy_KZ", "mr_MO", "rn_MN",
1345                                         "ss_SB", "tw_TC", "zu_ZW" };
1346     static const char iso3Countries [][4] = {     "AFG", "BWA", "KAZ", "MAC", "MNG",
1347                                         "SLB", "TCA", "ZWE" };
1348     int32_t i;
1349 
1350     for (i = 0; i < 8; i++) {
1351       UErrorCode err = U_ZERO_ERROR;
1352       const char *test;
1353       test = uloc_getISO3Language(iso2Languages[i]);
1354       if(strcmp(test, iso3Languages[i]) !=0 || U_FAILURE(err))
1355          log_err("Got wrong ISO3 code for %s : Expected \"%s\", got \"%s\". %s\n",
1356                      iso2Languages[i], iso3Languages[i], test, myErrorName(err));
1357     }
1358     for (i = 0; i < 8; i++) {
1359       UErrorCode err = U_ZERO_ERROR;
1360       const char *test;
1361       test = uloc_getISO3Country(iso2Countries[i]);
1362       if(strcmp(test, iso3Countries[i]) !=0 || U_FAILURE(err))
1363          log_err("Got wrong ISO3 code for %s : Expected \"%s\", got \"%s\". %s\n",
1364                      iso2Countries[i], iso3Countries[i], test, myErrorName(err));
1365     }
1366 }
1367 
1368 
TestVariantParsing()1369 static void TestVariantParsing()
1370 {
1371     static const char* en_US_custom="en_US_De Anza_Cupertino_California_United States_Earth";
1372     static const char* dispName="English (United States, DE ANZA_CUPERTINO_CALIFORNIA_UNITED STATES_EARTH)";
1373     static const char* dispVar="DE ANZA_CUPERTINO_CALIFORNIA_UNITED STATES_EARTH";
1374     static const char* shortVariant="fr_FR_foo";
1375     static const char* bogusVariant="fr_FR__foo";
1376     static const char* bogusVariant2="fr_FR_foo_";
1377     static const char* bogusVariant3="fr_FR__foo_";
1378 
1379 
1380     UChar displayVar[100];
1381     UChar displayName[100];
1382     UErrorCode status=U_ZERO_ERROR;
1383     UChar* got=0;
1384     int32_t size=0;
1385     size=uloc_getDisplayVariant(en_US_custom, "en_US", NULL, size, &status);
1386     if(status==U_BUFFER_OVERFLOW_ERROR) {
1387         status=U_ZERO_ERROR;
1388         got=(UChar*)realloc(got, sizeof(UChar) * (size+1));
1389         uloc_getDisplayVariant(en_US_custom, "en_US", got, size + 1, &status);
1390     }
1391     else {
1392         log_err("FAIL: Didn't get U_BUFFER_OVERFLOW_ERROR\n");
1393     }
1394     u_uastrcpy(displayVar, dispVar);
1395     if(u_strcmp(got,displayVar)!=0) {
1396         log_err("FAIL: getDisplayVariant() Wanted %s, got %s\n", dispVar, austrdup(got));
1397     }
1398     size=0;
1399     size=uloc_getDisplayName(en_US_custom, "en_US", NULL, size, &status);
1400     if(status==U_BUFFER_OVERFLOW_ERROR) {
1401         status=U_ZERO_ERROR;
1402         got=(UChar*)realloc(got, sizeof(UChar) * (size+1));
1403         uloc_getDisplayName(en_US_custom, "en_US", got, size + 1, &status);
1404     }
1405     else {
1406         log_err("FAIL: Didn't get U_BUFFER_OVERFLOW_ERROR\n");
1407     }
1408     u_uastrcpy(displayName, dispName);
1409     if(u_strcmp(got,displayName)!=0) {
1410         if (status == U_USING_DEFAULT_WARNING) {
1411             log_data_err("FAIL: getDisplayName() got %s. Perhaps you are missing data?\n", u_errorName(status));
1412         } else {
1413             log_err("FAIL: getDisplayName() Wanted %s, got %s\n", dispName, austrdup(got));
1414         }
1415     }
1416 
1417     size=0;
1418     status=U_ZERO_ERROR;
1419     size=uloc_getDisplayVariant(shortVariant, NULL, NULL, size, &status);
1420     if(status==U_BUFFER_OVERFLOW_ERROR) {
1421         status=U_ZERO_ERROR;
1422         got=(UChar*)realloc(got, sizeof(UChar) * (size+1));
1423         uloc_getDisplayVariant(shortVariant, NULL, got, size + 1, &status);
1424     }
1425     else {
1426         log_err("FAIL: Didn't get U_BUFFER_OVERFLOW_ERROR\n");
1427     }
1428     if(strcmp(austrdup(got),"FOO")!=0) {
1429         log_err("FAIL: getDisplayVariant()  Wanted: foo  Got: %s\n", austrdup(got));
1430     }
1431     size=0;
1432     status=U_ZERO_ERROR;
1433     size=uloc_getDisplayVariant(bogusVariant, NULL, NULL, size, &status);
1434     if(status==U_BUFFER_OVERFLOW_ERROR) {
1435         status=U_ZERO_ERROR;
1436         got=(UChar*)realloc(got, sizeof(UChar) * (size+1));
1437         uloc_getDisplayVariant(bogusVariant, NULL, got, size + 1, &status);
1438     }
1439     else {
1440         log_err("FAIL: Didn't get U_BUFFER_OVERFLOW_ERROR\n");
1441     }
1442     if(strcmp(austrdup(got),"_FOO")!=0) {
1443         log_err("FAIL: getDisplayVariant()  Wanted: _FOO  Got: %s\n", austrdup(got));
1444     }
1445     size=0;
1446     status=U_ZERO_ERROR;
1447     size=uloc_getDisplayVariant(bogusVariant2, NULL, NULL, size, &status);
1448     if(status==U_BUFFER_OVERFLOW_ERROR) {
1449         status=U_ZERO_ERROR;
1450         got=(UChar*)realloc(got, sizeof(UChar) * (size+1));
1451         uloc_getDisplayVariant(bogusVariant2, NULL, got, size + 1, &status);
1452     }
1453     else {
1454         log_err("FAIL: Didn't get U_BUFFER_OVERFLOW_ERROR\n");
1455     }
1456     if(strcmp(austrdup(got),"FOO_")!=0) {
1457         log_err("FAIL: getDisplayVariant()  Wanted: FOO_  Got: %s\n", austrdup(got));
1458     }
1459     size=0;
1460     status=U_ZERO_ERROR;
1461     size=uloc_getDisplayVariant(bogusVariant3, NULL, NULL, size, &status);
1462     if(status==U_BUFFER_OVERFLOW_ERROR) {
1463         status=U_ZERO_ERROR;
1464         got=(UChar*)realloc(got, sizeof(UChar) * (size+1));
1465         uloc_getDisplayVariant(bogusVariant3, NULL, got, size + 1, &status);
1466     }
1467     else {
1468         log_err("FAIL: Didn't get U_BUFFER_OVERFLOW_ERROR\n");
1469     }
1470     if(strcmp(austrdup(got),"_FOO_")!=0) {
1471         log_err("FAIL: getDisplayVariant()  Wanted: _FOO_  Got: %s\n", austrdup(got));
1472     }
1473     free(got);
1474 }
1475 
1476 
TestObsoleteNames(void)1477 static void TestObsoleteNames(void)
1478 {
1479     int32_t i;
1480     UErrorCode status = U_ZERO_ERROR;
1481     char buff[256];
1482 
1483     static const struct
1484     {
1485         char locale[9];
1486         char lang3[4];
1487         char lang[4];
1488         char ctry3[4];
1489         char ctry[4];
1490     } tests[] =
1491     {
1492         { "eng_USA", "eng", "en", "USA", "US" },
1493         { "kok",  "kok", "kok", "", "" },
1494         { "in",  "ind", "in", "", "" },
1495         { "id",  "ind", "id", "", "" }, /* NO aliasing */
1496         { "sh",  "srp", "sh", "", "" },
1497         { "zz_CS",  "", "zz", "SCG", "CS" },
1498         { "zz_FX",  "", "zz", "FXX", "FX" },
1499         { "zz_RO",  "", "zz", "ROU", "RO" },
1500         { "zz_TP",  "", "zz", "TMP", "TP" },
1501         { "zz_TL",  "", "zz", "TLS", "TL" },
1502         { "zz_ZR",  "", "zz", "ZAR", "ZR" },
1503         { "zz_FXX",  "", "zz", "FXX", "FX" }, /* no aliasing. Doesn't go to PS(PSE). */
1504         { "zz_ROM",  "", "zz", "ROU", "RO" },
1505         { "zz_ROU",  "", "zz", "ROU", "RO" },
1506         { "zz_ZAR",  "", "zz", "ZAR", "ZR" },
1507         { "zz_TMP",  "", "zz", "TMP", "TP" },
1508         { "zz_TLS",  "", "zz", "TLS", "TL" },
1509         { "zz_YUG",  "", "zz", "YUG", "YU" },
1510         { "mlt_PSE", "mlt", "mt", "PSE", "PS" },
1511         { "iw", "heb", "iw", "", "" },
1512         { "ji", "yid", "ji", "", "" },
1513         { "jw", "jaw", "jw", "", "" },
1514         { "sh", "srp", "sh", "", "" },
1515         { "", "", "", "", "" }
1516     };
1517 
1518     for(i=0;tests[i].locale[0];i++)
1519     {
1520         const char *locale;
1521 
1522         locale = tests[i].locale;
1523         log_verbose("** %s:\n", locale);
1524 
1525         status = U_ZERO_ERROR;
1526         if(strcmp(tests[i].lang3,uloc_getISO3Language(locale)))
1527         {
1528             log_err("FAIL: uloc_getISO3Language(%s)==\t\"%s\",\t expected \"%s\"\n",
1529                 locale,  uloc_getISO3Language(locale), tests[i].lang3);
1530         }
1531         else
1532         {
1533             log_verbose("   uloc_getISO3Language()==\t\"%s\"\n",
1534                 uloc_getISO3Language(locale) );
1535         }
1536 
1537         status = U_ZERO_ERROR;
1538         uloc_getLanguage(locale, buff, 256, &status);
1539         if(U_FAILURE(status))
1540         {
1541             log_err("FAIL: error getting language from %s\n", locale);
1542         }
1543         else
1544         {
1545             if(strcmp(buff,tests[i].lang))
1546             {
1547                 log_err("FAIL: uloc_getLanguage(%s)==\t\"%s\"\t expected \"%s\"\n",
1548                     locale, buff, tests[i].lang);
1549             }
1550             else
1551             {
1552                 log_verbose("  uloc_getLanguage(%s)==\t%s\n", locale, buff);
1553             }
1554         }
1555         if(strcmp(tests[i].lang3,uloc_getISO3Language(locale)))
1556         {
1557             log_err("FAIL: uloc_getISO3Language(%s)==\t\"%s\",\t expected \"%s\"\n",
1558                 locale,  uloc_getISO3Language(locale), tests[i].lang3);
1559         }
1560         else
1561         {
1562             log_verbose("   uloc_getISO3Language()==\t\"%s\"\n",
1563                 uloc_getISO3Language(locale) );
1564         }
1565 
1566         if(strcmp(tests[i].ctry3,uloc_getISO3Country(locale)))
1567         {
1568             log_err("FAIL: uloc_getISO3Country(%s)==\t\"%s\",\t expected \"%s\"\n",
1569                 locale,  uloc_getISO3Country(locale), tests[i].ctry3);
1570         }
1571         else
1572         {
1573             log_verbose("   uloc_getISO3Country()==\t\"%s\"\n",
1574                 uloc_getISO3Country(locale) );
1575         }
1576 
1577         status = U_ZERO_ERROR;
1578         uloc_getCountry(locale, buff, 256, &status);
1579         if(U_FAILURE(status))
1580         {
1581             log_err("FAIL: error getting country from %s\n", locale);
1582         }
1583         else
1584         {
1585             if(strcmp(buff,tests[i].ctry))
1586             {
1587                 log_err("FAIL: uloc_getCountry(%s)==\t\"%s\"\t expected \"%s\"\n",
1588                     locale, buff, tests[i].ctry);
1589             }
1590             else
1591             {
1592                 log_verbose("  uloc_getCountry(%s)==\t%s\n", locale, buff);
1593             }
1594         }
1595     }
1596 
1597     if (uloc_getLCID("iw_IL") != uloc_getLCID("he_IL")) {
1598         log_err("he,iw LCID mismatch: %X versus %X\n", uloc_getLCID("iw_IL"), uloc_getLCID("he_IL"));
1599     }
1600 
1601     if (uloc_getLCID("iw") != uloc_getLCID("he")) {
1602         log_err("he,iw LCID mismatch: %X versus %X\n", uloc_getLCID("iw"), uloc_getLCID("he"));
1603     }
1604 
1605 #if 0
1606 
1607     i = uloc_getLanguage("kok",NULL,0,&icu_err);
1608     if(U_FAILURE(icu_err))
1609     {
1610         log_err("FAIL: Got %s trying to do uloc_getLanguage(kok)\n", u_errorName(icu_err));
1611     }
1612 
1613     icu_err = U_ZERO_ERROR;
1614     uloc_getLanguage("kok",r1_buff,12,&icu_err);
1615     if(U_FAILURE(icu_err))
1616     {
1617         log_err("FAIL: Got %s trying to do uloc_getLanguage(kok, buff)\n", u_errorName(icu_err));
1618     }
1619 
1620     r1_addr = (char *)uloc_getISO3Language("kok");
1621 
1622     icu_err = U_ZERO_ERROR;
1623     if (strcmp(r1_buff,"kok") != 0)
1624     {
1625         log_err("FAIL: uloc_getLanguage(kok)==%s not kok\n",r1_buff);
1626         line--;
1627     }
1628     r1_addr = (char *)uloc_getISO3Language("in");
1629     i = uloc_getLanguage(r1_addr,r1_buff,12,&icu_err);
1630     if (strcmp(r1_buff,"id") != 0)
1631     {
1632         printf("uloc_getLanguage error (%s)\n",r1_buff);
1633         line--;
1634     }
1635     r1_addr = (char *)uloc_getISO3Language("sh");
1636     i = uloc_getLanguage(r1_addr,r1_buff,12,&icu_err);
1637     if (strcmp(r1_buff,"sr") != 0)
1638     {
1639         printf("uloc_getLanguage error (%s)\n",r1_buff);
1640         line--;
1641     }
1642 
1643     r1_addr = (char *)uloc_getISO3Country("zz_ZR");
1644     strcpy(p1_buff,"zz_");
1645     strcat(p1_buff,r1_addr);
1646     i = uloc_getCountry(p1_buff,r1_buff,12,&icu_err);
1647     if (strcmp(r1_buff,"ZR") != 0)
1648     {
1649         printf("uloc_getCountry error (%s)\n",r1_buff);
1650         line--;
1651     }
1652     r1_addr = (char *)uloc_getISO3Country("zz_FX");
1653     strcpy(p1_buff,"zz_");
1654     strcat(p1_buff,r1_addr);
1655     i = uloc_getCountry(p1_buff,r1_buff,12,&icu_err);
1656     if (strcmp(r1_buff,"FX") != 0)
1657     {
1658         printf("uloc_getCountry error (%s)\n",r1_buff);
1659         line--;
1660     }
1661 
1662 #endif
1663 
1664 }
1665 
TestKeywordVariants(void)1666 static void TestKeywordVariants(void)
1667 {
1668     static const struct {
1669         const char *localeID;
1670         const char *expectedLocaleID;
1671         const char *expectedLocaleIDNoKeywords;
1672         const char *expectedCanonicalID;
1673         const char *expectedKeywords[10];
1674         int32_t numKeywords;
1675         UErrorCode expectedStatus; /* from uloc_openKeywords */
1676     } testCases[] = {
1677         {
1678             "de_DE@  currency = euro; C o ll A t i o n   = Phonebook   ; C alen dar = buddhist   ",
1679             "de_DE@calendar=buddhist;collation=Phonebook;currency=euro",
1680             "de_DE",
1681             "de_DE@calendar=buddhist;collation=Phonebook;currency=euro",
1682             {"calendar", "collation", "currency"},
1683             3,
1684             U_ZERO_ERROR
1685         },
1686         {
1687             "de_DE@euro",
1688             "de_DE@euro",
1689             "de_DE",
1690             "de_DE@currency=EUR",
1691             {"","","","","","",""},
1692             0,
1693             U_INVALID_FORMAT_ERROR /* must have '=' after '@' */
1694         },
1695         {
1696             "de_DE@euro;collation=phonebook",
1697             "de_DE", /* error result; bad format */
1698             "de_DE", /* error result; bad format */
1699             "de_DE", /* error result; bad format */
1700             {"","","","","","",""},
1701             0,
1702             U_INVALID_FORMAT_ERROR
1703         }
1704     };
1705     UErrorCode status = U_ZERO_ERROR;
1706 
1707     int32_t i = 0, j = 0;
1708     int32_t resultLen = 0;
1709     char buffer[256];
1710     UEnumeration *keywords;
1711     int32_t keyCount = 0;
1712     const char *keyword = NULL;
1713     int32_t keywordLen = 0;
1714 
1715     for(i = 0; i < sizeof(testCases)/sizeof(testCases[0]); i++) {
1716         status = U_ZERO_ERROR;
1717         *buffer = 0;
1718         keywords = uloc_openKeywords(testCases[i].localeID, &status);
1719 
1720         if(status != testCases[i].expectedStatus) {
1721             log_err("Expected to uloc_openKeywords(\"%s\") => status %s. Got %s instead\n",
1722                     testCases[i].localeID,
1723                     u_errorName(testCases[i].expectedStatus), u_errorName(status));
1724         }
1725         status = U_ZERO_ERROR;
1726         if(keywords) {
1727             if((keyCount = uenum_count(keywords, &status)) != testCases[i].numKeywords) {
1728                 log_err("Expected to get %i keywords, got %i\n", testCases[i].numKeywords, keyCount);
1729             }
1730             if(keyCount) {
1731                 j = 0;
1732                 while((keyword = uenum_next(keywords, &keywordLen, &status))) {
1733                     if(strcmp(keyword, testCases[i].expectedKeywords[j]) != 0) {
1734                         log_err("Expected to get keyword value %s, got %s\n", testCases[i].expectedKeywords[j], keyword);
1735                     }
1736                     j++;
1737                 }
1738                 j = 0;
1739                 uenum_reset(keywords, &status);
1740                 while((keyword = uenum_next(keywords, &keywordLen, &status))) {
1741                     if(strcmp(keyword, testCases[i].expectedKeywords[j]) != 0) {
1742                         log_err("Expected to get keyword value %s, got %s\n", testCases[i].expectedKeywords[j], keyword);
1743                     }
1744                     j++;
1745                 }
1746             }
1747             uenum_close(keywords);
1748         }
1749         resultLen = uloc_getName(testCases[i].localeID, buffer, 256, &status);
1750         (void)resultLen;    /* Suppress set but not used warning. */
1751         if (uprv_strcmp(testCases[i].expectedLocaleID, buffer) != 0) {
1752             log_err("Expected uloc_getName(\"%s\") => \"%s\"; got \"%s\"\n",
1753                     testCases[i].localeID, testCases[i].expectedLocaleID, buffer);
1754         }
1755         resultLen = uloc_canonicalize(testCases[i].localeID, buffer, 256, &status);
1756         if (uprv_strcmp(testCases[i].expectedCanonicalID, buffer) != 0) {
1757             log_err("Expected uloc_canonicalize(\"%s\") => \"%s\"; got \"%s\"\n",
1758                     testCases[i].localeID, testCases[i].expectedCanonicalID, buffer);
1759         }
1760     }
1761 
1762 }
1763 
TestKeywordVariantParsing(void)1764 static void TestKeywordVariantParsing(void)
1765 {
1766     static const struct {
1767         const char *localeID;
1768         const char *keyword;
1769         const char *expectedValue;
1770     } testCases[] = {
1771         { "de_DE@  C o ll A t i o n   = Phonebook   ", "c o ll a t i o n", "Phonebook" },
1772         { "de_DE", "collation", ""},
1773         { "de_DE@collation=PHONEBOOK", "collation", "PHONEBOOK" },
1774         { "de_DE@currency = euro; CoLLaTion   = PHONEBOOk", "collatiON", "PHONEBOOk" },
1775     };
1776 
1777     UErrorCode status = U_ZERO_ERROR;
1778 
1779     int32_t i = 0;
1780     int32_t resultLen = 0;
1781     char buffer[256];
1782 
1783     for(i = 0; i < sizeof(testCases)/sizeof(testCases[0]); i++) {
1784         *buffer = 0;
1785         resultLen = uloc_getKeywordValue(testCases[i].localeID, testCases[i].keyword, buffer, 256, &status);
1786         (void)resultLen;    /* Suppress set but not used warning. */
1787         if(uprv_strcmp(testCases[i].expectedValue, buffer) != 0) {
1788             log_err("Expected to extract \"%s\" from \"%s\" for keyword \"%s\". Got \"%s\" instead\n",
1789                 testCases[i].expectedValue, testCases[i].localeID, testCases[i].keyword, buffer);
1790         }
1791     }
1792 }
1793 
1794 static const struct {
1795   const char *l; /* locale */
1796   const char *k; /* kw */
1797   const char *v; /* value */
1798   const char *x; /* expected */
1799 } kwSetTestCases[] = {
1800 #if 1
1801   { "en_US", "calendar", "japanese", "en_US@calendar=japanese" },
1802   { "en_US@", "calendar", "japanese", "en_US@calendar=japanese" },
1803   { "en_US@calendar=islamic", "calendar", "japanese", "en_US@calendar=japanese" },
1804   { "en_US@calendar=slovakian", "calendar", "gregorian", "en_US@calendar=gregorian" }, /* don't know what this means, but it has the same # of chars as gregorian */
1805   { "en_US@calendar=gregorian", "calendar", "japanese", "en_US@calendar=japanese" },
1806   { "de", "Currency", "CHF", "de@currency=CHF" },
1807   { "de", "Currency", "CHF", "de@currency=CHF" },
1808 
1809   { "en_US@collation=phonebook", "calendar", "japanese", "en_US@calendar=japanese;collation=phonebook" },
1810   { "en_US@calendar=japanese", "collation", "phonebook", "en_US@calendar=japanese;collation=phonebook" },
1811   { "de@collation=phonebook", "Currency", "CHF", "de@collation=phonebook;currency=CHF" },
1812   { "en_US@calendar=gregorian;collation=phonebook", "calendar", "japanese", "en_US@calendar=japanese;collation=phonebook" },
1813   { "en_US@calendar=slovakian;collation=phonebook", "calendar", "gregorian", "en_US@calendar=gregorian;collation=phonebook" }, /* don't know what this means, but it has the same # of chars as gregorian */
1814   { "en_US@calendar=slovakian;collation=videobook", "collation", "phonebook", "en_US@calendar=slovakian;collation=phonebook" }, /* don't know what this means, but it has the same # of chars as gregorian */
1815   { "en_US@calendar=islamic;collation=phonebook", "calendar", "japanese", "en_US@calendar=japanese;collation=phonebook" },
1816   { "de@collation=phonebook", "Currency", "CHF", "de@collation=phonebook;currency=CHF" },
1817 #endif
1818 #if 1
1819   { "mt@a=0;b=1;c=2;d=3", "c","j", "mt@a=0;b=1;c=j;d=3" },
1820   { "mt@a=0;b=1;c=2;d=3", "x","j", "mt@a=0;b=1;c=2;d=3;x=j" },
1821   { "mt@a=0;b=1;c=2;d=3", "a","f", "mt@a=f;b=1;c=2;d=3" },
1822   { "mt@a=0;aa=1;aaa=3", "a","x", "mt@a=x;aa=1;aaa=3" },
1823   { "mt@a=0;aa=1;aaa=3", "aa","x", "mt@a=0;aa=x;aaa=3" },
1824   { "mt@a=0;aa=1;aaa=3", "aaa","x", "mt@a=0;aa=1;aaa=x" },
1825   { "mt@a=0;aa=1;aaa=3", "a","yy", "mt@a=yy;aa=1;aaa=3" },
1826   { "mt@a=0;aa=1;aaa=3", "aa","yy", "mt@a=0;aa=yy;aaa=3" },
1827   { "mt@a=0;aa=1;aaa=3", "aaa","yy", "mt@a=0;aa=1;aaa=yy" },
1828 #endif
1829 #if 1
1830   /* removal tests */
1831   /* 1. removal of item at end */
1832   { "de@collation=phonebook;currency=CHF", "currency",   "", "de@collation=phonebook" },
1833   { "de@collation=phonebook;currency=CHF", "currency", NULL, "de@collation=phonebook" },
1834   /* 2. removal of item at beginning */
1835   { "de@collation=phonebook;currency=CHF", "collation", "", "de@currency=CHF" },
1836   { "de@collation=phonebook;currency=CHF", "collation", NULL, "de@currency=CHF" },
1837   /* 3. removal of an item not there */
1838   { "de@collation=phonebook;currency=CHF", "calendar", NULL, "de@collation=phonebook;currency=CHF" },
1839   /* 4. removal of only item */
1840   { "de@collation=phonebook", "collation", NULL, "de" },
1841 #endif
1842   { "de@collation=phonebook", "Currency", "CHF", "de@collation=phonebook;currency=CHF" }
1843 };
1844 
1845 
TestKeywordSet(void)1846 static void TestKeywordSet(void)
1847 {
1848     int32_t i = 0;
1849     int32_t resultLen = 0;
1850     char buffer[1024];
1851 
1852     char cbuffer[1024];
1853 
1854     for(i = 0; i < sizeof(kwSetTestCases)/sizeof(kwSetTestCases[0]); i++) {
1855         UErrorCode status = U_ZERO_ERROR;
1856         memset(buffer,'%',1023);
1857         strcpy(buffer, kwSetTestCases[i].l);
1858 
1859         uloc_canonicalize(kwSetTestCases[i].l, cbuffer, 1023, &status);
1860         if(strcmp(buffer,cbuffer)) {
1861           log_verbose("note: [%d] wasn't canonical, should be: '%s' not '%s'. Won't check for canonicity in output.\n", i, cbuffer, buffer);
1862         }
1863           /* sanity check test case results for canonicity */
1864         uloc_canonicalize(kwSetTestCases[i].x, cbuffer, 1023, &status);
1865         if(strcmp(kwSetTestCases[i].x,cbuffer)) {
1866           log_err("%s:%d: ERROR: kwSetTestCases[%d].x = '%s', should be %s (must be canonical)\n", __FILE__, __LINE__, i, kwSetTestCases[i].x, cbuffer);
1867         }
1868 
1869         resultLen = uloc_setKeywordValue(kwSetTestCases[i].k, kwSetTestCases[i].v, buffer, 1023, &status);
1870         if(U_FAILURE(status)) {
1871           log_err("Err on test case %d: got error %s\n", i, u_errorName(status));
1872           continue;
1873         }
1874         if(strcmp(buffer,kwSetTestCases[i].x) || ((int32_t)strlen(buffer)!=resultLen)) {
1875           log_err("FAIL: #%d: %s + [%s=%s] -> %s (%d) expected %s (%d)\n", i, kwSetTestCases[i].l, kwSetTestCases[i].k,
1876                   kwSetTestCases[i].v, buffer, resultLen, kwSetTestCases[i].x, strlen(buffer));
1877         } else {
1878           log_verbose("pass: #%d: %s + [%s=%s] -> %s\n", i, kwSetTestCases[i].l, kwSetTestCases[i].k, kwSetTestCases[i].v,buffer);
1879         }
1880     }
1881 }
1882 
TestKeywordSetError(void)1883 static void TestKeywordSetError(void)
1884 {
1885     char buffer[1024];
1886     UErrorCode status;
1887     int32_t res;
1888     int32_t i;
1889     int32_t blen;
1890 
1891     /* 0-test whether an error condition modifies the buffer at all */
1892     blen=0;
1893     i=0;
1894     memset(buffer,'%',1023);
1895     status = U_ZERO_ERROR;
1896     res = uloc_setKeywordValue(kwSetTestCases[i].k, kwSetTestCases[i].v, buffer, blen, &status);
1897     if(status != U_ILLEGAL_ARGUMENT_ERROR) {
1898         log_err("expected illegal err got %s\n", u_errorName(status));
1899         return;
1900     }
1901     /*  if(res!=strlen(kwSetTestCases[i].x)) {
1902     log_err("expected result %d got %d\n", strlen(kwSetTestCases[i].x), res);
1903     return;
1904     } */
1905     if(buffer[blen]!='%') {
1906         log_err("Buffer byte %d was modified: now %c\n", blen, buffer[blen]);
1907         return;
1908     }
1909     log_verbose("0-buffer modify OK\n");
1910 
1911     for(i=0;i<=2;i++) {
1912         /* 1- test a short buffer with growing text */
1913         blen=(int32_t)strlen(kwSetTestCases[i].l)+1;
1914         memset(buffer,'%',1023);
1915         strcpy(buffer,kwSetTestCases[i].l);
1916         status = U_ZERO_ERROR;
1917         res = uloc_setKeywordValue(kwSetTestCases[i].k, kwSetTestCases[i].v, buffer, blen, &status);
1918         if(status != U_BUFFER_OVERFLOW_ERROR) {
1919             log_err("expected buffer overflow on buffer %d got %s, len %d (%s + [%s=%s])\n", blen, u_errorName(status), res, kwSetTestCases[i].l, kwSetTestCases[i].k, kwSetTestCases[i].v);
1920             return;
1921         }
1922         if(res!=(int32_t)strlen(kwSetTestCases[i].x)) {
1923             log_err("expected result %d got %d\n", strlen(kwSetTestCases[i].x), res);
1924             return;
1925         }
1926         if(buffer[blen]!='%') {
1927             log_err("Buffer byte %d was modified: now %c\n", blen, buffer[blen]);
1928             return;
1929         }
1930         log_verbose("1/%d-buffer modify OK\n",i);
1931     }
1932 
1933     for(i=3;i<=4;i++) {
1934         /* 2- test a short buffer - text the same size or shrinking   */
1935         blen=(int32_t)strlen(kwSetTestCases[i].l)+1;
1936         memset(buffer,'%',1023);
1937         strcpy(buffer,kwSetTestCases[i].l);
1938         status = U_ZERO_ERROR;
1939         res = uloc_setKeywordValue(kwSetTestCases[i].k, kwSetTestCases[i].v, buffer, blen, &status);
1940         if(status != U_ZERO_ERROR) {
1941             log_err("expected zero error got %s\n", u_errorName(status));
1942             return;
1943         }
1944         if(buffer[blen+1]!='%') {
1945             log_err("Buffer byte %d was modified: now %c\n", blen+1, buffer[blen+1]);
1946             return;
1947         }
1948         if(res!=(int32_t)strlen(kwSetTestCases[i].x)) {
1949             log_err("expected result %d got %d\n", strlen(kwSetTestCases[i].x), res);
1950             return;
1951         }
1952         if(strcmp(buffer,kwSetTestCases[i].x) || ((int32_t)strlen(buffer)!=res)) {
1953             log_err("FAIL: #%d: %s + [%s=%s] -> %s (%d) expected %s (%d)\n", i, kwSetTestCases[i].l, kwSetTestCases[i].k,
1954                 kwSetTestCases[i].v, buffer, res, kwSetTestCases[i].x, strlen(buffer));
1955         } else {
1956             log_verbose("pass: #%d: %s + [%s=%s] -> %s\n", i, kwSetTestCases[i].l, kwSetTestCases[i].k, kwSetTestCases[i].v,
1957                 buffer);
1958         }
1959         log_verbose("2/%d-buffer modify OK\n",i);
1960     }
1961 }
1962 
_canonicalize(int32_t selector,const char * localeID,char * result,int32_t resultCapacity,UErrorCode * ec)1963 static int32_t _canonicalize(int32_t selector, /* 0==getName, 1==canonicalize */
1964                              const char* localeID,
1965                              char* result,
1966                              int32_t resultCapacity,
1967                              UErrorCode* ec) {
1968     /* YOU can change this to use function pointers if you like */
1969     switch (selector) {
1970     case 0:
1971         return uloc_getName(localeID, result, resultCapacity, ec);
1972     case 1:
1973         return uloc_canonicalize(localeID, result, resultCapacity, ec);
1974     default:
1975         return -1;
1976     }
1977 }
1978 
TestCanonicalization(void)1979 static void TestCanonicalization(void)
1980 {
1981     static const struct {
1982         const char *localeID;    /* input */
1983         const char *getNameID;   /* expected getName() result */
1984         const char *canonicalID; /* expected canonicalize() result */
1985     } testCases[] = {
1986         { "ca_ES_PREEURO-with-extra-stuff-that really doesn't make any sense-unless-you're trying to increase code coverage",
1987           "ca_ES_PREEURO_WITH_EXTRA_STUFF_THAT REALLY DOESN'T MAKE ANY SENSE_UNLESS_YOU'RE TRYING TO INCREASE CODE COVERAGE",
1988           "ca_ES_PREEURO_WITH_EXTRA_STUFF_THAT REALLY DOESN'T MAKE ANY SENSE_UNLESS_YOU'RE TRYING TO INCREASE CODE COVERAGE"},
1989         { "ca_ES_PREEURO", "ca_ES_PREEURO", "ca_ES@currency=ESP" },
1990         { "de_AT_PREEURO", "de_AT_PREEURO", "de_AT@currency=ATS" },
1991         { "de_DE_PREEURO", "de_DE_PREEURO", "de_DE@currency=DEM" },
1992         { "de_LU_PREEURO", "de_LU_PREEURO", "de_LU@currency=LUF" },
1993         { "el_GR_PREEURO", "el_GR_PREEURO", "el_GR@currency=GRD" },
1994         { "en_BE_PREEURO", "en_BE_PREEURO", "en_BE@currency=BEF" },
1995         { "en_IE_PREEURO", "en_IE_PREEURO", "en_IE@currency=IEP" },
1996         { "es_ES_PREEURO", "es_ES_PREEURO", "es_ES@currency=ESP" },
1997         { "eu_ES_PREEURO", "eu_ES_PREEURO", "eu_ES@currency=ESP" },
1998         { "fi_FI_PREEURO", "fi_FI_PREEURO", "fi_FI@currency=FIM" },
1999         { "fr_BE_PREEURO", "fr_BE_PREEURO", "fr_BE@currency=BEF" },
2000         { "fr_FR_PREEURO", "fr_FR_PREEURO", "fr_FR@currency=FRF" },
2001         { "fr_LU_PREEURO", "fr_LU_PREEURO", "fr_LU@currency=LUF" },
2002         { "ga_IE_PREEURO", "ga_IE_PREEURO", "ga_IE@currency=IEP" },
2003         { "gl_ES_PREEURO", "gl_ES_PREEURO", "gl_ES@currency=ESP" },
2004         { "it_IT_PREEURO", "it_IT_PREEURO", "it_IT@currency=ITL" },
2005         { "nl_BE_PREEURO", "nl_BE_PREEURO", "nl_BE@currency=BEF" },
2006         { "nl_NL_PREEURO", "nl_NL_PREEURO", "nl_NL@currency=NLG" },
2007         { "pt_PT_PREEURO", "pt_PT_PREEURO", "pt_PT@currency=PTE" },
2008         { "de__PHONEBOOK", "de__PHONEBOOK", "de@collation=phonebook" },
2009         { "en_GB_EURO", "en_GB_EURO", "en_GB@currency=EUR" },
2010         { "en_GB@EURO", "en_GB@EURO", "en_GB@currency=EUR" }, /* POSIX ID */
2011         { "es__TRADITIONAL", "es__TRADITIONAL", "es@collation=traditional" },
2012         { "hi__DIRECT", "hi__DIRECT", "hi@collation=direct" },
2013         { "ja_JP_TRADITIONAL", "ja_JP_TRADITIONAL", "ja_JP@calendar=japanese" },
2014         { "th_TH_TRADITIONAL", "th_TH_TRADITIONAL", "th_TH@calendar=buddhist" },
2015         { "zh_TW_STROKE", "zh_TW_STROKE", "zh_TW@collation=stroke" },
2016         { "zh__PINYIN", "zh__PINYIN", "zh@collation=pinyin" },
2017         { "zh@collation=pinyin", "zh@collation=pinyin", "zh@collation=pinyin" },
2018         { "zh_CN@collation=pinyin", "zh_CN@collation=pinyin", "zh_CN@collation=pinyin" },
2019         { "zh_CN_STROKE", "zh_CN_STROKE", "zh_CN@collation=stroke" },
2020         { "zh_CN_CA@collation=pinyin", "zh_CN_CA@collation=pinyin", "zh_CN_CA@collation=pinyin" },
2021         { "en_US_POSIX", "en_US_POSIX", "en_US_POSIX" },
2022         { "hy_AM_REVISED", "hy_AM_REVISED", "hy_AM_REVISED" },
2023         { "no_NO_NY", "no_NO_NY", "no_NO_NY" /* not: "nn_NO" [alan ICU3.0] */ },
2024         { "no@ny", "no@ny", "no__NY" /* not: "nn" [alan ICU3.0] */ }, /* POSIX ID */
2025         { "no-no.utf32@B", "no_NO.utf32@B", "no_NO_B" /* not: "nb_NO_B" [alan ICU3.0] */ }, /* POSIX ID */
2026         { "qz-qz@Euro", "qz_QZ@Euro", "qz_QZ@currency=EUR" }, /* qz-qz uses private use iso codes */
2027         { "en-BOONT", "en__BOONT", "en__BOONT" }, /* registered name */
2028         { "de-1901", "de__1901", "de__1901" }, /* registered name */
2029         { "de-1906", "de__1906", "de__1906" }, /* registered name */
2030         { "sr-SP-Cyrl", "sr_SP_CYRL", "sr_Cyrl_RS" }, /* .NET name */
2031         { "sr-SP-Latn", "sr_SP_LATN", "sr_Latn_RS" }, /* .NET name */
2032         { "sr_YU_CYRILLIC", "sr_YU_CYRILLIC", "sr_Cyrl_RS" }, /* Linux name */
2033         { "uz-UZ-Cyrl", "uz_UZ_CYRL", "uz_Cyrl_UZ" }, /* .NET name */
2034         { "uz-UZ-Latn", "uz_UZ_LATN", "uz_Latn_UZ" }, /* .NET name */
2035         { "zh-CHS", "zh_CHS", "zh_Hans" }, /* .NET name */
2036         { "zh-CHT", "zh_CHT", "zh_Hant" }, /* .NET name This may change back to zh_Hant */
2037 
2038         /* posix behavior that used to be performed by getName */
2039         { "mr.utf8", "mr.utf8", "mr" },
2040         { "de-tv.koi8r", "de_TV.koi8r", "de_TV" },
2041         { "x-piglatin_ML.MBE", "x-piglatin_ML.MBE", "x-piglatin_ML" },
2042         { "i-cherokee_US.utf7", "i-cherokee_US.utf7", "i-cherokee_US" },
2043         { "x-filfli_MT_FILFLA.gb-18030", "x-filfli_MT_FILFLA.gb-18030", "x-filfli_MT_FILFLA" },
2044         { "no-no-ny.utf8@B", "no_NO_NY.utf8@B", "no_NO_NY_B" /* not: "nn_NO" [alan ICU3.0] */ }, /* @ ignored unless variant is empty */
2045 
2046         /* fleshing out canonicalization */
2047         /* trim space and sort keywords, ';' is separator so not present at end in canonical form */
2048         { "en_Hant_IL_VALLEY_GIRL@ currency = EUR; calendar = Japanese ;", "en_Hant_IL_VALLEY_GIRL@calendar=Japanese;currency=EUR", "en_Hant_IL_VALLEY_GIRL@calendar=Japanese;currency=EUR" },
2049         /* already-canonical ids are not changed */
2050         { "en_Hant_IL_VALLEY_GIRL@calendar=Japanese;currency=EUR", "en_Hant_IL_VALLEY_GIRL@calendar=Japanese;currency=EUR", "en_Hant_IL_VALLEY_GIRL@calendar=Japanese;currency=EUR" },
2051         /* PRE_EURO and EURO conversions don't affect other keywords */
2052         { "es_ES_PREEURO@CALendar=Japanese", "es_ES_PREEURO@calendar=Japanese", "es_ES@calendar=Japanese;currency=ESP" },
2053         { "es_ES_EURO@SHOUT=zipeedeedoodah", "es_ES_EURO@shout=zipeedeedoodah", "es_ES@currency=EUR;shout=zipeedeedoodah" },
2054         /* currency keyword overrides PRE_EURO and EURO currency */
2055         { "es_ES_PREEURO@currency=EUR", "es_ES_PREEURO@currency=EUR", "es_ES@currency=EUR" },
2056         { "es_ES_EURO@currency=ESP", "es_ES_EURO@currency=ESP", "es_ES@currency=ESP" },
2057         /* norwegian is just too weird, if we handle things in their full generality */
2058         { "no-Hant-GB_NY@currency=$$$", "no_Hant_GB_NY@currency=$$$", "no_Hant_GB_NY@currency=$$$" /* not: "nn_Hant_GB@currency=$$$" [alan ICU3.0] */ },
2059 
2060         /* test cases reflecting internal resource bundle usage */
2061         { "root@kw=foo", "root@kw=foo", "root@kw=foo" },
2062         { "@calendar=gregorian", "@calendar=gregorian", "@calendar=gregorian" },
2063         { "ja_JP@calendar=Japanese", "ja_JP@calendar=Japanese", "ja_JP@calendar=Japanese" },
2064         { "ja_JP", "ja_JP", "ja_JP" },
2065 
2066         /* test case for "i-default" */
2067         { "i-default", "en@x=i-default", "en@x=i-default" }
2068     };
2069 
2070     static const char* label[] = { "getName", "canonicalize" };
2071 
2072     UErrorCode status = U_ZERO_ERROR;
2073     int32_t i, j, resultLen = 0, origResultLen;
2074     char buffer[256];
2075 
2076     for (i=0; i < sizeof(testCases)/sizeof(testCases[0]); i++) {
2077         for (j=0; j<2; ++j) {
2078             const char* expected = (j==0) ? testCases[i].getNameID : testCases[i].canonicalID;
2079             *buffer = 0;
2080             status = U_ZERO_ERROR;
2081 
2082             if (expected == NULL) {
2083                 expected = uloc_getDefault();
2084             }
2085 
2086             /* log_verbose("testing %s -> %s\n", testCases[i], testCases[i].canonicalID); */
2087             origResultLen = _canonicalize(j, testCases[i].localeID, NULL, 0, &status);
2088             if (status != U_BUFFER_OVERFLOW_ERROR) {
2089                 log_err("FAIL: uloc_%s(%s) => %s, expected U_BUFFER_OVERFLOW_ERROR\n",
2090                         label[j], testCases[i].localeID, u_errorName(status));
2091                 continue;
2092             }
2093             status = U_ZERO_ERROR;
2094             resultLen = _canonicalize(j, testCases[i].localeID, buffer, sizeof(buffer), &status);
2095             if (U_FAILURE(status)) {
2096                 log_err("FAIL: uloc_%s(%s) => %s, expected U_ZERO_ERROR\n",
2097                         label[j], testCases[i].localeID, u_errorName(status));
2098                 continue;
2099             }
2100             if(uprv_strcmp(expected, buffer) != 0) {
2101                 log_err("FAIL: uloc_%s(%s) => \"%s\", expected \"%s\"\n",
2102                         label[j], testCases[i].localeID, buffer, expected);
2103             } else {
2104                 log_verbose("Ok: uloc_%s(%s) => \"%s\"\n",
2105                             label[j], testCases[i].localeID, buffer);
2106             }
2107             if (resultLen != (int32_t)strlen(buffer)) {
2108                 log_err("FAIL: uloc_%s(%s) => len %d, expected len %d\n",
2109                         label[j], testCases[i].localeID, resultLen, strlen(buffer));
2110             }
2111             if (origResultLen != resultLen) {
2112                 log_err("FAIL: uloc_%s(%s) => preflight len %d != actual len %d\n",
2113                         label[j], testCases[i].localeID, origResultLen, resultLen);
2114             }
2115         }
2116     }
2117 }
2118 
TestDisplayKeywords(void)2119 static void TestDisplayKeywords(void)
2120 {
2121     int32_t i;
2122 
2123     static const struct {
2124         const char *localeID;
2125         const char *displayLocale;
2126         UChar displayKeyword[200];
2127     } testCases[] = {
2128         {   "ca_ES@currency=ESP",         "de_AT",
2129             {0x0057, 0x00e4, 0x0068, 0x0072, 0x0075, 0x006e, 0x0067, 0x0000},
2130         },
2131         {   "ja_JP@calendar=japanese",         "de",
2132             { 0x004b, 0x0061, 0x006c, 0x0065, 0x006e, 0x0064, 0x0065, 0x0072, 0x0000}
2133         },
2134         {   "de_DE@collation=traditional",       "de_DE",
2135             {0x0053, 0x006f, 0x0072, 0x0074, 0x0069, 0x0065, 0x0072, 0x0075, 0x006e, 0x0067, 0x0000}
2136         },
2137     };
2138     for(i = 0; i < sizeof(testCases)/sizeof(testCases[0]); i++) {
2139         UErrorCode status = U_ZERO_ERROR;
2140         const char* keyword =NULL;
2141         int32_t keywordLen = 0;
2142         int32_t keywordCount = 0;
2143         UChar *displayKeyword=NULL;
2144         int32_t displayKeywordLen = 0;
2145         UEnumeration* keywordEnum = uloc_openKeywords(testCases[i].localeID, &status);
2146         for(keywordCount = uenum_count(keywordEnum, &status); keywordCount > 0 ; keywordCount--){
2147               if(U_FAILURE(status)){
2148                   log_err("uloc_getKeywords failed for locale id: %s with error : %s \n", testCases[i].localeID, u_errorName(status));
2149                   break;
2150               }
2151               /* the uenum_next returns NUL terminated string */
2152               keyword = uenum_next(keywordEnum, &keywordLen, &status);
2153               /* fetch the displayKeyword */
2154               displayKeywordLen = uloc_getDisplayKeyword(keyword, testCases[i].displayLocale, displayKeyword, displayKeywordLen, &status);
2155               if(status==U_BUFFER_OVERFLOW_ERROR){
2156                   status = U_ZERO_ERROR;
2157                   displayKeywordLen++; /* for null termination */
2158                   displayKeyword = (UChar*) malloc(displayKeywordLen * U_SIZEOF_UCHAR);
2159                   displayKeywordLen = uloc_getDisplayKeyword(keyword, testCases[i].displayLocale, displayKeyword, displayKeywordLen, &status);
2160                   if(U_FAILURE(status)){
2161                       log_err("uloc_getDisplayKeyword filed for keyword : %s in locale id: %s for display locale: %s \n", testCases[i].localeID, keyword, testCases[i].displayLocale, u_errorName(status));
2162                       break;
2163                   }
2164                   if(u_strncmp(displayKeyword, testCases[i].displayKeyword, displayKeywordLen)!=0){
2165                       if (status == U_USING_DEFAULT_WARNING) {
2166                           log_data_err("uloc_getDisplayKeyword did not get the expected value for keyword : %s in locale id: %s for display locale: %s . Got error: %s. Perhaps you are missing data?\n", testCases[i].localeID, keyword, testCases[i].displayLocale, u_errorName(status));
2167                       } else {
2168                           log_err("uloc_getDisplayKeyword did not get the expected value for keyword : %s in locale id: %s for display locale: %s \n", testCases[i].localeID, keyword, testCases[i].displayLocale);
2169                       }
2170                       break;
2171                   }
2172               }else{
2173                   log_err("uloc_getDisplayKeyword did not return the expected error. Error: %s\n", u_errorName(status));
2174               }
2175 
2176               free(displayKeyword);
2177 
2178         }
2179         uenum_close(keywordEnum);
2180     }
2181 }
2182 
TestDisplayKeywordValues(void)2183 static void TestDisplayKeywordValues(void){
2184     int32_t i;
2185 
2186     static const struct {
2187         const char *localeID;
2188         const char *displayLocale;
2189         UChar displayKeywordValue[500];
2190     } testCases[] = {
2191         {   "ca_ES@currency=ESP",         "de_AT",
2192             {0x0053, 0x0070, 0x0061, 0x006e, 0x0069, 0x0073, 0x0063, 0x0068, 0x0065, 0x0020, 0x0050, 0x0065, 0x0073, 0x0065, 0x0074, 0x0061, 0x0000}
2193         },
2194         {   "de_AT@currency=ATS",         "fr_FR",
2195             {0x0073, 0x0063, 0x0068, 0x0069, 0x006c, 0x006c, 0x0069, 0x006e, 0x0067, 0x0020, 0x0061, 0x0075, 0x0074, 0x0072, 0x0069, 0x0063, 0x0068, 0x0069, 0x0065, 0x006e, 0x0000}
2196         },
2197         { "de_DE@currency=DEM",         "it",
2198             {0x004d, 0x0061, 0x0072, 0x0063, 0x006f, 0x0020, 0x0054, 0x0065, 0x0064, 0x0065, 0x0073, 0x0063, 0x006f, 0x0000}
2199         },
2200         {   "el_GR@currency=GRD",         "en",
2201             {0x0047, 0x0072, 0x0065, 0x0065, 0x006b, 0x0020, 0x0044, 0x0072, 0x0061, 0x0063, 0x0068, 0x006d, 0x0061, 0x0000}
2202         },
2203         {   "eu_ES@currency=ESP",         "it_IT",
2204             {0x0050, 0x0065, 0x0073, 0x0065, 0x0074, 0x0061, 0x0020, 0x0053, 0x0070, 0x0061, 0x0067, 0x006e, 0x006f, 0x006c, 0x0061, 0x0000}
2205         },
2206         {   "de@collation=phonebook",     "es",
2207             {0x006F, 0x0072, 0x0064, 0x0065, 0x006E, 0x0020, 0x0064, 0x0065, 0x0020, 0x006C, 0x0069, 0x0073, 0x0074, 0x00ED, 0x006E, 0x0020, 0x0074, 0x0065, 0x006C, 0x0065, 0x0066, 0x00F3, 0x006E, 0x0069, 0x0063, 0x006F, 0x0000}
2208         },
2209 
2210         { "de_DE@collation=phonebook",  "es",
2211           {0x006F, 0x0072, 0x0064, 0x0065, 0x006E, 0x0020, 0x0064, 0x0065, 0x0020, 0x006C, 0x0069, 0x0073, 0x0074, 0x00ED, 0x006E, 0x0020, 0x0074, 0x0065, 0x006C, 0x0065, 0x0066, 0x00F3, 0x006E, 0x0069, 0x0063, 0x006F, 0x0000}
2212         },
2213         { "es_ES@collation=traditional","de",
2214           {0x0054, 0x0072, 0x0061, 0x0064, 0x0069, 0x0074, 0x0069, 0x006f, 0x006e, 0x0065, 0x006c, 0x006c, 0x0065, 0x0020, 0x0053, 0x006f, 0x0072, 0x0074, 0x0069, 0x0065, 0x0072, 0x0072, 0x0065, 0x0067, 0x0065, 0x006c, 0x006e, 0x0000}
2215         },
2216         { "ja_JP@calendar=japanese",    "de",
2217            {0x004a, 0x0061, 0x0070, 0x0061, 0x006e, 0x0069, 0x0073, 0x0063, 0x0068, 0x0065, 0x0072, 0x0020, 0x004b, 0x0061, 0x006c, 0x0065, 0x006e, 0x0064, 0x0065, 0x0072, 0x0000}
2218         },
2219     };
2220     for(i = 0; i < sizeof(testCases)/sizeof(testCases[0]); i++) {
2221         UErrorCode status = U_ZERO_ERROR;
2222         const char* keyword =NULL;
2223         int32_t keywordLen = 0;
2224         int32_t keywordCount = 0;
2225         UChar *displayKeywordValue = NULL;
2226         int32_t displayKeywordValueLen = 0;
2227         UEnumeration* keywordEnum = uloc_openKeywords(testCases[i].localeID, &status);
2228         for(keywordCount = uenum_count(keywordEnum, &status); keywordCount > 0 ; keywordCount--){
2229               if(U_FAILURE(status)){
2230                   log_err("uloc_getKeywords failed for locale id: %s in display locale: % with error : %s \n", testCases[i].localeID, testCases[i].displayLocale, u_errorName(status));
2231                   break;
2232               }
2233               /* the uenum_next returns NUL terminated string */
2234               keyword = uenum_next(keywordEnum, &keywordLen, &status);
2235 
2236               /* fetch the displayKeywordValue */
2237               displayKeywordValueLen = uloc_getDisplayKeywordValue(testCases[i].localeID, keyword, testCases[i].displayLocale, displayKeywordValue, displayKeywordValueLen, &status);
2238               if(status==U_BUFFER_OVERFLOW_ERROR){
2239                   status = U_ZERO_ERROR;
2240                   displayKeywordValueLen++; /* for null termination */
2241                   displayKeywordValue = (UChar*)malloc(displayKeywordValueLen * U_SIZEOF_UCHAR);
2242                   displayKeywordValueLen = uloc_getDisplayKeywordValue(testCases[i].localeID, keyword, testCases[i].displayLocale, displayKeywordValue, displayKeywordValueLen, &status);
2243                   if(U_FAILURE(status)){
2244                       log_err("uloc_getDisplayKeywordValue failed for keyword : %s in locale id: %s for display locale: %s with error : %s \n", testCases[i].localeID, keyword, testCases[i].displayLocale, u_errorName(status));
2245                       break;
2246                   }
2247                   if(u_strncmp(displayKeywordValue, testCases[i].displayKeywordValue, displayKeywordValueLen)!=0){
2248                       if (status == U_USING_DEFAULT_WARNING) {
2249                           log_data_err("uloc_getDisplayKeywordValue did not return the expected value keyword : %s in locale id: %s for display locale: %s with error : %s Perhaps you are missing data\n", testCases[i].localeID, keyword, testCases[i].displayLocale, u_errorName(status));
2250                       } else {
2251                           log_err("uloc_getDisplayKeywordValue did not return the expected value keyword : %s in locale id: %s for display locale: %s with error : %s \n", testCases[i].localeID, keyword, testCases[i].displayLocale, u_errorName(status));
2252                       }
2253                       break;
2254                   }
2255               }else{
2256                   log_err("uloc_getDisplayKeywordValue did not return the expected error. Error: %s\n", u_errorName(status));
2257               }
2258               free(displayKeywordValue);
2259         }
2260         uenum_close(keywordEnum);
2261     }
2262     {
2263         /* test a multiple keywords */
2264         UErrorCode status = U_ZERO_ERROR;
2265         const char* keyword =NULL;
2266         int32_t keywordLen = 0;
2267         int32_t keywordCount = 0;
2268         const char* localeID = "es@collation=phonebook;calendar=buddhist;currency=DEM";
2269         const char* displayLocale = "de";
2270         static const UChar expected[][50] = {
2271             {0x0042, 0x0075, 0x0064, 0x0064, 0x0068, 0x0069, 0x0073, 0x0074, 0x0069, 0x0073, 0x0063, 0x0068, 0x0065, 0x0072, 0x0020, 0x004b, 0x0061, 0x006c, 0x0065, 0x006e, 0x0064, 0x0065, 0x0072, 0x0000},
2272 
2273             {0x0054, 0x0065, 0x006c, 0x0065, 0x0066, 0x006f, 0x006e, 0x0062, 0x0075, 0x0063, 0x0068, 0x002d, 0x0053, 0x006f, 0x0072, 0x0074, 0x0069, 0x0065, 0x0072, 0x0075, 0x006e, 0x0067, 0x0000},
2274             {0x0044, 0x0065, 0x0075, 0x0074, 0x0073, 0x0063, 0x0068, 0x0065, 0x0020, 0x004d, 0x0061, 0x0072, 0x006b, 0x0000},
2275         };
2276 
2277         UEnumeration* keywordEnum = uloc_openKeywords(localeID, &status);
2278 
2279         for(keywordCount = 0; keywordCount < uenum_count(keywordEnum, &status) ; keywordCount++){
2280               UChar *displayKeywordValue = NULL;
2281               int32_t displayKeywordValueLen = 0;
2282               if(U_FAILURE(status)){
2283                   log_err("uloc_getKeywords failed for locale id: %s in display locale: % with error : %s \n", localeID, displayLocale, u_errorName(status));
2284                   break;
2285               }
2286               /* the uenum_next returns NUL terminated string */
2287               keyword = uenum_next(keywordEnum, &keywordLen, &status);
2288 
2289               /* fetch the displayKeywordValue */
2290               displayKeywordValueLen = uloc_getDisplayKeywordValue(localeID, keyword, displayLocale, displayKeywordValue, displayKeywordValueLen, &status);
2291               if(status==U_BUFFER_OVERFLOW_ERROR){
2292                   status = U_ZERO_ERROR;
2293                   displayKeywordValueLen++; /* for null termination */
2294                   displayKeywordValue = (UChar*)malloc(displayKeywordValueLen * U_SIZEOF_UCHAR);
2295                   displayKeywordValueLen = uloc_getDisplayKeywordValue(localeID, keyword, displayLocale, displayKeywordValue, displayKeywordValueLen, &status);
2296                   if(U_FAILURE(status)){
2297                       log_err("uloc_getDisplayKeywordValue failed for keyword : %s in locale id: %s for display locale: %s with error : %s \n", localeID, keyword, displayLocale, u_errorName(status));
2298                       break;
2299                   }
2300                   if(u_strncmp(displayKeywordValue, expected[keywordCount], displayKeywordValueLen)!=0){
2301                       if (status == U_USING_DEFAULT_WARNING) {
2302                           log_data_err("uloc_getDisplayKeywordValue did not return the expected value keyword : %s in locale id: %s for display locale: %s  got error: %s. Perhaps you are missing data?\n", localeID, keyword, displayLocale, u_errorName(status));
2303                       } else {
2304                           log_err("uloc_getDisplayKeywordValue did not return the expected value keyword : %s in locale id: %s for display locale: %s \n", localeID, keyword, displayLocale);
2305                       }
2306                       break;
2307                   }
2308               }else{
2309                   log_err("uloc_getDisplayKeywordValue did not return the expected error. Error: %s\n", u_errorName(status));
2310               }
2311               free(displayKeywordValue);
2312         }
2313         uenum_close(keywordEnum);
2314 
2315     }
2316     {
2317         /* Test non existent keywords */
2318         UErrorCode status = U_ZERO_ERROR;
2319         const char* localeID = "es";
2320         const char* displayLocale = "de";
2321         UChar *displayKeywordValue = NULL;
2322         int32_t displayKeywordValueLen = 0;
2323 
2324         /* fetch the displayKeywordValue */
2325         displayKeywordValueLen = uloc_getDisplayKeywordValue(localeID, "calendar", displayLocale, displayKeywordValue, displayKeywordValueLen, &status);
2326         if(U_FAILURE(status)) {
2327           log_err("uloc_getDisplaykeywordValue returned error status %s\n", u_errorName(status));
2328         } else if(displayKeywordValueLen != 0) {
2329           log_err("uloc_getDisplaykeywordValue returned %d should be 0 \n", displayKeywordValueLen);
2330         }
2331     }
2332 }
2333 
2334 
TestGetBaseName(void)2335 static void TestGetBaseName(void) {
2336     static const struct {
2337         const char *localeID;
2338         const char *baseName;
2339     } testCases[] = {
2340         { "de_DE@  C o ll A t i o n   = Phonebook   ", "de_DE" },
2341         { "de@currency = euro; CoLLaTion   = PHONEBOOk", "de" },
2342         { "ja@calendar = buddhist", "ja" }
2343     };
2344 
2345     int32_t i = 0, baseNameLen = 0;
2346     char baseName[256];
2347     UErrorCode status = U_ZERO_ERROR;
2348 
2349     for(i = 0; i < sizeof(testCases)/sizeof(testCases[0]); i++) {
2350         baseNameLen = uloc_getBaseName(testCases[i].localeID, baseName, 256, &status);
2351         (void)baseNameLen;    /* Suppress set but not used warning. */
2352         if(strcmp(testCases[i].baseName, baseName)) {
2353             log_err("For locale \"%s\" expected baseName \"%s\", but got \"%s\"\n",
2354                 testCases[i].localeID, testCases[i].baseName, baseName);
2355             return;
2356         }
2357     }
2358 }
2359 
TestTrailingNull(void)2360 static void TestTrailingNull(void) {
2361   const char* localeId = "zh_Hans";
2362   UChar buffer[128]; /* sufficient for this test */
2363   int32_t len;
2364   UErrorCode status = U_ZERO_ERROR;
2365   int i;
2366 
2367   len = uloc_getDisplayName(localeId, localeId, buffer, 128, &status);
2368   if (len > 128) {
2369     log_err("buffer too small");
2370     return;
2371   }
2372 
2373   for (i = 0; i < len; ++i) {
2374     if (buffer[i] == 0) {
2375       log_err("name contained null");
2376       return;
2377     }
2378   }
2379 }
2380 
2381 /* Jitterbug 4115 */
TestDisplayNameWarning(void)2382 static void TestDisplayNameWarning(void) {
2383     UChar name[256];
2384     int32_t size;
2385     UErrorCode status = U_ZERO_ERROR;
2386 
2387     size = uloc_getDisplayLanguage("qqq", "kl", name, sizeof(name)/sizeof(name[0]), &status);
2388     (void)size;    /* Suppress set but not used warning. */
2389     if (status != U_USING_DEFAULT_WARNING) {
2390         log_err("For language \"qqq\" in locale \"kl\", expecting U_USING_DEFAULT_WARNING, but got %s\n",
2391             u_errorName(status));
2392     }
2393 }
2394 
2395 
2396 /**
2397  * Compare two locale IDs.  If they are equal, return 0.  If `string'
2398  * starts with `prefix' plus an additional element, that is, string ==
2399  * prefix + '_' + x, then return 1.  Otherwise return a value < 0.
2400  */
_loccmp(const char * string,const char * prefix)2401 static UBool _loccmp(const char* string, const char* prefix) {
2402     int32_t slen = (int32_t)uprv_strlen(string),
2403             plen = (int32_t)uprv_strlen(prefix);
2404     int32_t c = uprv_strncmp(string, prefix, plen);
2405     /* 'root' is less than everything */
2406     if (uprv_strcmp(prefix, "root") == 0) {
2407         return (uprv_strcmp(string, "root") == 0) ? 0 : 1;
2408     }
2409     if (c) return -1; /* mismatch */
2410     if (slen == plen) return 0;
2411     if (string[plen] == '_') return 1;
2412     return -2; /* false match, e.g. "en_USX" cmp "en_US" */
2413 }
2414 
_checklocs(const char * label,const char * req,const char * valid,const char * actual)2415 static void _checklocs(const char* label,
2416                        const char* req,
2417                        const char* valid,
2418                        const char* actual) {
2419     /* We want the valid to be strictly > the bogus requested locale,
2420        and the valid to be >= the actual. */
2421     if (_loccmp(req, valid) > 0 &&
2422         _loccmp(valid, actual) >= 0) {
2423         log_verbose("%s; req=%s, valid=%s, actual=%s\n",
2424                     label, req, valid, actual);
2425     } else {
2426         log_err("FAIL: %s; req=%s, valid=%s, actual=%s\n",
2427                 label, req, valid, actual);
2428     }
2429 }
2430 
TestGetLocale(void)2431 static void TestGetLocale(void) {
2432     UErrorCode ec = U_ZERO_ERROR;
2433     UParseError pe;
2434     UChar EMPTY[1] = {0};
2435 
2436     /* === udat === */
2437 #if !UCONFIG_NO_FORMATTING
2438     {
2439         UDateFormat *obj;
2440         const char *req = "en_US_REDWOODSHORES", *valid, *actual;
2441         obj = udat_open(UDAT_DEFAULT, UDAT_DEFAULT,
2442                         req,
2443                         NULL, 0,
2444                         NULL, 0, &ec);
2445         if (U_FAILURE(ec)) {
2446             log_data_err("udat_open failed.Error %s\n", u_errorName(ec));
2447             return;
2448         }
2449         valid = udat_getLocaleByType(obj, ULOC_VALID_LOCALE, &ec);
2450         actual = udat_getLocaleByType(obj, ULOC_ACTUAL_LOCALE, &ec);
2451         if (U_FAILURE(ec)) {
2452             log_err("udat_getLocaleByType() failed\n");
2453             return;
2454         }
2455         _checklocs("udat", req, valid, actual);
2456         udat_close(obj);
2457     }
2458 #endif
2459 
2460     /* === ucal === */
2461 #if !UCONFIG_NO_FORMATTING
2462     {
2463         UCalendar *obj;
2464         const char *req = "fr_FR_PROVENCAL", *valid, *actual;
2465         obj = ucal_open(NULL, 0,
2466                         req,
2467                         UCAL_GREGORIAN,
2468                         &ec);
2469         if (U_FAILURE(ec)) {
2470             log_err("ucal_open failed with error: %s\n", u_errorName(ec));
2471             return;
2472         }
2473         valid = ucal_getLocaleByType(obj, ULOC_VALID_LOCALE, &ec);
2474         actual = ucal_getLocaleByType(obj, ULOC_ACTUAL_LOCALE, &ec);
2475         if (U_FAILURE(ec)) {
2476             log_err("ucal_getLocaleByType() failed\n");
2477             return;
2478         }
2479         _checklocs("ucal", req, valid, actual);
2480         ucal_close(obj);
2481     }
2482 #endif
2483 
2484     /* === unum === */
2485 #if !UCONFIG_NO_FORMATTING
2486     {
2487         UNumberFormat *obj;
2488         const char *req = "zh_Hant_TW_TAINAN", *valid, *actual;
2489         obj = unum_open(UNUM_DECIMAL,
2490                         NULL, 0,
2491                         req,
2492                         &pe, &ec);
2493         if (U_FAILURE(ec)) {
2494             log_err("unum_open failed\n");
2495             return;
2496         }
2497         valid = unum_getLocaleByType(obj, ULOC_VALID_LOCALE, &ec);
2498         actual = unum_getLocaleByType(obj, ULOC_ACTUAL_LOCALE, &ec);
2499         if (U_FAILURE(ec)) {
2500             log_err("unum_getLocaleByType() failed\n");
2501             return;
2502         }
2503         _checklocs("unum", req, valid, actual);
2504         unum_close(obj);
2505     }
2506 #endif
2507 
2508     /* === umsg === */
2509 #if 0
2510     /* commented out by weiv 01/12/2005. umsg_getLocaleByType is to be removed */
2511 #if !UCONFIG_NO_FORMATTING
2512     {
2513         UMessageFormat *obj;
2514         const char *req = "ja_JP_TAKAYAMA", *valid, *actual;
2515         UBool test;
2516         obj = umsg_open(EMPTY, 0,
2517                         req,
2518                         &pe, &ec);
2519         if (U_FAILURE(ec)) {
2520             log_err("umsg_open failed\n");
2521             return;
2522         }
2523         valid = umsg_getLocaleByType(obj, ULOC_VALID_LOCALE, &ec);
2524         actual = umsg_getLocaleByType(obj, ULOC_ACTUAL_LOCALE, &ec);
2525         if (U_FAILURE(ec)) {
2526             log_err("umsg_getLocaleByType() failed\n");
2527             return;
2528         }
2529         /* We want the valid to be strictly > the bogus requested locale,
2530            and the valid to be >= the actual. */
2531         /* TODO MessageFormat is currently just storing the locale it is given.
2532            As a result, it will return whatever it was given, even if the
2533            locale is invalid. */
2534         test = (_cmpversion("3.2") <= 0) ?
2535             /* Here is the weakened test for 3.0: */
2536             (_loccmp(req, valid) >= 0) :
2537             /* Here is what the test line SHOULD be: */
2538             (_loccmp(req, valid) > 0);
2539 
2540         if (test &&
2541             _loccmp(valid, actual) >= 0) {
2542             log_verbose("umsg; req=%s, valid=%s, actual=%s\n", req, valid, actual);
2543         } else {
2544             log_err("FAIL: umsg; req=%s, valid=%s, actual=%s\n", req, valid, actual);
2545         }
2546         umsg_close(obj);
2547     }
2548 #endif
2549 #endif
2550 
2551     /* === ubrk === */
2552 #if !UCONFIG_NO_BREAK_ITERATION
2553     {
2554         UBreakIterator *obj;
2555         const char *req = "ar_KW_ABDALI", *valid, *actual;
2556         obj = ubrk_open(UBRK_WORD,
2557                         req,
2558                         EMPTY,
2559                         0,
2560                         &ec);
2561         if (U_FAILURE(ec)) {
2562             log_err("ubrk_open failed. Error: %s \n", u_errorName(ec));
2563             return;
2564         }
2565         valid = ubrk_getLocaleByType(obj, ULOC_VALID_LOCALE, &ec);
2566         actual = ubrk_getLocaleByType(obj, ULOC_ACTUAL_LOCALE, &ec);
2567         if (U_FAILURE(ec)) {
2568             log_err("ubrk_getLocaleByType() failed\n");
2569             return;
2570         }
2571         _checklocs("ubrk", req, valid, actual);
2572         ubrk_close(obj);
2573     }
2574 #endif
2575 
2576     /* === ucol === */
2577 #if !UCONFIG_NO_COLLATION
2578     {
2579         UCollator *obj;
2580         const char *req = "es_AR_BUENOSAIRES", *valid, *actual;
2581         obj = ucol_open(req, &ec);
2582         if (U_FAILURE(ec)) {
2583             log_err("ucol_open failed - %s\n", u_errorName(ec));
2584             return;
2585         }
2586         valid = ucol_getLocaleByType(obj, ULOC_VALID_LOCALE, &ec);
2587         actual = ucol_getLocaleByType(obj, ULOC_ACTUAL_LOCALE, &ec);
2588         if (U_FAILURE(ec)) {
2589             log_err("ucol_getLocaleByType() failed\n");
2590             return;
2591         }
2592         _checklocs("ucol", req, valid, actual);
2593         ucol_close(obj);
2594     }
2595 #endif
2596 }
TestEnglishExemplarCharacters(void)2597 static void TestEnglishExemplarCharacters(void) {
2598     UErrorCode status = U_ZERO_ERROR;
2599     int i;
2600     USet *exSet = NULL;
2601     UChar testChars[] = {
2602         0x61,   /* standard */
2603         0xE1,   /* auxiliary */
2604         0x41,   /* index */
2605         0x2D    /* punctuation */
2606     };
2607     ULocaleData *uld = ulocdata_open("en", &status);
2608     if (U_FAILURE(status)) {
2609         log_data_err("ulocdata_open() failed : %s - (Are you missing data?)\n", u_errorName(status));
2610         return;
2611     }
2612 
2613     for (i = 0; i < ULOCDATA_ES_COUNT; i++) {
2614         exSet = ulocdata_getExemplarSet(uld, exSet, 0, (ULocaleDataExemplarSetType)i, &status);
2615         if (U_FAILURE(status)) {
2616             log_err_status(status, "ulocdata_getExemplarSet() for type %d failed\n", i);
2617             status = U_ZERO_ERROR;
2618             continue;
2619         }
2620         if (!uset_contains(exSet, (UChar32)testChars[i])) {
2621             log_err("Character U+%04X is not included in exemplar type %d\n", testChars[i], i);
2622         }
2623     }
2624 
2625     uset_close(exSet);
2626     ulocdata_close(uld);
2627 }
2628 
TestNonexistentLanguageExemplars(void)2629 static void TestNonexistentLanguageExemplars(void) {
2630     /* JB 4068 - Nonexistent language */
2631     UErrorCode ec = U_ZERO_ERROR;
2632     ULocaleData *uld = ulocdata_open("qqq",&ec);
2633     if (ec != U_USING_DEFAULT_WARNING) {
2634         log_err_status(ec, "Exemplar set for \"qqq\", expecting U_USING_DEFAULT_WARNING, but got %s\n",
2635             u_errorName(ec));
2636     }
2637     uset_close(ulocdata_getExemplarSet(uld, NULL, 0, ULOCDATA_ES_STANDARD, &ec));
2638     ulocdata_close(uld);
2639 }
2640 
TestLocDataErrorCodeChaining(void)2641 static void TestLocDataErrorCodeChaining(void) {
2642     UErrorCode ec = U_USELESS_COLLATOR_ERROR;
2643     ulocdata_open(NULL, &ec);
2644     ulocdata_getExemplarSet(NULL, NULL, 0, ULOCDATA_ES_STANDARD, &ec);
2645     ulocdata_getDelimiter(NULL, ULOCDATA_DELIMITER_COUNT, NULL, -1, &ec);
2646     ulocdata_getMeasurementSystem(NULL, &ec);
2647     ulocdata_getPaperSize(NULL, NULL, NULL, &ec);
2648     if (ec != U_USELESS_COLLATOR_ERROR) {
2649         log_err("ulocdata API changed the error code to %s\n", u_errorName(ec));
2650     }
2651 }
2652 
TestLanguageExemplarsFallbacks(void)2653 static void TestLanguageExemplarsFallbacks(void) {
2654     /* Test that en_US fallsback, but en doesn't fallback. */
2655     UErrorCode ec = U_ZERO_ERROR;
2656     ULocaleData *uld = ulocdata_open("en_US",&ec);
2657     uset_close(ulocdata_getExemplarSet(uld, NULL, 0, ULOCDATA_ES_STANDARD, &ec));
2658     if (ec != U_USING_FALLBACK_WARNING) {
2659         log_err_status(ec, "Exemplar set for \"en_US\", expecting U_USING_FALLBACK_WARNING, but got %s\n",
2660             u_errorName(ec));
2661     }
2662     ulocdata_close(uld);
2663     ec = U_ZERO_ERROR;
2664     uld = ulocdata_open("en",&ec);
2665     uset_close(ulocdata_getExemplarSet(uld, NULL, 0, ULOCDATA_ES_STANDARD, &ec));
2666     if (ec != U_ZERO_ERROR) {
2667         log_err_status(ec, "Exemplar set for \"en\", expecting U_ZERO_ERROR, but got %s\n",
2668             u_errorName(ec));
2669     }
2670     ulocdata_close(uld);
2671 }
2672 
acceptResult(UAcceptResult uar)2673 static const char *acceptResult(UAcceptResult uar) {
2674     return  udbg_enumName(UDBG_UAcceptResult, uar);
2675 }
2676 
TestAcceptLanguage(void)2677 static void TestAcceptLanguage(void) {
2678     UErrorCode status = U_ZERO_ERROR;
2679     UAcceptResult outResult;
2680     UEnumeration *available;
2681     char tmp[200];
2682     int i;
2683     int32_t rc = 0;
2684 
2685     struct {
2686         int32_t httpSet;       /**< Which of http[] should be used? */
2687         const char *icuSet;    /**< ? */
2688         const char *expect;    /**< The expected locale result */
2689         UAcceptResult res;     /**< The expected error code */
2690     } tests[] = {
2691         /*0*/{ 0, NULL, "mt_MT", ULOC_ACCEPT_VALID },
2692         /*1*/{ 1, NULL, "en", ULOC_ACCEPT_VALID },
2693         /*2*/{ 2, NULL, "en", ULOC_ACCEPT_FALLBACK },
2694         /*3*/{ 3, NULL, "", ULOC_ACCEPT_FAILED },
2695         /*4*/{ 4, NULL, "es", ULOC_ACCEPT_VALID },
2696 
2697         /*5*/{ 5, NULL, "en", ULOC_ACCEPT_VALID },  /* XF */
2698         /*6*/{ 6, NULL, "ja", ULOC_ACCEPT_FALLBACK },  /* XF */
2699         /*7*/{ 7, NULL, "zh", ULOC_ACCEPT_FALLBACK },  /* XF */
2700     };
2701     const int32_t numTests = sizeof(tests)/sizeof(tests[0]);
2702     static const char *http[] = {
2703         /*0*/ "mt-mt, ja;q=0.76, en-us;q=0.95, en;q=0.92, en-gb;q=0.89, fr;q=0.87, iu-ca;q=0.84, iu;q=0.82, ja-jp;q=0.79, mt;q=0.97, de-de;q=0.74, de;q=0.71, es;q=0.68, it-it;q=0.66, it;q=0.63, vi-vn;q=0.61, vi;q=0.58, nl-nl;q=0.55, nl;q=0.53, th-th-traditional;q=.01",
2704         /*1*/ "ja;q=0.5, en;q=0.8, tlh",
2705         /*2*/ "en-wf, de-lx;q=0.8",
2706         /*3*/ "mga-ie;q=0.9, tlh",
2707         /*4*/ "xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, "
2708               "xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, "
2709               "xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, "
2710               "xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, "
2711               "xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, "
2712               "xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, "
2713               "xxx-yyy;q=.01, xxx-yyy;q=.01, xxx-yyy;q=.01, xx-yy;q=.1, "
2714               "es",
2715 
2716         /*5*/ "zh-xx;q=0.9, en;q=0.6",
2717         /*6*/ "ja-JA",
2718         /*7*/ "zh-xx;q=0.9",
2719     };
2720 
2721     for(i=0;i<numTests;i++) {
2722         outResult = -3;
2723         status=U_ZERO_ERROR;
2724         log_verbose("test #%d: http[%s], ICU[%s], expect %s, %s\n",
2725             i, http[tests[i].httpSet], tests[i].icuSet, tests[i].expect, acceptResult(tests[i].res));
2726 
2727         available = ures_openAvailableLocales(tests[i].icuSet, &status);
2728         tmp[0]=0;
2729         rc = uloc_acceptLanguageFromHTTP(tmp, 199, &outResult, http[tests[i].httpSet], available, &status);
2730         (void)rc;    /* Suppress set but not used warning. */
2731         uenum_close(available);
2732         log_verbose(" got %s, %s [%s]\n", tmp[0]?tmp:"(EMPTY)", acceptResult(outResult), u_errorName(status));
2733         if(outResult != tests[i].res) {
2734             log_err_status(status, "FAIL: #%d: expected outResult of %s but got %s\n", i,
2735                 acceptResult( tests[i].res),
2736                 acceptResult( outResult));
2737             log_info("test #%d: http[%s], ICU[%s], expect %s, %s\n",
2738                 i, http[tests[i].httpSet], tests[i].icuSet, tests[i].expect,acceptResult(tests[i].res));
2739         }
2740         if((outResult>0)&&uprv_strcmp(tmp, tests[i].expect)) {
2741             log_err_status(status, "FAIL: #%d: expected %s but got %s\n", i, tests[i].expect, tmp);
2742             log_info("test #%d: http[%s], ICU[%s], expect %s, %s\n",
2743                 i, http[tests[i].httpSet], tests[i].icuSet, tests[i].expect, acceptResult(tests[i].res));
2744         }
2745     }
2746 }
2747 
2748 static const char* LOCALE_ALIAS[][2] = {
2749     {"in", "id"},
2750     {"in_ID", "id_ID"},
2751     {"iw", "he"},
2752     {"iw_IL", "he_IL"},
2753     {"ji", "yi"},
2754     {"en_BU", "en_MM"},
2755     {"en_DY", "en_BJ"},
2756     {"en_HV", "en_BF"},
2757     {"en_NH", "en_VU"},
2758     {"en_RH", "en_ZW"},
2759     {"en_TP", "en_TL"},
2760     {"en_ZR", "en_CD"}
2761 };
isLocaleAvailable(UResourceBundle * resIndex,const char * loc)2762 static UBool isLocaleAvailable(UResourceBundle* resIndex, const char* loc){
2763     UErrorCode status = U_ZERO_ERROR;
2764     int32_t len = 0;
2765     ures_getStringByKey(resIndex, loc,&len, &status);
2766     if(U_FAILURE(status)){
2767         return FALSE;
2768     }
2769     return TRUE;
2770 }
2771 
TestCalendar()2772 static void TestCalendar() {
2773 #if !UCONFIG_NO_FORMATTING
2774     int i;
2775     UErrorCode status = U_ZERO_ERROR;
2776     UResourceBundle *resIndex = ures_open(NULL,"res_index", &status);
2777     if(U_FAILURE(status)){
2778         log_err_status(status, "Could not open res_index.res. Exiting. Error: %s\n", u_errorName(status));
2779         return;
2780     }
2781     for (i=0; i<LENGTHOF(LOCALE_ALIAS); i++) {
2782         const char* oldLoc = LOCALE_ALIAS[i][0];
2783         const char* newLoc = LOCALE_ALIAS[i][1];
2784         UCalendar* c1 = NULL;
2785         UCalendar* c2 = NULL;
2786 
2787         /*Test function "getLocale(ULocale.VALID_LOCALE)"*/
2788         const char* l1 = ucal_getLocaleByType(c1, ULOC_VALID_LOCALE, &status);
2789         const char* l2 = ucal_getLocaleByType(c2, ULOC_VALID_LOCALE, &status);
2790 
2791         if(!isLocaleAvailable(resIndex, newLoc)){
2792             continue;
2793         }
2794         c1 = ucal_open(NULL, -1, oldLoc, UCAL_GREGORIAN, &status);
2795         c2 = ucal_open(NULL, -1, newLoc, UCAL_GREGORIAN, &status);
2796 
2797         if (strcmp(newLoc,l1)!=0 || strcmp(l1,l2)!=0 || status!=U_ZERO_ERROR) {
2798             log_err("The locales are not equal!.Old: %s, New: %s \n", oldLoc, newLoc);
2799         }
2800         log_verbose("ucal_getLocaleByType old:%s   new:%s\n", l1, l2);
2801         ucal_close(c1);
2802         ucal_close(c2);
2803     }
2804     ures_close(resIndex);
2805 #endif
2806 }
2807 
TestDateFormat()2808 static void TestDateFormat() {
2809 #if !UCONFIG_NO_FORMATTING
2810     int i;
2811     UErrorCode status = U_ZERO_ERROR;
2812     UResourceBundle *resIndex = ures_open(NULL,"res_index", &status);
2813     if(U_FAILURE(status)){
2814         log_err_status(status, "Could not open res_index.res. Exiting. Error: %s\n", u_errorName(status));
2815         return;
2816     }
2817     for (i=0; i<LENGTHOF(LOCALE_ALIAS); i++) {
2818         const char* oldLoc = LOCALE_ALIAS[i][0];
2819         const char* newLoc = LOCALE_ALIAS[i][1];
2820         UDateFormat* df1 = NULL;
2821         UDateFormat* df2 = NULL;
2822         const char* l1 = NULL;
2823         const char* l2 = NULL;
2824 
2825         if(!isLocaleAvailable(resIndex, newLoc)){
2826             continue;
2827         }
2828         df1 = udat_open(UDAT_FULL, UDAT_FULL,oldLoc, NULL, 0, NULL, -1, &status);
2829         df2 = udat_open(UDAT_FULL, UDAT_FULL,newLoc, NULL, 0, NULL, -1, &status);
2830         if(U_FAILURE(status)){
2831             log_err("Creation of date format failed  %s\n", u_errorName(status));
2832             return;
2833         }
2834         /*Test function "getLocale"*/
2835         l1 = udat_getLocaleByType(df1, ULOC_VALID_LOCALE, &status);
2836         l2 = udat_getLocaleByType(df2, ULOC_VALID_LOCALE, &status);
2837         if(U_FAILURE(status)){
2838             log_err("Fetching the locale by type failed.  %s\n", u_errorName(status));
2839         }
2840         if (strcmp(newLoc,l1)!=0 || strcmp(l1,l2)!=0) {
2841             log_err("The locales are not equal!.Old: %s, New: %s \n", oldLoc, newLoc);
2842         }
2843         log_verbose("udat_getLocaleByType old:%s   new:%s\n", l1, l2);
2844         udat_close(df1);
2845         udat_close(df2);
2846     }
2847     ures_close(resIndex);
2848 #endif
2849 }
2850 
TestCollation()2851 static void TestCollation() {
2852 #if !UCONFIG_NO_COLLATION
2853     int i;
2854     UErrorCode status = U_ZERO_ERROR;
2855     UResourceBundle *resIndex = ures_open(NULL,"res_index", &status);
2856     if(U_FAILURE(status)){
2857         log_err_status(status, "Could not open res_index.res. Exiting. Error: %s\n", u_errorName(status));
2858         return;
2859     }
2860     for (i=0; i<LENGTHOF(LOCALE_ALIAS); i++) {
2861         const char* oldLoc = LOCALE_ALIAS[i][0];
2862         const char* newLoc = LOCALE_ALIAS[i][1];
2863         UCollator* c1 = NULL;
2864         UCollator* c2 = NULL;
2865         const char* l1 = NULL;
2866         const char* l2 = NULL;
2867 
2868         status = U_ZERO_ERROR;
2869         if(!isLocaleAvailable(resIndex, newLoc)){
2870             continue;
2871         }
2872         if(U_FAILURE(status)){
2873             log_err("Creation of collators failed  %s\n", u_errorName(status));
2874             return;
2875         }
2876         c1 = ucol_open(oldLoc, &status);
2877         c2 = ucol_open(newLoc, &status);
2878         l1 = ucol_getLocaleByType(c1, ULOC_VALID_LOCALE, &status);
2879         l2 = ucol_getLocaleByType(c2, ULOC_VALID_LOCALE, &status);
2880         if(U_FAILURE(status)){
2881             log_err("Fetching the locale names failed failed  %s\n", u_errorName(status));
2882         }
2883         if (strcmp(newLoc,l1)!=0 || strcmp(l1,l2)!=0) {
2884             log_err("The locales are not equal!.Old: %s, New: %s \n", oldLoc, newLoc);
2885         }
2886         log_verbose("ucol_getLocaleByType old:%s   new:%s\n", l1, l2);
2887         ucol_close(c1);
2888         ucol_close(c2);
2889     }
2890     ures_close(resIndex);
2891 #endif
2892 }
2893 
2894 typedef struct OrientationStructTag {
2895     const char* localeId;
2896     ULayoutType character;
2897     ULayoutType line;
2898 } OrientationStruct;
2899 
ULayoutTypeToString(ULayoutType type)2900 static const char* ULayoutTypeToString(ULayoutType type)
2901 {
2902     switch(type)
2903     {
2904     case ULOC_LAYOUT_LTR:
2905         return "ULOC_LAYOUT_LTR";
2906         break;
2907     case ULOC_LAYOUT_RTL:
2908         return "ULOC_LAYOUT_RTL";
2909         break;
2910     case ULOC_LAYOUT_TTB:
2911         return "ULOC_LAYOUT_TTB";
2912         break;
2913     case ULOC_LAYOUT_BTT:
2914         return "ULOC_LAYOUT_BTT";
2915         break;
2916     case ULOC_LAYOUT_UNKNOWN:
2917         break;
2918     }
2919 
2920     return "Unknown enum value for ULayoutType!";
2921 }
2922 
TestOrientation()2923 static void  TestOrientation()
2924 {
2925     static const OrientationStruct toTest [] = {
2926         { "ar", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2927         { "aR", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2928         { "ar_Arab", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2929         { "fa", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2930         { "Fa", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2931         { "he", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2932         { "ps", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2933         { "ur", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2934         { "UR", ULOC_LAYOUT_RTL, ULOC_LAYOUT_TTB },
2935         { "en", ULOC_LAYOUT_LTR, ULOC_LAYOUT_TTB }
2936     };
2937 
2938     size_t i = 0;
2939     for (; i < sizeof(toTest) / sizeof(toTest[0]); ++i) {
2940         UErrorCode statusCO = U_ZERO_ERROR;
2941         UErrorCode statusLO = U_ZERO_ERROR;
2942         const char* const localeId = toTest[i].localeId;
2943         const ULayoutType co = uloc_getCharacterOrientation(localeId, &statusCO);
2944         const ULayoutType expectedCO = toTest[i].character;
2945         const ULayoutType lo = uloc_getLineOrientation(localeId, &statusLO);
2946         const ULayoutType expectedLO = toTest[i].line;
2947         if (U_FAILURE(statusCO)) {
2948             log_err_status(statusCO,
2949                 "  unexpected failure for uloc_getCharacterOrientation(), with localId \"%s\" and status %s\n",
2950                 localeId,
2951                 u_errorName(statusCO));
2952         }
2953         else if (co != expectedCO) {
2954             log_err(
2955                 "  unexpected result for uloc_getCharacterOrientation(), with localeId \"%s\". Expected %s but got result %s\n",
2956                 localeId,
2957                 ULayoutTypeToString(expectedCO),
2958                 ULayoutTypeToString(co));
2959         }
2960         if (U_FAILURE(statusLO)) {
2961             log_err_status(statusLO,
2962                 "  unexpected failure for uloc_getLineOrientation(), with localId \"%s\" and status %s\n",
2963                 localeId,
2964                 u_errorName(statusLO));
2965         }
2966         else if (lo != expectedLO) {
2967             log_err(
2968                 "  unexpected result for uloc_getLineOrientation(), with localeId \"%s\". Expected %s but got result %s\n",
2969                 localeId,
2970                 ULayoutTypeToString(expectedLO),
2971                 ULayoutTypeToString(lo));
2972         }
2973     }
2974 }
2975 
TestULocale()2976 static void  TestULocale() {
2977     int i;
2978     UErrorCode status = U_ZERO_ERROR;
2979     UResourceBundle *resIndex = ures_open(NULL,"res_index", &status);
2980     if(U_FAILURE(status)){
2981         log_err_status(status, "Could not open res_index.res. Exiting. Error: %s\n", u_errorName(status));
2982         return;
2983     }
2984     for (i=0; i<LENGTHOF(LOCALE_ALIAS); i++) {
2985         const char* oldLoc = LOCALE_ALIAS[i][0];
2986         const char* newLoc = LOCALE_ALIAS[i][1];
2987         UChar name1[256], name2[256];
2988         char names1[256], names2[256];
2989         int32_t capacity = 256;
2990 
2991         status = U_ZERO_ERROR;
2992         if(!isLocaleAvailable(resIndex, newLoc)){
2993             continue;
2994         }
2995         uloc_getDisplayName(oldLoc, ULOC_US, name1, capacity, &status);
2996         if(U_FAILURE(status)){
2997             log_err("uloc_getDisplayName(%s) failed %s\n", oldLoc, u_errorName(status));
2998         }
2999 
3000         uloc_getDisplayName(newLoc, ULOC_US, name2, capacity, &status);
3001         if(U_FAILURE(status)){
3002             log_err("uloc_getDisplayName(%s) failed %s\n", newLoc, u_errorName(status));
3003         }
3004 
3005         if (u_strcmp(name1, name2)!=0) {
3006             log_err("The locales are not equal!.Old: %s, New: %s \n", oldLoc, newLoc);
3007         }
3008         u_austrcpy(names1, name1);
3009         u_austrcpy(names2, name2);
3010         log_verbose("uloc_getDisplayName old:%s   new:%s\n", names1, names2);
3011     }
3012     ures_close(resIndex);
3013 
3014 }
3015 
TestUResourceBundle()3016 static void TestUResourceBundle() {
3017     const char* us1;
3018     const char* us2;
3019 
3020     UResourceBundle* rb1 = NULL;
3021     UResourceBundle* rb2 = NULL;
3022     UErrorCode status = U_ZERO_ERROR;
3023     int i;
3024     UResourceBundle *resIndex = NULL;
3025     if(U_FAILURE(status)){
3026         log_err("Could not open res_index.res. Exiting. Error: %s\n", u_errorName(status));
3027         return;
3028     }
3029     resIndex = ures_open(NULL,"res_index", &status);
3030     for (i=0; i<LENGTHOF(LOCALE_ALIAS); i++) {
3031 
3032         const char* oldLoc = LOCALE_ALIAS[i][0];
3033         const char* newLoc = LOCALE_ALIAS[i][1];
3034         if(!isLocaleAvailable(resIndex, newLoc)){
3035             continue;
3036         }
3037         rb1 = ures_open(NULL, oldLoc, &status);
3038         if (U_FAILURE(status)) {
3039             log_err("ures_open(%s) failed %s\n", oldLoc, u_errorName(status));
3040         }
3041 
3042         us1 = ures_getLocaleByType(rb1, ULOC_ACTUAL_LOCALE, &status);
3043 
3044         status = U_ZERO_ERROR;
3045         rb2 = ures_open(NULL, newLoc, &status);
3046         if (U_FAILURE(status)) {
3047             log_err("ures_open(%s) failed %s\n", oldLoc, u_errorName(status));
3048         }
3049         us2 = ures_getLocaleByType(rb2, ULOC_ACTUAL_LOCALE, &status);
3050 
3051         if (strcmp(us1,newLoc)!=0 || strcmp(us1,us2)!=0 ) {
3052             log_err("The locales are not equal!.Old: %s, New: %s \n", oldLoc, newLoc);
3053         }
3054 
3055         log_verbose("ures_getStringByKey old:%s   new:%s\n", us1, us2);
3056         ures_close(rb1);
3057         rb1 = NULL;
3058         ures_close(rb2);
3059         rb2 = NULL;
3060     }
3061     ures_close(resIndex);
3062 }
3063 
TestDisplayName()3064 static void TestDisplayName() {
3065 
3066     UChar oldCountry[256] = {'\0'};
3067     UChar newCountry[256] = {'\0'};
3068     UChar oldLang[256] = {'\0'};
3069     UChar newLang[256] = {'\0'};
3070     char country[256] ={'\0'};
3071     char language[256] ={'\0'};
3072     int32_t capacity = 256;
3073     int i =0;
3074     int j=0;
3075     for (i=0; i<LENGTHOF(LOCALE_ALIAS); i++) {
3076         const char* oldLoc = LOCALE_ALIAS[i][0];
3077         const char* newLoc = LOCALE_ALIAS[i][1];
3078         UErrorCode status = U_ZERO_ERROR;
3079         int32_t available = uloc_countAvailable();
3080 
3081         for(j=0; j<available; j++){
3082 
3083             const char* dispLoc = uloc_getAvailable(j);
3084             int32_t oldCountryLen = uloc_getDisplayCountry(oldLoc,dispLoc, oldCountry, capacity, &status);
3085             int32_t newCountryLen = uloc_getDisplayCountry(newLoc, dispLoc, newCountry, capacity, &status);
3086             int32_t oldLangLen = uloc_getDisplayLanguage(oldLoc, dispLoc, oldLang, capacity, &status);
3087             int32_t newLangLen = uloc_getDisplayLanguage(newLoc, dispLoc, newLang, capacity, &status );
3088 
3089             int32_t countryLen = uloc_getCountry(newLoc, country, capacity, &status);
3090             int32_t langLen  = uloc_getLanguage(newLoc, language, capacity, &status);
3091             /* there is a display name for the current country ID */
3092             if(countryLen != newCountryLen ){
3093                 if(u_strncmp(oldCountry,newCountry,oldCountryLen)!=0){
3094                     log_err("uloc_getDisplayCountry() failed for %s in display locale %s \n", oldLoc, dispLoc);
3095                 }
3096             }
3097             /* there is a display name for the current lang ID */
3098             if(langLen!=newLangLen){
3099                 if(u_strncmp(oldLang,newLang,oldLangLen)){
3100                     log_err("uloc_getDisplayLanguage() failed for %s in display locale %s \n", oldLoc, dispLoc);                }
3101             }
3102         }
3103     }
3104 }
3105 
TestGetLocaleForLCID()3106 static void TestGetLocaleForLCID() {
3107     int32_t i, length, lengthPre;
3108     const char* testLocale = 0;
3109     UErrorCode status = U_ZERO_ERROR;
3110     char            temp2[40], temp3[40];
3111     uint32_t lcid;
3112 
3113     lcid = uloc_getLCID("en_US");
3114     if (lcid != 0x0409) {
3115         log_err("  uloc_getLCID(\"en_US\") = %d, expected 0x0409\n", lcid);
3116     }
3117 
3118     lengthPre = uloc_getLocaleForLCID(lcid, temp2, 4, &status);
3119     if (status != U_BUFFER_OVERFLOW_ERROR) {
3120         log_err("  unexpected result from uloc_getLocaleForLCID with small buffer: %s\n", u_errorName(status));
3121     }
3122     else {
3123         status = U_ZERO_ERROR;
3124     }
3125 
3126     length = uloc_getLocaleForLCID(lcid, temp2, sizeof(temp2)/sizeof(char), &status);
3127     if (U_FAILURE(status)) {
3128         log_err("  unexpected result from uloc_getLocaleForLCID(0x0409): %s\n", u_errorName(status));
3129         status = U_ZERO_ERROR;
3130     }
3131 
3132     if (length != lengthPre) {
3133         log_err("  uloc_getLocaleForLCID(0x0409): returned length %d does not match preflight length %d\n", length, lengthPre);
3134     }
3135 
3136     length = uloc_getLocaleForLCID(0x12345, temp2, sizeof(temp2)/sizeof(char), &status);
3137     if (U_SUCCESS(status)) {
3138         log_err("  unexpected result from uloc_getLocaleForLCID(0x12345): %s, status %s\n", temp2, u_errorName(status));
3139     }
3140     status = U_ZERO_ERROR;
3141 
3142     log_verbose("Testing getLocaleForLCID vs. locale data\n");
3143     for (i = 0; i < LOCALE_SIZE; i++) {
3144 
3145         testLocale=rawData2[NAME][i];
3146 
3147         log_verbose("Testing   %s ......\n", testLocale);
3148 
3149         sscanf(rawData2[LCID][i], "%x", &lcid);
3150         length = uloc_getLocaleForLCID(lcid, temp2, sizeof(temp2)/sizeof(char), &status);
3151         if (U_FAILURE(status)) {
3152             log_err("  unexpected failure of uloc_getLocaleForLCID(%#04x), status %s\n", lcid, u_errorName(status));
3153             status = U_ZERO_ERROR;
3154             continue;
3155         }
3156 
3157         if (length != uprv_strlen(temp2)) {
3158             log_err("  returned length %d not correct for uloc_getLocaleForLCID(%#04x), expected %d\n", length, lcid, uprv_strlen(temp2));
3159         }
3160 
3161         /* Compare language, country, script */
3162         length = uloc_getLanguage(temp2, temp3, sizeof(temp3)/sizeof(char), &status);
3163         if (U_FAILURE(status)) {
3164             log_err("  couldn't get language in uloc_getLocaleForLCID(%#04x) = %s, status %s\n", lcid, temp2, u_errorName(status));
3165             status = U_ZERO_ERROR;
3166         }
3167         else if (uprv_strcmp(temp3, rawData2[LANG][i]) && !(uprv_strcmp(temp3, "nn") == 0 && uprv_strcmp(rawData2[VAR][i], "NY") == 0)) {
3168             log_err("  language doesn't match expected %s in in uloc_getLocaleForLCID(%#04x) = %s\n", rawData2[LANG][i], lcid, temp2);
3169         }
3170 
3171         length = uloc_getScript(temp2, temp3, sizeof(temp3)/sizeof(char), &status);
3172         if (U_FAILURE(status)) {
3173             log_err("  couldn't get script in uloc_getLocaleForLCID(%#04x) = %s, status %s\n", lcid, temp2, u_errorName(status));
3174             status = U_ZERO_ERROR;
3175         }
3176         else if (uprv_strcmp(temp3, rawData2[SCRIPT][i])) {
3177             log_err("  script doesn't match expected %s in in uloc_getLocaleForLCID(%#04x) = %s\n", rawData2[SCRIPT][i], lcid, temp2);
3178         }
3179 
3180         length = uloc_getCountry(temp2, temp3, sizeof(temp3)/sizeof(char), &status);
3181         if (U_FAILURE(status)) {
3182             log_err("  couldn't get country in uloc_getLocaleForLCID(%#04x) = %s, status %s\n", lcid, temp2, u_errorName(status));
3183             status = U_ZERO_ERROR;
3184         }
3185         else if (uprv_strlen(rawData2[CTRY][i]) && uprv_strcmp(temp3, rawData2[CTRY][i])) {
3186             log_err("  country doesn't match expected %s in in uloc_getLocaleForLCID(%#04x) = %s\n", rawData2[CTRY][i], lcid, temp2);
3187         }
3188     }
3189 
3190 }
3191 
3192 const char* const basic_maximize_data[][2] = {
3193   {
3194     "zu_Zzzz_Zz",
3195     "zu_Latn_ZA",
3196   }, {
3197     "ZU_Zz",
3198     "zu_Latn_ZA"
3199   }, {
3200     "zu_LATN",
3201     "zu_Latn_ZA"
3202   }, {
3203     "en_Zz",
3204     "en_Latn_US"
3205   }, {
3206     "en_us",
3207     "en_Latn_US"
3208   }, {
3209     "en_Kore",
3210     "en_Kore_US"
3211   }, {
3212     "en_Kore_Zz",
3213     "en_Kore_US"
3214   }, {
3215     "en_Kore_ZA",
3216     "en_Kore_ZA"
3217   }, {
3218     "en_Kore_ZA_POSIX",
3219     "en_Kore_ZA_POSIX"
3220   }, {
3221     "en_Gujr",
3222     "en_Gujr_US"
3223   }, {
3224     "en_ZA",
3225     "en_Latn_ZA"
3226   }, {
3227     "en_Gujr_Zz",
3228     "en_Gujr_US"
3229   }, {
3230     "en_Gujr_ZA",
3231     "en_Gujr_ZA"
3232   }, {
3233     "en_Gujr_ZA_POSIX",
3234     "en_Gujr_ZA_POSIX"
3235   }, {
3236     "en_US_POSIX_1901",
3237     "en_Latn_US_POSIX_1901"
3238   }, {
3239     "en_Latn__POSIX_1901",
3240     "en_Latn_US_POSIX_1901"
3241   }, {
3242     "en__POSIX_1901",
3243     "en_Latn_US_POSIX_1901"
3244   }, {
3245     "de__POSIX_1901",
3246     "de_Latn_DE_POSIX_1901"
3247   }, {
3248     "en_US_BOSTON",
3249     "en_Latn_US_BOSTON"
3250   }, {
3251     "th@calendar=buddhist",
3252     "th_Thai_TH@calendar=buddhist"
3253   }, {
3254     "ar_ZZ",
3255     "ar_Arab_EG"
3256   }, {
3257     "zh",
3258     "zh_Hans_CN"
3259   }, {
3260     "zh_TW",
3261     "zh_Hant_TW"
3262   }, {
3263     "zh_HK",
3264     "zh_Hant_HK"
3265   }, {
3266     "zh_Hant",
3267     "zh_Hant_TW"
3268   }, {
3269     "zh_Zzzz_CN",
3270     "zh_Hans_CN"
3271   }, {
3272     "und_US",
3273     "en_Latn_US"
3274   }, {
3275     "und_HK",
3276     "zh_Hant_HK"
3277   }, {
3278     "zzz",
3279     ""
3280   }, {
3281      "de_u_co_phonebk",
3282      "de_Latn_DE_U_CO_PHONEBK"
3283   }, {
3284      "de_Latn_u_co_phonebk",
3285      "de_Latn_DE_U_CO_PHONEBK"
3286   }, {
3287      "de_Latn_DE_u_co_phonebk",
3288      "de_Latn_DE_U_CO_PHONEBK"
3289   }
3290 };
3291 
3292 const char* const basic_minimize_data[][2] = {
3293   {
3294     "en_Latn_US",
3295     "en"
3296   }, {
3297     "en_Latn_US_POSIX_1901",
3298     "en__POSIX_1901"
3299   }, {
3300     "EN_Latn_US_POSIX_1901",
3301     "en__POSIX_1901"
3302   }, {
3303     "en_Zzzz_US_POSIX_1901",
3304     "en__POSIX_1901"
3305   }, {
3306     "de_Latn_DE_POSIX_1901",
3307     "de__POSIX_1901"
3308   }, {
3309     "und",
3310     ""
3311   }, {
3312     "en_Latn_US@calendar=gregorian",
3313     "en@calendar=gregorian"
3314   }
3315 };
3316 
3317 const char* const full_data[][3] = {
3318   {
3319     /*   "FROM", */
3320     /*   "ADD-LIKELY", */
3321     /*   "REMOVE-LIKELY" */
3322     /* }, { */
3323     "aa",
3324     "aa_Latn_ET",
3325     "aa"
3326   }, {
3327     "af",
3328     "af_Latn_ZA",
3329     "af"
3330   }, {
3331     "ak",
3332     "ak_Latn_GH",
3333     "ak"
3334   }, {
3335     "am",
3336     "am_Ethi_ET",
3337     "am"
3338   }, {
3339     "ar",
3340     "ar_Arab_EG",
3341     "ar"
3342   }, {
3343     "as",
3344     "as_Beng_IN",
3345     "as"
3346   }, {
3347     "az",
3348     "az_Latn_AZ",
3349     "az"
3350   }, {
3351     "be",
3352     "be_Cyrl_BY",
3353     "be"
3354   }, {
3355     "bg",
3356     "bg_Cyrl_BG",
3357     "bg"
3358   }, {
3359     "bn",
3360     "bn_Beng_BD",
3361     "bn"
3362   }, {
3363     "bo",
3364     "bo_Tibt_CN",
3365     "bo"
3366   }, {
3367     "bs",
3368     "bs_Latn_BA",
3369     "bs"
3370   }, {
3371     "ca",
3372     "ca_Latn_ES",
3373     "ca"
3374   }, {
3375     "ch",
3376     "ch_Latn_GU",
3377     "ch"
3378   }, {
3379     "chk",
3380     "chk_Latn_FM",
3381     "chk"
3382   }, {
3383     "cs",
3384     "cs_Latn_CZ",
3385     "cs"
3386   }, {
3387     "cy",
3388     "cy_Latn_GB",
3389     "cy"
3390   }, {
3391     "da",
3392     "da_Latn_DK",
3393     "da"
3394   }, {
3395     "de",
3396     "de_Latn_DE",
3397     "de"
3398   }, {
3399     "dv",
3400     "dv_Thaa_MV",
3401     "dv"
3402   }, {
3403     "dz",
3404     "dz_Tibt_BT",
3405     "dz"
3406   }, {
3407     "ee",
3408     "ee_Latn_GH",
3409     "ee"
3410   }, {
3411     "el",
3412     "el_Grek_GR",
3413     "el"
3414   }, {
3415     "en",
3416     "en_Latn_US",
3417     "en"
3418   }, {
3419     "es",
3420     "es_Latn_ES",
3421     "es"
3422   }, {
3423     "et",
3424     "et_Latn_EE",
3425     "et"
3426   }, {
3427     "eu",
3428     "eu_Latn_ES",
3429     "eu"
3430   }, {
3431     "fa",
3432     "fa_Arab_IR",
3433     "fa"
3434   }, {
3435     "fi",
3436     "fi_Latn_FI",
3437     "fi"
3438   }, {
3439     "fil",
3440     "fil_Latn_PH",
3441     "fil"
3442   }, {
3443     "fo",
3444     "fo_Latn_FO",
3445     "fo"
3446   }, {
3447     "fr",
3448     "fr_Latn_FR",
3449     "fr"
3450   }, {
3451     "fur",
3452     "fur_Latn_IT",
3453     "fur"
3454   }, {
3455     "ga",
3456     "ga_Latn_IE",
3457     "ga"
3458   }, {
3459     "gaa",
3460     "gaa_Latn_GH",
3461     "gaa"
3462   }, {
3463     "gl",
3464     "gl_Latn_ES",
3465     "gl"
3466   }, {
3467     "gn",
3468     "gn_Latn_PY",
3469     "gn"
3470   }, {
3471     "gu",
3472     "gu_Gujr_IN",
3473     "gu"
3474   }, {
3475     "ha",
3476     "ha_Latn_NG",
3477     "ha"
3478   }, {
3479     "haw",
3480     "haw_Latn_US",
3481     "haw"
3482   }, {
3483     "he",
3484     "he_Hebr_IL",
3485     "he"
3486   }, {
3487     "hi",
3488     "hi_Deva_IN",
3489     "hi"
3490   }, {
3491     "hr",
3492     "hr_Latn_HR",
3493     "hr"
3494   }, {
3495     "ht",
3496     "ht_Latn_HT",
3497     "ht"
3498   }, {
3499     "hu",
3500     "hu_Latn_HU",
3501     "hu"
3502   }, {
3503     "hy",
3504     "hy_Armn_AM",
3505     "hy"
3506   }, {
3507     "id",
3508     "id_Latn_ID",
3509     "id"
3510   }, {
3511     "ig",
3512     "ig_Latn_NG",
3513     "ig"
3514   }, {
3515     "ii",
3516     "ii_Yiii_CN",
3517     "ii"
3518   }, {
3519     "is",
3520     "is_Latn_IS",
3521     "is"
3522   }, {
3523     "it",
3524     "it_Latn_IT",
3525     "it"
3526   }, {
3527     "ja",
3528     "ja_Jpan_JP",
3529     "ja"
3530   }, {
3531     "ka",
3532     "ka_Geor_GE",
3533     "ka"
3534   }, {
3535     "kaj",
3536     "kaj_Latn_NG",
3537     "kaj"
3538   }, {
3539     "kam",
3540     "kam_Latn_KE",
3541     "kam"
3542   }, {
3543     "kk",
3544     "kk_Cyrl_KZ",
3545     "kk"
3546   }, {
3547     "kl",
3548     "kl_Latn_GL",
3549     "kl"
3550   }, {
3551     "km",
3552     "km_Khmr_KH",
3553     "km"
3554   }, {
3555     "kn",
3556     "kn_Knda_IN",
3557     "kn"
3558   }, {
3559     "ko",
3560     "ko_Kore_KR",
3561     "ko"
3562   }, {
3563     "kok",
3564     "kok_Deva_IN",
3565     "kok"
3566   }, {
3567     "kpe",
3568     "kpe_Latn_LR",
3569     "kpe"
3570   }, {
3571     "ku",
3572     "ku_Latn_TR",
3573     "ku"
3574   }, {
3575     "ky",
3576     "ky_Cyrl_KG",
3577     "ky"
3578   }, {
3579     "la",
3580     "la_Latn_VA",
3581     "la"
3582   }, {
3583     "ln",
3584     "ln_Latn_CD",
3585     "ln"
3586   }, {
3587     "lo",
3588     "lo_Laoo_LA",
3589     "lo"
3590   }, {
3591     "lt",
3592     "lt_Latn_LT",
3593     "lt"
3594   }, {
3595     "lv",
3596     "lv_Latn_LV",
3597     "lv"
3598   }, {
3599     "mg",
3600     "mg_Latn_MG",
3601     "mg"
3602   }, {
3603     "mh",
3604     "mh_Latn_MH",
3605     "mh"
3606   }, {
3607     "mk",
3608     "mk_Cyrl_MK",
3609     "mk"
3610   }, {
3611     "ml",
3612     "ml_Mlym_IN",
3613     "ml"
3614   }, {
3615     "mn",
3616     "mn_Cyrl_MN",
3617     "mn"
3618   }, {
3619     "mr",
3620     "mr_Deva_IN",
3621     "mr"
3622   }, {
3623     "ms",
3624     "ms_Latn_MY",
3625     "ms"
3626   }, {
3627     "mt",
3628     "mt_Latn_MT",
3629     "mt"
3630   }, {
3631     "my",
3632     "my_Mymr_MM",
3633     "my"
3634   }, {
3635     "na",
3636     "na_Latn_NR",
3637     "na"
3638   }, {
3639     "ne",
3640     "ne_Deva_NP",
3641     "ne"
3642   }, {
3643     "niu",
3644     "niu_Latn_NU",
3645     "niu"
3646   }, {
3647     "nl",
3648     "nl_Latn_NL",
3649     "nl"
3650   }, {
3651     "nn",
3652     "nn_Latn_NO",
3653     "nn"
3654   }, {
3655     "nr",
3656     "nr_Latn_ZA",
3657     "nr"
3658   }, {
3659     "nso",
3660     "nso_Latn_ZA",
3661     "nso"
3662   }, {
3663     "ny",
3664     "ny_Latn_MW",
3665     "ny"
3666   }, {
3667     "om",
3668     "om_Latn_ET",
3669     "om"
3670   }, {
3671     "or",
3672     "or_Orya_IN",
3673     "or"
3674   }, {
3675     "pa",
3676     "pa_Guru_IN",
3677     "pa"
3678   }, {
3679     "pa_Arab",
3680     "pa_Arab_PK",
3681     "pa_PK"
3682   }, {
3683     "pa_PK",
3684     "pa_Arab_PK",
3685     "pa_PK"
3686   }, {
3687     "pap",
3688     "pap_Latn_AW",
3689     "pap"
3690   }, {
3691     "pau",
3692     "pau_Latn_PW",
3693     "pau"
3694   }, {
3695     "pl",
3696     "pl_Latn_PL",
3697     "pl"
3698   }, {
3699     "ps",
3700     "ps_Arab_AF",
3701     "ps"
3702   }, {
3703     "pt",
3704     "pt_Latn_BR",
3705     "pt"
3706   }, {
3707     "rn",
3708     "rn_Latn_BI",
3709     "rn"
3710   }, {
3711     "ro",
3712     "ro_Latn_RO",
3713     "ro"
3714   }, {
3715     "ru",
3716     "ru_Cyrl_RU",
3717     "ru"
3718   }, {
3719     "rw",
3720     "rw_Latn_RW",
3721     "rw"
3722   }, {
3723     "sa",
3724     "sa_Deva_IN",
3725     "sa"
3726   }, {
3727     "se",
3728     "se_Latn_NO",
3729     "se"
3730   }, {
3731     "sg",
3732     "sg_Latn_CF",
3733     "sg"
3734   }, {
3735     "si",
3736     "si_Sinh_LK",
3737     "si"
3738   }, {
3739     "sid",
3740     "sid_Latn_ET",
3741     "sid"
3742   }, {
3743     "sk",
3744     "sk_Latn_SK",
3745     "sk"
3746   }, {
3747     "sl",
3748     "sl_Latn_SI",
3749     "sl"
3750   }, {
3751     "sm",
3752     "sm_Latn_WS",
3753     "sm"
3754   }, {
3755     "so",
3756     "so_Latn_SO",
3757     "so"
3758   }, {
3759     "sq",
3760     "sq_Latn_AL",
3761     "sq"
3762   }, {
3763     "sr",
3764     "sr_Cyrl_RS",
3765     "sr"
3766   }, {
3767     "ss",
3768     "ss_Latn_ZA",
3769     "ss"
3770   }, {
3771     "st",
3772     "st_Latn_ZA",
3773     "st"
3774   }, {
3775     "sv",
3776     "sv_Latn_SE",
3777     "sv"
3778   }, {
3779     "sw",
3780     "sw_Latn_TZ",
3781     "sw"
3782   }, {
3783     "ta",
3784     "ta_Taml_IN",
3785     "ta"
3786   }, {
3787     "te",
3788     "te_Telu_IN",
3789     "te"
3790   }, {
3791     "tet",
3792     "tet_Latn_TL",
3793     "tet"
3794   }, {
3795     "tg",
3796     "tg_Cyrl_TJ",
3797     "tg"
3798   }, {
3799     "th",
3800     "th_Thai_TH",
3801     "th"
3802   }, {
3803     "ti",
3804     "ti_Ethi_ET",
3805     "ti"
3806   }, {
3807     "tig",
3808     "tig_Ethi_ER",
3809     "tig"
3810   }, {
3811     "tk",
3812     "tk_Latn_TM",
3813     "tk"
3814   }, {
3815     "tkl",
3816     "tkl_Latn_TK",
3817     "tkl"
3818   }, {
3819     "tn",
3820     "tn_Latn_ZA",
3821     "tn"
3822   }, {
3823     "to",
3824     "to_Latn_TO",
3825     "to"
3826   }, {
3827     "tpi",
3828     "tpi_Latn_PG",
3829     "tpi"
3830   }, {
3831     "tr",
3832     "tr_Latn_TR",
3833     "tr"
3834   }, {
3835     "ts",
3836     "ts_Latn_ZA",
3837     "ts"
3838   }, {
3839     "tt",
3840     "tt_Cyrl_RU",
3841     "tt"
3842   }, {
3843     "tvl",
3844     "tvl_Latn_TV",
3845     "tvl"
3846   }, {
3847     "ty",
3848     "ty_Latn_PF",
3849     "ty"
3850   }, {
3851     "uk",
3852     "uk_Cyrl_UA",
3853     "uk"
3854   }, {
3855     "und",
3856     "en_Latn_US",
3857     "en"
3858   }, {
3859     "und_AD",
3860     "ca_Latn_AD",
3861     "ca_AD"
3862   }, {
3863     "und_AE",
3864     "ar_Arab_AE",
3865     "ar_AE"
3866   }, {
3867     "und_AF",
3868     "fa_Arab_AF",
3869     "fa_AF"
3870   }, {
3871     "und_AL",
3872     "sq_Latn_AL",
3873     "sq"
3874   }, {
3875     "und_AM",
3876     "hy_Armn_AM",
3877     "hy"
3878   }, {
3879     "und_AO",
3880     "pt_Latn_AO",
3881     "pt_AO"
3882   }, {
3883     "und_AR",
3884     "es_Latn_AR",
3885     "es_AR"
3886   }, {
3887     "und_AS",
3888     "sm_Latn_AS",
3889     "sm_AS"
3890   }, {
3891     "und_AT",
3892     "de_Latn_AT",
3893     "de_AT"
3894   }, {
3895     "und_AW",
3896     "nl_Latn_AW",
3897     "nl_AW"
3898   }, {
3899     "und_AX",
3900     "sv_Latn_AX",
3901     "sv_AX"
3902   }, {
3903     "und_AZ",
3904     "az_Latn_AZ",
3905     "az"
3906   }, {
3907     "und_Arab",
3908     "ar_Arab_EG",
3909     "ar"
3910   }, {
3911     "und_Arab_IN",
3912     "ur_Arab_IN",
3913     "ur_IN"
3914   }, {
3915     "und_Arab_PK",
3916     "ur_Arab_PK",
3917     "ur"
3918   }, {
3919     "und_Arab_SN",
3920     "ar_Arab_SN",
3921     "ar_SN"
3922   }, {
3923     "und_Armn",
3924     "hy_Armn_AM",
3925     "hy"
3926   }, {
3927     "und_BA",
3928     "bs_Latn_BA",
3929     "bs"
3930   }, {
3931     "und_BD",
3932     "bn_Beng_BD",
3933     "bn"
3934   }, {
3935     "und_BE",
3936     "nl_Latn_BE",
3937     "nl_BE"
3938   }, {
3939     "und_BF",
3940     "fr_Latn_BF",
3941     "fr_BF"
3942   }, {
3943     "und_BG",
3944     "bg_Cyrl_BG",
3945     "bg"
3946   }, {
3947     "und_BH",
3948     "ar_Arab_BH",
3949     "ar_BH"
3950   }, {
3951     "und_BI",
3952     "rn_Latn_BI",
3953     "rn"
3954   }, {
3955     "und_BJ",
3956     "fr_Latn_BJ",
3957     "fr_BJ"
3958   }, {
3959     "und_BN",
3960     "ms_Latn_BN",
3961     "ms_BN"
3962   }, {
3963     "und_BO",
3964     "es_Latn_BO",
3965     "es_BO"
3966   }, {
3967     "und_BR",
3968     "pt_Latn_BR",
3969     "pt"
3970   }, {
3971     "und_BT",
3972     "dz_Tibt_BT",
3973     "dz"
3974   }, {
3975     "und_BY",
3976     "be_Cyrl_BY",
3977     "be"
3978   }, {
3979     "und_Beng",
3980     "bn_Beng_BD",
3981     "bn"
3982   }, {
3983     "und_Beng_IN",
3984     "bn_Beng_IN",
3985     "bn_IN"
3986   }, {
3987     "und_CD",
3988     "sw_Latn_CD",
3989     "sw_CD"
3990   }, {
3991     "und_CF",
3992     "fr_Latn_CF",
3993     "fr_CF"
3994   }, {
3995     "und_CG",
3996     "fr_Latn_CG",
3997     "fr_CG"
3998   }, {
3999     "und_CH",
4000     "de_Latn_CH",
4001     "de_CH"
4002   }, {
4003     "und_CI",
4004     "fr_Latn_CI",
4005     "fr_CI"
4006   }, {
4007     "und_CL",
4008     "es_Latn_CL",
4009     "es_CL"
4010   }, {
4011     "und_CM",
4012     "fr_Latn_CM",
4013     "fr_CM"
4014   }, {
4015     "und_CN",
4016     "zh_Hans_CN",
4017     "zh"
4018   }, {
4019     "und_CO",
4020     "es_Latn_CO",
4021     "es_CO"
4022   }, {
4023     "und_CR",
4024     "es_Latn_CR",
4025     "es_CR"
4026   }, {
4027     "und_CU",
4028     "es_Latn_CU",
4029     "es_CU"
4030   }, {
4031     "und_CV",
4032     "pt_Latn_CV",
4033     "pt_CV"
4034   }, {
4035     "und_CY",
4036     "el_Grek_CY",
4037     "el_CY"
4038   }, {
4039     "und_CZ",
4040     "cs_Latn_CZ",
4041     "cs"
4042   }, {
4043     "und_Cher",
4044     "chr_Cher_US",
4045     "chr"
4046   }, {
4047     "und_Cyrl",
4048     "ru_Cyrl_RU",
4049     "ru"
4050   }, {
4051     "und_Cyrl_KZ",
4052     "ru_Cyrl_KZ",
4053     "ru_KZ"
4054   }, {
4055     "und_DE",
4056     "de_Latn_DE",
4057     "de"
4058   }, {
4059     "und_DJ",
4060     "aa_Latn_DJ",
4061     "aa_DJ"
4062   }, {
4063     "und_DK",
4064     "da_Latn_DK",
4065     "da"
4066   }, {
4067     "und_DO",
4068     "es_Latn_DO",
4069     "es_DO"
4070   }, {
4071     "und_DZ",
4072     "ar_Arab_DZ",
4073     "ar_DZ"
4074   }, {
4075     "und_Deva",
4076     "hi_Deva_IN",
4077     "hi"
4078   }, {
4079     "und_EC",
4080     "es_Latn_EC",
4081     "es_EC"
4082   }, {
4083     "und_EE",
4084     "et_Latn_EE",
4085     "et"
4086   }, {
4087     "und_EG",
4088     "ar_Arab_EG",
4089     "ar"
4090   }, {
4091     "und_EH",
4092     "ar_Arab_EH",
4093     "ar_EH"
4094   }, {
4095     "und_ER",
4096     "ti_Ethi_ER",
4097     "ti_ER"
4098   }, {
4099     "und_ES",
4100     "es_Latn_ES",
4101     "es"
4102   }, {
4103     "und_ET",
4104     "am_Ethi_ET",
4105     "am"
4106   }, {
4107     "und_Ethi",
4108     "am_Ethi_ET",
4109     "am"
4110   }, {
4111     "und_Ethi_ER",
4112     "am_Ethi_ER",
4113     "am_ER"
4114   }, {
4115     "und_FI",
4116     "fi_Latn_FI",
4117     "fi"
4118   }, {
4119     "und_FM",
4120     "chk_Latn_FM",
4121     "chk"
4122   }, {
4123     "und_FO",
4124     "fo_Latn_FO",
4125     "fo"
4126   }, {
4127     "und_FR",
4128     "fr_Latn_FR",
4129     "fr"
4130   }, {
4131     "und_GA",
4132     "fr_Latn_GA",
4133     "fr_GA"
4134   }, {
4135     "und_GE",
4136     "ka_Geor_GE",
4137     "ka"
4138   }, {
4139     "und_GF",
4140     "fr_Latn_GF",
4141     "fr_GF"
4142   }, {
4143     "und_GL",
4144     "kl_Latn_GL",
4145     "kl"
4146   }, {
4147     "und_GN",
4148     "fr_Latn_GN",
4149     "fr_GN"
4150   }, {
4151     "und_GP",
4152     "fr_Latn_GP",
4153     "fr_GP"
4154   }, {
4155     "und_GQ",
4156     "es_Latn_GQ",
4157     "es_GQ"
4158   }, {
4159     "und_GR",
4160     "el_Grek_GR",
4161     "el"
4162   }, {
4163     "und_GT",
4164     "es_Latn_GT",
4165     "es_GT"
4166   }, {
4167     "und_GU",
4168     "en_Latn_GU",
4169     "en_GU"
4170   }, {
4171     "und_GW",
4172     "pt_Latn_GW",
4173     "pt_GW"
4174   }, {
4175     "und_Geor",
4176     "ka_Geor_GE",
4177     "ka"
4178   }, {
4179     "und_Grek",
4180     "el_Grek_GR",
4181     "el"
4182   }, {
4183     "und_Gujr",
4184     "gu_Gujr_IN",
4185     "gu"
4186   }, {
4187     "und_Guru",
4188     "pa_Guru_IN",
4189     "pa"
4190   }, {
4191     "und_HK",
4192     "zh_Hant_HK",
4193     "zh_HK"
4194   }, {
4195     "und_HN",
4196     "es_Latn_HN",
4197     "es_HN"
4198   }, {
4199     "und_HR",
4200     "hr_Latn_HR",
4201     "hr"
4202   }, {
4203     "und_HT",
4204     "ht_Latn_HT",
4205     "ht"
4206   }, {
4207     "und_HU",
4208     "hu_Latn_HU",
4209     "hu"
4210   }, {
4211     "und_Hani",
4212     "zh_Hani_CN",
4213     "zh_Hani"
4214   }, {
4215     "und_Hans",
4216     "zh_Hans_CN",
4217     "zh"
4218   }, {
4219     "und_Hant",
4220     "zh_Hant_TW",
4221     "zh_TW"
4222   }, {
4223     "und_Hebr",
4224     "he_Hebr_IL",
4225     "he"
4226   }, {
4227     "und_IL",
4228     "he_Hebr_IL",
4229     "he"
4230   }, {
4231     "und_IN",
4232     "hi_Deva_IN",
4233     "hi"
4234   }, {
4235     "und_IQ",
4236     "ar_Arab_IQ",
4237     "ar_IQ"
4238   }, {
4239     "und_IR",
4240     "fa_Arab_IR",
4241     "fa"
4242   }, {
4243     "und_IS",
4244     "is_Latn_IS",
4245     "is"
4246   }, {
4247     "und_IT",
4248     "it_Latn_IT",
4249     "it"
4250   }, {
4251     "und_JO",
4252     "ar_Arab_JO",
4253     "ar_JO"
4254   }, {
4255     "und_JP",
4256     "ja_Jpan_JP",
4257     "ja"
4258   }, {
4259     "und_Jpan",
4260     "ja_Jpan_JP",
4261     "ja"
4262   }, {
4263     "und_KG",
4264     "ky_Cyrl_KG",
4265     "ky"
4266   }, {
4267     "und_KH",
4268     "km_Khmr_KH",
4269     "km"
4270   }, {
4271     "und_KM",
4272     "ar_Arab_KM",
4273     "ar_KM"
4274   }, {
4275     "und_KP",
4276     "ko_Kore_KP",
4277     "ko_KP"
4278   }, {
4279     "und_KR",
4280     "ko_Kore_KR",
4281     "ko"
4282   }, {
4283     "und_KW",
4284     "ar_Arab_KW",
4285     "ar_KW"
4286   }, {
4287     "und_KZ",
4288     "ru_Cyrl_KZ",
4289     "ru_KZ"
4290   }, {
4291     "und_Khmr",
4292     "km_Khmr_KH",
4293     "km"
4294   }, {
4295     "und_Knda",
4296     "kn_Knda_IN",
4297     "kn"
4298   }, {
4299     "und_Kore",
4300     "ko_Kore_KR",
4301     "ko"
4302   }, {
4303     "und_LA",
4304     "lo_Laoo_LA",
4305     "lo"
4306   }, {
4307     "und_LB",
4308     "ar_Arab_LB",
4309     "ar_LB"
4310   }, {
4311     "und_LI",
4312     "de_Latn_LI",
4313     "de_LI"
4314   }, {
4315     "und_LK",
4316     "si_Sinh_LK",
4317     "si"
4318   }, {
4319     "und_LS",
4320     "st_Latn_LS",
4321     "st_LS"
4322   }, {
4323     "und_LT",
4324     "lt_Latn_LT",
4325     "lt"
4326   }, {
4327     "und_LU",
4328     "fr_Latn_LU",
4329     "fr_LU"
4330   }, {
4331     "und_LV",
4332     "lv_Latn_LV",
4333     "lv"
4334   }, {
4335     "und_LY",
4336     "ar_Arab_LY",
4337     "ar_LY"
4338   }, {
4339     "und_Laoo",
4340     "lo_Laoo_LA",
4341     "lo"
4342   }, {
4343     "und_Latn_ES",
4344     "es_Latn_ES",
4345     "es"
4346   }, {
4347     "und_Latn_ET",
4348     "en_Latn_ET",
4349     "en_ET"
4350   }, {
4351     "und_Latn_GB",
4352     "en_Latn_GB",
4353     "en_GB"
4354   }, {
4355     "und_Latn_GH",
4356     "ak_Latn_GH",
4357     "ak"
4358   }, {
4359     "und_Latn_ID",
4360     "id_Latn_ID",
4361     "id"
4362   }, {
4363     "und_Latn_IT",
4364     "it_Latn_IT",
4365     "it"
4366   }, {
4367     "und_Latn_NG",
4368     "en_Latn_NG",
4369     "en_NG"
4370   }, {
4371     "und_Latn_TR",
4372     "tr_Latn_TR",
4373     "tr"
4374   }, {
4375     "und_Latn_ZA",
4376     "en_Latn_ZA",
4377     "en_ZA"
4378   }, {
4379     "und_MA",
4380     "ar_Arab_MA",
4381     "ar_MA"
4382   }, {
4383     "und_MC",
4384     "fr_Latn_MC",
4385     "fr_MC"
4386   }, {
4387     "und_MD",
4388     "ro_Latn_MD",
4389     "ro_MD"
4390   }, {
4391     "und_ME",
4392     "sr_Latn_ME",
4393     "sr_ME"
4394   }, {
4395     "und_MG",
4396     "mg_Latn_MG",
4397     "mg"
4398   }, {
4399     "und_MH",
4400     "en_Latn_MH",
4401     "en_MH"
4402   }, {
4403     "und_MK",
4404     "mk_Cyrl_MK",
4405     "mk"
4406   }, {
4407     "und_ML",
4408     "bm_Latn_ML",
4409     "bm"
4410   }, {
4411     "und_MM",
4412     "my_Mymr_MM",
4413     "my"
4414   }, {
4415     "und_MN",
4416     "mn_Cyrl_MN",
4417     "mn"
4418   }, {
4419     "und_MO",
4420     "zh_Hant_MO",
4421     "zh_MO"
4422   }, {
4423     "und_MQ",
4424     "fr_Latn_MQ",
4425     "fr_MQ"
4426   }, {
4427     "und_MR",
4428     "ar_Arab_MR",
4429     "ar_MR"
4430   }, {
4431     "und_MT",
4432     "mt_Latn_MT",
4433     "mt"
4434   }, {
4435     "und_MV",
4436     "dv_Thaa_MV",
4437     "dv"
4438   }, {
4439     "und_MW",
4440     "en_Latn_MW",
4441     "en_MW"
4442   }, {
4443     "und_MX",
4444     "es_Latn_MX",
4445     "es_MX"
4446   }, {
4447     "und_MY",
4448     "ms_Latn_MY",
4449     "ms"
4450   }, {
4451     "und_MZ",
4452     "pt_Latn_MZ",
4453     "pt_MZ"
4454   }, {
4455     "und_Mlym",
4456     "ml_Mlym_IN",
4457     "ml"
4458   }, {
4459     "und_Mymr",
4460     "my_Mymr_MM",
4461     "my"
4462   }, {
4463     "und_NC",
4464     "fr_Latn_NC",
4465     "fr_NC"
4466   }, {
4467     "und_NE",
4468     "ha_Latn_NE",
4469     "ha_NE"
4470   }, {
4471     "und_NG",
4472     "en_Latn_NG",
4473     "en_NG"
4474   }, {
4475     "und_NI",
4476     "es_Latn_NI",
4477     "es_NI"
4478   }, {
4479     "und_NL",
4480     "nl_Latn_NL",
4481     "nl"
4482   }, {
4483     "und_NO",
4484     "nb_Latn_NO",
4485     "nb"
4486   }, {
4487     "und_NP",
4488     "ne_Deva_NP",
4489     "ne"
4490   }, {
4491     "und_NR",
4492     "en_Latn_NR",
4493     "en_NR"
4494   }, {
4495     "und_NU",
4496     "en_Latn_NU",
4497     "en_NU"
4498   }, {
4499     "und_OM",
4500     "ar_Arab_OM",
4501     "ar_OM"
4502   }, {
4503     "und_Orya",
4504     "or_Orya_IN",
4505     "or"
4506   }, {
4507     "und_PA",
4508     "es_Latn_PA",
4509     "es_PA"
4510   }, {
4511     "und_PE",
4512     "es_Latn_PE",
4513     "es_PE"
4514   }, {
4515     "und_PF",
4516     "fr_Latn_PF",
4517     "fr_PF"
4518   }, {
4519     "und_PG",
4520     "tpi_Latn_PG",
4521     "tpi"
4522   }, {
4523     "und_PH",
4524     "fil_Latn_PH",
4525     "fil"
4526   }, {
4527     "und_PL",
4528     "pl_Latn_PL",
4529     "pl"
4530   }, {
4531     "und_PM",
4532     "fr_Latn_PM",
4533     "fr_PM"
4534   }, {
4535     "und_PR",
4536     "es_Latn_PR",
4537     "es_PR"
4538   }, {
4539     "und_PS",
4540     "ar_Arab_PS",
4541     "ar_PS"
4542   }, {
4543     "und_PT",
4544     "pt_Latn_PT",
4545     "pt_PT"
4546   }, {
4547     "und_PW",
4548     "pau_Latn_PW",
4549     "pau"
4550   }, {
4551     "und_PY",
4552     "gn_Latn_PY",
4553     "gn"
4554   }, {
4555     "und_QA",
4556     "ar_Arab_QA",
4557     "ar_QA"
4558   }, {
4559     "und_RE",
4560     "fr_Latn_RE",
4561     "fr_RE"
4562   }, {
4563     "und_RO",
4564     "ro_Latn_RO",
4565     "ro"
4566   }, {
4567     "und_RS",
4568     "sr_Cyrl_RS",
4569     "sr"
4570   }, {
4571     "und_RU",
4572     "ru_Cyrl_RU",
4573     "ru"
4574   }, {
4575     "und_RW",
4576     "rw_Latn_RW",
4577     "rw"
4578   }, {
4579     "und_SA",
4580     "ar_Arab_SA",
4581     "ar_SA"
4582   }, {
4583     "und_SD",
4584     "ar_Arab_SD",
4585     "ar_SD"
4586   }, {
4587     "und_SE",
4588     "sv_Latn_SE",
4589     "sv"
4590   }, {
4591     "und_SG",
4592     "en_Latn_SG",
4593     "en_SG"
4594   }, {
4595     "und_SI",
4596     "sl_Latn_SI",
4597     "sl"
4598   }, {
4599     "und_SJ",
4600     "nb_Latn_SJ",
4601     "nb_SJ"
4602   }, {
4603     "und_SK",
4604     "sk_Latn_SK",
4605     "sk"
4606   }, {
4607     "und_SM",
4608     "it_Latn_SM",
4609     "it_SM"
4610   }, {
4611     "und_SN",
4612     "fr_Latn_SN",
4613     "fr_SN"
4614   }, {
4615     "und_SO",
4616     "so_Latn_SO",
4617     "so"
4618   }, {
4619     "und_SR",
4620     "nl_Latn_SR",
4621     "nl_SR"
4622   }, {
4623     "und_ST",
4624     "pt_Latn_ST",
4625     "pt_ST"
4626   }, {
4627     "und_SV",
4628     "es_Latn_SV",
4629     "es_SV"
4630   }, {
4631     "und_SY",
4632     "ar_Arab_SY",
4633     "ar_SY"
4634   }, {
4635     "und_Sinh",
4636     "si_Sinh_LK",
4637     "si"
4638   }, {
4639     "und_TD",
4640     "fr_Latn_TD",
4641     "fr_TD"
4642   }, {
4643     "und_TG",
4644     "fr_Latn_TG",
4645     "fr_TG"
4646   }, {
4647     "und_TH",
4648     "th_Thai_TH",
4649     "th"
4650   }, {
4651     "und_TJ",
4652     "tg_Cyrl_TJ",
4653     "tg"
4654   }, {
4655     "und_TK",
4656     "tkl_Latn_TK",
4657     "tkl"
4658   }, {
4659     "und_TL",
4660     "pt_Latn_TL",
4661     "pt_TL"
4662   }, {
4663     "und_TM",
4664     "tk_Latn_TM",
4665     "tk"
4666   }, {
4667     "und_TN",
4668     "ar_Arab_TN",
4669     "ar_TN"
4670   }, {
4671     "und_TO",
4672     "to_Latn_TO",
4673     "to"
4674   }, {
4675     "und_TR",
4676     "tr_Latn_TR",
4677     "tr"
4678   }, {
4679     "und_TV",
4680     "tvl_Latn_TV",
4681     "tvl"
4682   }, {
4683     "und_TW",
4684     "zh_Hant_TW",
4685     "zh_TW"
4686   }, {
4687     "und_Taml",
4688     "ta_Taml_IN",
4689     "ta"
4690   }, {
4691     "und_Telu",
4692     "te_Telu_IN",
4693     "te"
4694   }, {
4695     "und_Thaa",
4696     "dv_Thaa_MV",
4697     "dv"
4698   }, {
4699     "und_Thai",
4700     "th_Thai_TH",
4701     "th"
4702   }, {
4703     "und_Tibt",
4704     "bo_Tibt_CN",
4705     "bo"
4706   }, {
4707     "und_UA",
4708     "uk_Cyrl_UA",
4709     "uk"
4710   }, {
4711     "und_UY",
4712     "es_Latn_UY",
4713     "es_UY"
4714   }, {
4715     "und_UZ",
4716     "uz_Latn_UZ",
4717     "uz"
4718   }, {
4719     "und_VA",
4720     "la_Latn_VA",
4721     "la"
4722   }, {
4723     "und_VE",
4724     "es_Latn_VE",
4725     "es_VE"
4726   }, {
4727     "und_VN",
4728     "vi_Latn_VN",
4729     "vi"
4730   }, {
4731     "und_VU",
4732     "bi_Latn_VU",
4733     "bi"
4734   }, {
4735     "und_WF",
4736     "fr_Latn_WF",
4737     "fr_WF"
4738   }, {
4739     "und_WS",
4740     "sm_Latn_WS",
4741     "sm"
4742   }, {
4743     "und_YE",
4744     "ar_Arab_YE",
4745     "ar_YE"
4746   }, {
4747     "und_YT",
4748     "fr_Latn_YT",
4749     "fr_YT"
4750   }, {
4751     "und_Yiii",
4752     "ii_Yiii_CN",
4753     "ii"
4754   }, {
4755     "ur",
4756     "ur_Arab_PK",
4757     "ur"
4758   }, {
4759     "uz",
4760     "uz_Latn_UZ",
4761     "uz"
4762   }, {
4763     "uz_AF",
4764     "uz_Arab_AF",
4765     "uz_AF"
4766   }, {
4767     "uz_Arab",
4768     "uz_Arab_AF",
4769     "uz_AF"
4770   }, {
4771     "ve",
4772     "ve_Latn_ZA",
4773     "ve"
4774   }, {
4775     "vi",
4776     "vi_Latn_VN",
4777     "vi"
4778   }, {
4779     "wal",
4780     "wal_Ethi_ET",
4781     "wal"
4782   }, {
4783     "wo",
4784     "wo_Latn_SN",
4785     "wo"
4786   }, {
4787     "xh",
4788     "xh_Latn_ZA",
4789     "xh"
4790   }, {
4791     "yo",
4792     "yo_Latn_NG",
4793     "yo"
4794   }, {
4795     "zh",
4796     "zh_Hans_CN",
4797     "zh"
4798   }, {
4799     "zh_HK",
4800     "zh_Hant_HK",
4801     "zh_HK"
4802   }, {
4803     "zh_Hani",
4804     "zh_Hani_CN", /* changed due to cldrbug 6204, may be an error */
4805     "zh_Hani", /* changed due to cldrbug 6204, may be an error */
4806   }, {
4807     "zh_Hant",
4808     "zh_Hant_TW",
4809     "zh_TW"
4810   }, {
4811     "zh_MO",
4812     "zh_Hant_MO",
4813     "zh_MO"
4814   }, {
4815     "zh_TW",
4816     "zh_Hant_TW",
4817     "zh_TW"
4818   }, {
4819     "zu",
4820     "zu_Latn_ZA",
4821     "zu"
4822   }, {
4823     "und",
4824     "en_Latn_US",
4825     "en"
4826   }, {
4827     "und_ZZ",
4828     "en_Latn_US",
4829     "en"
4830   }, {
4831     "und_CN",
4832     "zh_Hans_CN",
4833     "zh"
4834   }, {
4835     "und_TW",
4836     "zh_Hant_TW",
4837     "zh_TW"
4838   }, {
4839     "und_HK",
4840     "zh_Hant_HK",
4841     "zh_HK"
4842   }, {
4843     "und_AQ",
4844     "und_Latn_AQ",
4845     "und_AQ"
4846   }, {
4847     "und_Zzzz",
4848     "en_Latn_US",
4849     "en"
4850   }, {
4851     "und_Zzzz_ZZ",
4852     "en_Latn_US",
4853     "en"
4854   }, {
4855     "und_Zzzz_CN",
4856     "zh_Hans_CN",
4857     "zh"
4858   }, {
4859     "und_Zzzz_TW",
4860     "zh_Hant_TW",
4861     "zh_TW"
4862   }, {
4863     "und_Zzzz_HK",
4864     "zh_Hant_HK",
4865     "zh_HK"
4866   }, {
4867     "und_Zzzz_AQ",
4868     "und_Latn_AQ",
4869     "und_AQ"
4870   }, {
4871     "und_Latn",
4872     "en_Latn_US",
4873     "en"
4874   }, {
4875     "und_Latn_ZZ",
4876     "en_Latn_US",
4877     "en"
4878   }, {
4879     "und_Latn_CN",
4880     "za_Latn_CN",
4881     "za"
4882   }, {
4883     "und_Latn_TW",
4884     "trv_Latn_TW",
4885     "trv"
4886   }, {
4887     "und_Latn_HK",
4888     "zh_Latn_HK",
4889     "zh_Latn_HK"
4890   }, {
4891     "und_Latn_AQ",
4892     "und_Latn_AQ",
4893     "und_AQ"
4894   }, {
4895     "und_Hans",
4896     "zh_Hans_CN",
4897     "zh"
4898   }, {
4899     "und_Hans_ZZ",
4900     "zh_Hans_CN",
4901     "zh"
4902   }, {
4903     "und_Hans_CN",
4904     "zh_Hans_CN",
4905     "zh"
4906   }, {
4907     "und_Hans_TW",
4908     "zh_Hans_TW",
4909     "zh_Hans_TW"
4910   }, {
4911     "und_Hans_HK",
4912     "zh_Hans_HK",
4913     "zh_Hans_HK"
4914   }, {
4915     "und_Hans_AQ",
4916     "zh_Hans_AQ",
4917     "zh_AQ"
4918   }, {
4919     "und_Hant",
4920     "zh_Hant_TW",
4921     "zh_TW"
4922   }, {
4923     "und_Hant_ZZ",
4924     "zh_Hant_TW",
4925     "zh_TW"
4926   }, {
4927     "und_Hant_CN",
4928     "zh_Hant_CN",
4929     "zh_Hant_CN"
4930   }, {
4931     "und_Hant_TW",
4932     "zh_Hant_TW",
4933     "zh_TW"
4934   }, {
4935     "und_Hant_HK",
4936     "zh_Hant_HK",
4937     "zh_HK"
4938   }, {
4939     "und_Hant_AQ",
4940     "zh_Hant_AQ",
4941     "zh_Hant_AQ"
4942   }, {
4943     "und_Moon",
4944     "en_Moon_US",
4945     "en_Moon"
4946   }, {
4947     "und_Moon_ZZ",
4948     "en_Moon_US",
4949     "en_Moon"
4950   }, {
4951     "und_Moon_CN",
4952     "zh_Moon_CN",
4953     "zh_Moon"
4954   }, {
4955     "und_Moon_TW",
4956     "zh_Moon_TW",
4957     "zh_Moon_TW"
4958   }, {
4959     "und_Moon_HK",
4960     "zh_Moon_HK",
4961     "zh_Moon_HK"
4962   }, {
4963     "und_Moon_AQ",
4964     "und_Moon_AQ",
4965     "und_Moon_AQ"
4966   }, {
4967     "es",
4968     "es_Latn_ES",
4969     "es"
4970   }, {
4971     "es_ZZ",
4972     "es_Latn_ES",
4973     "es"
4974   }, {
4975     "es_CN",
4976     "es_Latn_CN",
4977     "es_CN"
4978   }, {
4979     "es_TW",
4980     "es_Latn_TW",
4981     "es_TW"
4982   }, {
4983     "es_HK",
4984     "es_Latn_HK",
4985     "es_HK"
4986   }, {
4987     "es_AQ",
4988     "es_Latn_AQ",
4989     "es_AQ"
4990   }, {
4991     "es_Zzzz",
4992     "es_Latn_ES",
4993     "es"
4994   }, {
4995     "es_Zzzz_ZZ",
4996     "es_Latn_ES",
4997     "es"
4998   }, {
4999     "es_Zzzz_CN",
5000     "es_Latn_CN",
5001     "es_CN"
5002   }, {
5003     "es_Zzzz_TW",
5004     "es_Latn_TW",
5005     "es_TW"
5006   }, {
5007     "es_Zzzz_HK",
5008     "es_Latn_HK",
5009     "es_HK"
5010   }, {
5011     "es_Zzzz_AQ",
5012     "es_Latn_AQ",
5013     "es_AQ"
5014   }, {
5015     "es_Latn",
5016     "es_Latn_ES",
5017     "es"
5018   }, {
5019     "es_Latn_ZZ",
5020     "es_Latn_ES",
5021     "es"
5022   }, {
5023     "es_Latn_CN",
5024     "es_Latn_CN",
5025     "es_CN"
5026   }, {
5027     "es_Latn_TW",
5028     "es_Latn_TW",
5029     "es_TW"
5030   }, {
5031     "es_Latn_HK",
5032     "es_Latn_HK",
5033     "es_HK"
5034   }, {
5035     "es_Latn_AQ",
5036     "es_Latn_AQ",
5037     "es_AQ"
5038   }, {
5039     "es_Hans",
5040     "es_Hans_ES",
5041     "es_Hans"
5042   }, {
5043     "es_Hans_ZZ",
5044     "es_Hans_ES",
5045     "es_Hans"
5046   }, {
5047     "es_Hans_CN",
5048     "es_Hans_CN",
5049     "es_Hans_CN"
5050   }, {
5051     "es_Hans_TW",
5052     "es_Hans_TW",
5053     "es_Hans_TW"
5054   }, {
5055     "es_Hans_HK",
5056     "es_Hans_HK",
5057     "es_Hans_HK"
5058   }, {
5059     "es_Hans_AQ",
5060     "es_Hans_AQ",
5061     "es_Hans_AQ"
5062   }, {
5063     "es_Hant",
5064     "es_Hant_ES",
5065     "es_Hant"
5066   }, {
5067     "es_Hant_ZZ",
5068     "es_Hant_ES",
5069     "es_Hant"
5070   }, {
5071     "es_Hant_CN",
5072     "es_Hant_CN",
5073     "es_Hant_CN"
5074   }, {
5075     "es_Hant_TW",
5076     "es_Hant_TW",
5077     "es_Hant_TW"
5078   }, {
5079     "es_Hant_HK",
5080     "es_Hant_HK",
5081     "es_Hant_HK"
5082   }, {
5083     "es_Hant_AQ",
5084     "es_Hant_AQ",
5085     "es_Hant_AQ"
5086   }, {
5087     "es_Moon",
5088     "es_Moon_ES",
5089     "es_Moon"
5090   }, {
5091     "es_Moon_ZZ",
5092     "es_Moon_ES",
5093     "es_Moon"
5094   }, {
5095     "es_Moon_CN",
5096     "es_Moon_CN",
5097     "es_Moon_CN"
5098   }, {
5099     "es_Moon_TW",
5100     "es_Moon_TW",
5101     "es_Moon_TW"
5102   }, {
5103     "es_Moon_HK",
5104     "es_Moon_HK",
5105     "es_Moon_HK"
5106   }, {
5107     "es_Moon_AQ",
5108     "es_Moon_AQ",
5109     "es_Moon_AQ"
5110   }, {
5111     "zh",
5112     "zh_Hans_CN",
5113     "zh"
5114   }, {
5115     "zh_ZZ",
5116     "zh_Hans_CN",
5117     "zh"
5118   }, {
5119     "zh_CN",
5120     "zh_Hans_CN",
5121     "zh"
5122   }, {
5123     "zh_TW",
5124     "zh_Hant_TW",
5125     "zh_TW"
5126   }, {
5127     "zh_HK",
5128     "zh_Hant_HK",
5129     "zh_HK"
5130   }, {
5131     "zh_AQ",
5132     "zh_Hans_AQ",
5133     "zh_AQ"
5134   }, {
5135     "zh_Zzzz",
5136     "zh_Hans_CN",
5137     "zh"
5138   }, {
5139     "zh_Zzzz_ZZ",
5140     "zh_Hans_CN",
5141     "zh"
5142   }, {
5143     "zh_Zzzz_CN",
5144     "zh_Hans_CN",
5145     "zh"
5146   }, {
5147     "zh_Zzzz_TW",
5148     "zh_Hant_TW",
5149     "zh_TW"
5150   }, {
5151     "zh_Zzzz_HK",
5152     "zh_Hant_HK",
5153     "zh_HK"
5154   }, {
5155     "zh_Zzzz_AQ",
5156     "zh_Hans_AQ",
5157     "zh_AQ"
5158   }, {
5159     "zh_Latn",
5160     "zh_Latn_CN",
5161     "zh_Latn"
5162   }, {
5163     "zh_Latn_ZZ",
5164     "zh_Latn_CN",
5165     "zh_Latn"
5166   }, {
5167     "zh_Latn_CN",
5168     "zh_Latn_CN",
5169     "zh_Latn"
5170   }, {
5171     "zh_Latn_TW",
5172     "zh_Latn_TW",
5173     "zh_Latn_TW"
5174   }, {
5175     "zh_Latn_HK",
5176     "zh_Latn_HK",
5177     "zh_Latn_HK"
5178   }, {
5179     "zh_Latn_AQ",
5180     "zh_Latn_AQ",
5181     "zh_Latn_AQ"
5182   }, {
5183     "zh_Hans",
5184     "zh_Hans_CN",
5185     "zh"
5186   }, {
5187     "zh_Hans_ZZ",
5188     "zh_Hans_CN",
5189     "zh"
5190   }, {
5191     "zh_Hans_TW",
5192     "zh_Hans_TW",
5193     "zh_Hans_TW"
5194   }, {
5195     "zh_Hans_HK",
5196     "zh_Hans_HK",
5197     "zh_Hans_HK"
5198   }, {
5199     "zh_Hans_AQ",
5200     "zh_Hans_AQ",
5201     "zh_AQ"
5202   }, {
5203     "zh_Hant",
5204     "zh_Hant_TW",
5205     "zh_TW"
5206   }, {
5207     "zh_Hant_ZZ",
5208     "zh_Hant_TW",
5209     "zh_TW"
5210   }, {
5211     "zh_Hant_CN",
5212     "zh_Hant_CN",
5213     "zh_Hant_CN"
5214   }, {
5215     "zh_Hant_AQ",
5216     "zh_Hant_AQ",
5217     "zh_Hant_AQ"
5218   }, {
5219     "zh_Moon",
5220     "zh_Moon_CN",
5221     "zh_Moon"
5222   }, {
5223     "zh_Moon_ZZ",
5224     "zh_Moon_CN",
5225     "zh_Moon"
5226   }, {
5227     "zh_Moon_CN",
5228     "zh_Moon_CN",
5229     "zh_Moon"
5230   }, {
5231     "zh_Moon_TW",
5232     "zh_Moon_TW",
5233     "zh_Moon_TW"
5234   }, {
5235     "zh_Moon_HK",
5236     "zh_Moon_HK",
5237     "zh_Moon_HK"
5238   }, {
5239     "zh_Moon_AQ",
5240     "zh_Moon_AQ",
5241     "zh_Moon_AQ"
5242   }, {
5243     "art",
5244     "",
5245     ""
5246   }, {
5247     "art_ZZ",
5248     "",
5249     ""
5250   }, {
5251     "art_CN",
5252     "",
5253     ""
5254   }, {
5255     "art_TW",
5256     "",
5257     ""
5258   }, {
5259     "art_HK",
5260     "",
5261     ""
5262   }, {
5263     "art_AQ",
5264     "",
5265     ""
5266   }, {
5267     "art_Zzzz",
5268     "",
5269     ""
5270   }, {
5271     "art_Zzzz_ZZ",
5272     "",
5273     ""
5274   }, {
5275     "art_Zzzz_CN",
5276     "",
5277     ""
5278   }, {
5279     "art_Zzzz_TW",
5280     "",
5281     ""
5282   }, {
5283     "art_Zzzz_HK",
5284     "",
5285     ""
5286   }, {
5287     "art_Zzzz_AQ",
5288     "",
5289     ""
5290   }, {
5291     "art_Latn",
5292     "",
5293     ""
5294   }, {
5295     "art_Latn_ZZ",
5296     "",
5297     ""
5298   }, {
5299     "art_Latn_CN",
5300     "",
5301     ""
5302   }, {
5303     "art_Latn_TW",
5304     "",
5305     ""
5306   }, {
5307     "art_Latn_HK",
5308     "",
5309     ""
5310   }, {
5311     "art_Latn_AQ",
5312     "",
5313     ""
5314   }, {
5315     "art_Hans",
5316     "",
5317     ""
5318   }, {
5319     "art_Hans_ZZ",
5320     "",
5321     ""
5322   }, {
5323     "art_Hans_CN",
5324     "",
5325     ""
5326   }, {
5327     "art_Hans_TW",
5328     "",
5329     ""
5330   }, {
5331     "art_Hans_HK",
5332     "",
5333     ""
5334   }, {
5335     "art_Hans_AQ",
5336     "",
5337     ""
5338   }, {
5339     "art_Hant",
5340     "",
5341     ""
5342   }, {
5343     "art_Hant_ZZ",
5344     "",
5345     ""
5346   }, {
5347     "art_Hant_CN",
5348     "",
5349     ""
5350   }, {
5351     "art_Hant_TW",
5352     "",
5353     ""
5354   }, {
5355     "art_Hant_HK",
5356     "",
5357     ""
5358   }, {
5359     "art_Hant_AQ",
5360     "",
5361     ""
5362   }, {
5363     "art_Moon",
5364     "",
5365     ""
5366   }, {
5367     "art_Moon_ZZ",
5368     "",
5369     ""
5370   }, {
5371     "art_Moon_CN",
5372     "",
5373     ""
5374   }, {
5375     "art_Moon_TW",
5376     "",
5377     ""
5378   }, {
5379     "art_Moon_HK",
5380     "",
5381     ""
5382   }, {
5383     "art_Moon_AQ",
5384     "",
5385     ""
5386   }, {
5387     "de@collation=phonebook",
5388     "de_Latn_DE@collation=phonebook",
5389     "de@collation=phonebook"
5390   }
5391 };
5392 
5393 typedef struct errorDataTag {
5394     const char* tag;
5395     const char* expected;
5396     UErrorCode uerror;
5397     int32_t  bufferSize;
5398 } errorData;
5399 
5400 const errorData maximizeErrors[] = {
5401     {
5402         "enfueiujhytdf",
5403         NULL,
5404         U_ILLEGAL_ARGUMENT_ERROR,
5405         -1
5406     },
5407     {
5408         "en_THUJIOGIURJHGJFURYHFJGURYYYHHGJURHG",
5409         NULL,
5410         U_ILLEGAL_ARGUMENT_ERROR,
5411         -1
5412     },
5413     {
5414         "en_THUJIOGIURJHGJFURYHFJGURYYYHHGJURHG",
5415         NULL,
5416         U_ILLEGAL_ARGUMENT_ERROR,
5417         -1
5418     },
5419     {
5420         "en_Latn_US_POSIX@currency=EURO",
5421         "en_Latn_US_POSIX@currency=EURO",
5422         U_BUFFER_OVERFLOW_ERROR,
5423         29
5424     },
5425     {
5426         "en_Latn_US_POSIX@currency=EURO",
5427         "en_Latn_US_POSIX@currency=EURO",
5428         U_STRING_NOT_TERMINATED_WARNING,
5429         30
5430     }
5431 };
5432 
5433 const errorData minimizeErrors[] = {
5434     {
5435         "enfueiujhytdf",
5436         NULL,
5437         U_ILLEGAL_ARGUMENT_ERROR,
5438         -1
5439     },
5440     {
5441         "en_THUJIOGIURJHGJFURYHFJGURYYYHHGJURHG",
5442         NULL,
5443         U_ILLEGAL_ARGUMENT_ERROR,
5444         -1
5445     },
5446     {
5447         "en_Latn_US_POSIX@currency=EURO",
5448         "en__POSIX@currency=EURO",
5449         U_BUFFER_OVERFLOW_ERROR,
5450         22
5451     },
5452     {
5453         "en_Latn_US_POSIX@currency=EURO",
5454         "en__POSIX@currency=EURO",
5455         U_STRING_NOT_TERMINATED_WARNING,
5456         23
5457     }
5458 };
5459 
getExpectedReturnValue(const errorData * data)5460 static int32_t getExpectedReturnValue(const errorData* data)
5461 {
5462     if (data->uerror == U_BUFFER_OVERFLOW_ERROR ||
5463         data->uerror == U_STRING_NOT_TERMINATED_WARNING)
5464     {
5465         return strlen(data->expected);
5466     }
5467     else
5468     {
5469         return -1;
5470     }
5471 }
5472 
getBufferSize(const errorData * data,int32_t actualSize)5473 static int32_t getBufferSize(const errorData* data, int32_t actualSize)
5474 {
5475     if (data->expected == NULL)
5476     {
5477         return actualSize;
5478     }
5479     else if (data->bufferSize < 0)
5480     {
5481         return strlen(data->expected) + 1;
5482     }
5483     else
5484     {
5485         return data->bufferSize;
5486     }
5487 }
5488 
TestLikelySubtags()5489 static void TestLikelySubtags()
5490 {
5491     char buffer[ULOC_FULLNAME_CAPACITY + ULOC_KEYWORD_AND_VALUES_CAPACITY + 1];
5492     int32_t i = 0;
5493 
5494     for (; i < sizeof(basic_maximize_data) / sizeof(basic_maximize_data[0]); ++i)
5495     {
5496         UErrorCode status = U_ZERO_ERROR;
5497         const char* const minimal = basic_maximize_data[i][0];
5498         const char* const maximal = basic_maximize_data[i][1];
5499 
5500         /* const int32_t length = */
5501             uloc_addLikelySubtags(
5502                 minimal,
5503                 buffer,
5504                 sizeof(buffer),
5505                 &status);
5506         if (U_FAILURE(status)) {
5507             log_err_status(status, "  unexpected failure of uloc_addLikelySubtags(), minimal \"%s\" status %s\n", minimal, u_errorName(status));
5508             status = U_ZERO_ERROR;
5509         }
5510         else if (uprv_strlen(maximal) == 0) {
5511             if (uprv_stricmp(minimal, buffer) != 0) {
5512                 log_err("  unexpected maximal value \"%s\" in uloc_addLikelySubtags(), minimal \"%s\" = \"%s\"\n", maximal, minimal, buffer);
5513             }
5514         }
5515         else if (uprv_stricmp(maximal, buffer) != 0) {
5516             log_err("  maximal doesn't match expected %s in uloc_addLikelySubtags(), minimal \"%s\" = %s\n", maximal, minimal, buffer);
5517         }
5518     }
5519 
5520     for (i = 0; i < sizeof(basic_minimize_data) / sizeof(basic_minimize_data[0]); ++i) {
5521 
5522         UErrorCode status = U_ZERO_ERROR;
5523         const char* const maximal = basic_minimize_data[i][0];
5524         const char* const minimal = basic_minimize_data[i][1];
5525 
5526         /* const int32_t length = */
5527             uloc_minimizeSubtags(
5528                 maximal,
5529                 buffer,
5530                 sizeof(buffer),
5531                 &status);
5532 
5533         if (U_FAILURE(status)) {
5534             log_err_status(status, "  unexpected failure of uloc_MinimizeSubtags(), maximal \"%s\" status %s\n", maximal, u_errorName(status));
5535             status = U_ZERO_ERROR;
5536         }
5537         else if (uprv_strlen(minimal) == 0) {
5538             if (uprv_stricmp(maximal, buffer) != 0) {
5539                 log_err("  unexpected minimal value \"%s\" in uloc_minimizeSubtags(), maximal \"%s\" = \"%s\"\n", minimal, maximal, buffer);
5540             }
5541         }
5542         else if (uprv_stricmp(minimal, buffer) != 0) {
5543             log_err("  minimal doesn't match expected %s in uloc_MinimizeSubtags(), maximal \"%s\" = %s\n", minimal, maximal, buffer);
5544         }
5545     }
5546 
5547     for (i = 0; i < sizeof(full_data) / sizeof(full_data[0]); ++i) {
5548 
5549         UErrorCode status = U_ZERO_ERROR;
5550         const char* const minimal = full_data[i][0];
5551         const char* const maximal = full_data[i][1];
5552 
5553         /* const int32_t length = */
5554             uloc_addLikelySubtags(
5555                 minimal,
5556                 buffer,
5557                 sizeof(buffer),
5558                 &status);
5559         if (U_FAILURE(status)) {
5560             log_err_status(status, "  unexpected failure of uloc_addLikelySubtags(), minimal \"%s\" status \"%s\"\n", minimal, u_errorName(status));
5561             status = U_ZERO_ERROR;
5562         }
5563         else if (uprv_strlen(maximal) == 0) {
5564             if (uprv_stricmp(minimal, buffer) != 0) {
5565                 log_err("  unexpected maximal value \"%s\" in uloc_addLikelySubtags(), minimal \"%s\" = \"%s\"\n", maximal, minimal, buffer);
5566             }
5567         }
5568         else if (uprv_stricmp(maximal, buffer) != 0) {
5569             log_err("  maximal doesn't match expected \"%s\" in uloc_addLikelySubtags(), minimal \"%s\" = \"%s\"\n", maximal, minimal, buffer);
5570         }
5571     }
5572 
5573     for (i = 0; i < sizeof(full_data) / sizeof(full_data[0]); ++i) {
5574 
5575         UErrorCode status = U_ZERO_ERROR;
5576         const char* const maximal = full_data[i][1];
5577         const char* const minimal = full_data[i][2];
5578 
5579         if (strlen(maximal) > 0) {
5580 
5581             /* const int32_t length = */
5582                 uloc_minimizeSubtags(
5583                     maximal,
5584                     buffer,
5585                     sizeof(buffer),
5586                     &status);
5587 
5588             if (U_FAILURE(status)) {
5589                 log_err_status(status, "  unexpected failure of uloc_minimizeSubtags(), maximal \"%s\" status %s\n", maximal, u_errorName(status));
5590                 status = U_ZERO_ERROR;
5591             }
5592             else if (uprv_strlen(minimal) == 0) {
5593                 if (uprv_stricmp(maximal, buffer) != 0) {
5594                     log_err("  unexpected minimal value \"%s\" in uloc_minimizeSubtags(), maximal \"%s\" = \"%s\"\n", minimal, maximal, buffer);
5595                 }
5596             }
5597             else if (uprv_stricmp(minimal, buffer) != 0) {
5598                 log_err("  minimal doesn't match expected %s in uloc_MinimizeSubtags(), maximal \"%s\" = %s\n", minimal, maximal, buffer);
5599             }
5600         }
5601     }
5602 
5603     for (i = 0; i < sizeof(maximizeErrors) / sizeof(maximizeErrors[0]); ++i) {
5604 
5605         UErrorCode status = U_ZERO_ERROR;
5606         const char* const minimal = maximizeErrors[i].tag;
5607         const char* const maximal = maximizeErrors[i].expected;
5608         const UErrorCode expectedStatus = maximizeErrors[i].uerror;
5609         const int32_t expectedLength = getExpectedReturnValue(&maximizeErrors[i]);
5610         const int32_t bufferSize = getBufferSize(&maximizeErrors[i], sizeof(buffer));
5611 
5612         const int32_t length =
5613             uloc_addLikelySubtags(
5614                 minimal,
5615                 buffer,
5616                 bufferSize,
5617                 &status);
5618 
5619         if (status == U_ZERO_ERROR) {
5620             log_err("  unexpected U_ZERO_ERROR for uloc_addLikelySubtags(), minimal \"%s\" expected status %s\n", minimal, u_errorName(expectedStatus));
5621             status = U_ZERO_ERROR;
5622         }
5623         else if (status != expectedStatus) {
5624             log_err_status(status, "  unexpected status for uloc_addLikelySubtags(), minimal \"%s\" expected status %s, but got %s\n", minimal, u_errorName(expectedStatus), u_errorName(status));
5625         }
5626         else if (length != expectedLength) {
5627             log_err("  unexpected length for uloc_addLikelySubtags(), minimal \"%s\" expected length %d, but got %d\n", minimal, expectedLength, length);
5628         }
5629         else if (status == U_BUFFER_OVERFLOW_ERROR || status == U_STRING_NOT_TERMINATED_WARNING) {
5630             if (uprv_strnicmp(maximal, buffer, bufferSize) != 0) {
5631                 log_err("  maximal doesn't match expected %s in uloc_addLikelySubtags(), minimal \"%s\" = %*s\n",
5632                     maximal, minimal, (int)sizeof(buffer), buffer);
5633             }
5634         }
5635     }
5636 
5637     for (i = 0; i < sizeof(minimizeErrors) / sizeof(minimizeErrors[0]); ++i) {
5638 
5639         UErrorCode status = U_ZERO_ERROR;
5640         const char* const maximal = minimizeErrors[i].tag;
5641         const char* const minimal = minimizeErrors[i].expected;
5642         const UErrorCode expectedStatus = minimizeErrors[i].uerror;
5643         const int32_t expectedLength = getExpectedReturnValue(&minimizeErrors[i]);
5644         const int32_t bufferSize = getBufferSize(&minimizeErrors[i], sizeof(buffer));
5645 
5646         const int32_t length =
5647             uloc_minimizeSubtags(
5648                 maximal,
5649                 buffer,
5650                 bufferSize,
5651                 &status);
5652 
5653         if (status == U_ZERO_ERROR) {
5654             log_err("  unexpected U_ZERO_ERROR for uloc_minimizeSubtags(), maximal \"%s\" expected status %s\n", maximal, u_errorName(expectedStatus));
5655             status = U_ZERO_ERROR;
5656         }
5657         else if (status != expectedStatus) {
5658             log_err_status(status, "  unexpected status for uloc_minimizeSubtags(), maximal \"%s\" expected status %s, but got %s\n", maximal, u_errorName(expectedStatus), u_errorName(status));
5659         }
5660         else if (length != expectedLength) {
5661             log_err("  unexpected length for uloc_minimizeSubtags(), maximal \"%s\" expected length %d, but got %d\n", maximal, expectedLength, length);
5662         }
5663         else if (status == U_BUFFER_OVERFLOW_ERROR || status == U_STRING_NOT_TERMINATED_WARNING) {
5664             if (uprv_strnicmp(minimal, buffer, bufferSize) != 0) {
5665                 log_err("  minimal doesn't match expected \"%s\" in uloc_minimizeSubtags(), minimal \"%s\" = \"%*s\"\n",
5666                     minimal, maximal, (int)sizeof(buffer), buffer);
5667             }
5668         }
5669     }
5670 }
5671 
5672 const char* const locale_to_langtag[][3] = {
5673     {"@x=elmer",    "x-elmer",      "x-elmer"},
5674     {"",            "und",          "und"},
5675     {"en",          "en",           "en"},
5676     {"en_US",       "en-US",        "en-US"},
5677     {"iw_IL",       "he-IL",        "he-IL"},
5678     {"sr_Latn_SR",  "sr-Latn-SR",   "sr-Latn-SR"},
5679     {"en__POSIX",   "en-u-va-posix", "en-u-va-posix"},
5680     {"en_POSIX",    "en-u-va-posix", "en-u-va-posix"},
5681     {"en_US_POSIX_VAR", "en-US-posix-x-lvariant-var", NULL},  /* variant POSIX_VAR is processed as regular variant */
5682     {"en_US_VAR_POSIX", "en-US-x-lvariant-var-posix", NULL},  /* variant VAR_POSIX is processed as regular variant */
5683     {"en_US_POSIX@va=posix2",   "en-US-u-va-posix2",  "en-US-u-va-posix2"},           /* if keyword va=xxx already exists, variant POSIX is simply dropped */
5684     {"en_US_POSIX@ca=japanese",  "en-US-u-ca-japanese-va-posix", "en-US-u-ca-japanese-va-posix"},
5685     {"und_555",     "und-555",      "und-555"},
5686     {"123",         "und",          NULL},
5687     {"%$#&",        "und",          NULL},
5688     {"_Latn",       "und-Latn",     "und-Latn"},
5689     {"_DE",         "und-DE",       "und-DE"},
5690     {"und_FR",      "und-FR",       "und-FR"},
5691     {"th_TH_TH",    "th-TH-x-lvariant-th", NULL},
5692     {"bogus",       "bogus",        "bogus"},
5693     {"foooobarrr",  "und",          NULL},
5694     {"az_AZ_CYRL",  "az-Cyrl-AZ",   "az-Cyrl-AZ"},
5695     {"aa_BB_CYRL",  "aa-BB-x-lvariant-cyrl", NULL},
5696     {"en_US_1234",  "en-US-1234",   "en-US-1234"},
5697     {"en_US_VARIANTA_VARIANTB", "en-US-varianta-variantb",  "en-US-varianta-variantb"},
5698     {"ja__9876_5432",   "ja-9876-5432", "ja-9876-5432"},
5699     {"zh_Hant__VAR",    "zh-Hant-x-lvariant-var", NULL},
5700     {"es__BADVARIANT_GOODVAR",  "es-goodvar",   NULL},
5701     {"en@calendar=gregorian",   "en-u-ca-gregory",  "en-u-ca-gregory"},
5702     {"de@collation=phonebook;calendar=gregorian",   "de-u-ca-gregory-co-phonebk",   "de-u-ca-gregory-co-phonebk"},
5703     {"th@numbers=thai;z=extz;x=priv-use;a=exta",   "th-a-exta-u-nu-thai-z-extz-x-priv-use", "th-a-exta-u-nu-thai-z-extz-x-priv-use"},
5704     {"en@timezone=America/New_York;calendar=japanese",    "en-u-ca-japanese-tz-usnyc",    "en-u-ca-japanese-tz-usnyc"},
5705     {"en@timezone=US/Eastern",  "en-u-tz-usnyc",    "en-u-tz-usnyc"},
5706     {"en@x=x-y-z;a=a-b-c",  "en-x-x-y-z",   NULL},
5707     {"it@collation=badcollationtype;colStrength=identical;cu=usd-eur", "it-u-ks-identic",  NULL},
5708     {"en_US_POSIX", "en-US-u-va-posix", "en-US-u-va-posix"},
5709     {"en_US_POSIX@calendar=japanese;currency=EUR","en-US-u-ca-japanese-cu-EUR-va-posix", "en-US-u-ca-japanese-cu-EUR-va-posix"},
5710     {"@x=elmer",    "x-elmer",      "x-elmer"},
5711     {"en@x=elmer",  "en-x-elmer",   "en-x-elmer"},
5712     {"@x=elmer;a=exta", "und-a-exta-x-elmer",   "und-a-exta-x-elmer"},
5713     {"en_US@attribute=attr1-attr2;calendar=gregorian", "en-US-u-attr1-attr2-ca-gregory", "en-US-u-attr1-attr2-ca-gregory"},
5714     {NULL,          NULL,           NULL}
5715 };
5716 
TestToLanguageTag(void)5717 static void TestToLanguageTag(void) {
5718     char langtag[256];
5719     int32_t i;
5720     UErrorCode status;
5721     int32_t len;
5722     const char *inloc;
5723     const char *expected;
5724 
5725     for (i = 0; locale_to_langtag[i][0] != NULL; i++) {
5726         inloc = locale_to_langtag[i][0];
5727 
5728         /* testing non-strict mode */
5729         status = U_ZERO_ERROR;
5730         langtag[0] = 0;
5731         expected = locale_to_langtag[i][1];
5732 
5733         len = uloc_toLanguageTag(inloc, langtag, sizeof(langtag), FALSE, &status);
5734         (void)len;    /* Suppress set but not used warning. */
5735         if (U_FAILURE(status)) {
5736             if (expected != NULL) {
5737                 log_err("Error returned by uloc_toLanguageTag for locale id [%s] - error: %s\n",
5738                     inloc, u_errorName(status));
5739             }
5740         } else {
5741             if (expected == NULL) {
5742                 log_err("Error should be returned by uloc_toLanguageTag for locale id [%s], but [%s] is returned without errors\n",
5743                     inloc, langtag);
5744             } else if (uprv_strcmp(langtag, expected) != 0) {
5745                 log_data_err("uloc_toLanguageTag returned language tag [%s] for input locale [%s] - expected: [%s]. Are you missing data?\n",
5746                     langtag, inloc, expected);
5747             }
5748         }
5749 
5750         /* testing strict mode */
5751         status = U_ZERO_ERROR;
5752         langtag[0] = 0;
5753         expected = locale_to_langtag[i][2];
5754 
5755         len = uloc_toLanguageTag(inloc, langtag, sizeof(langtag), TRUE, &status);
5756         if (U_FAILURE(status)) {
5757             if (expected != NULL) {
5758                 log_data_err("Error returned by uloc_toLanguageTag {strict} for locale id [%s] - error: %s Are you missing data?\n",
5759                     inloc, u_errorName(status));
5760             }
5761         } else {
5762             if (expected == NULL) {
5763                 log_err("Error should be returned by uloc_toLanguageTag {strict} for locale id [%s], but [%s] is returned without errors\n",
5764                     inloc, langtag);
5765             } else if (uprv_strcmp(langtag, expected) != 0) {
5766                 log_err("uloc_toLanguageTag {strict} returned language tag [%s] for input locale [%s] - expected: [%s]\n",
5767                     langtag, inloc, expected);
5768             }
5769         }
5770     }
5771 }
5772 
5773 #define FULL_LENGTH -1
5774 static const struct {
5775     const char  *bcpID;
5776     const char  *locID;
5777     int32_t     len;
5778 } langtag_to_locale[] = {
5779     {"en",                  "en",                   FULL_LENGTH},
5780     {"en-us",               "en_US",                FULL_LENGTH},
5781     {"und-US",              "_US",                  FULL_LENGTH},
5782     {"und-latn",            "_Latn",                FULL_LENGTH},
5783     {"en-US-posix",         "en_US_POSIX",          FULL_LENGTH},
5784     {"de-de_euro",          "de",                   2},
5785     {"kok-IN",              "kok_IN",               FULL_LENGTH},
5786     {"123",                 "",                     0},
5787     {"en_us",               "",                     0},
5788     {"en-latn-x",           "en_Latn",              7},
5789     {"art-lojban",          "jbo",                  FULL_LENGTH},
5790     {"zh-hakka",            "hak",                  FULL_LENGTH},
5791     {"zh-cmn-CH",           "cmn_CH",               FULL_LENGTH},
5792     {"xxx-yy",              "xxx_YY",               FULL_LENGTH},
5793     {"fr-234",              "fr_234",               FULL_LENGTH},
5794     {"i-default",           "en@x=i-default",       FULL_LENGTH},
5795     {"i-test",              "",                     0},
5796     {"ja-jp-jp",            "ja_JP",                5},
5797     {"bogus",               "bogus",                FULL_LENGTH},
5798     {"boguslang",           "",                     0},
5799     {"EN-lATN-us",          "en_Latn_US",           FULL_LENGTH},
5800     {"und-variant-1234",    "__VARIANT_1234",       FULL_LENGTH},
5801     {"und-varzero-var1-vartwo", "__VARZERO",        11},
5802     {"en-u-ca-gregory",     "en@calendar=gregorian",    FULL_LENGTH},
5803     {"en-U-cu-USD",         "en@currency=usd",      FULL_LENGTH},
5804     {"en-US-u-va-posix",    "en_US_POSIX",          FULL_LENGTH},
5805     {"en-us-u-ca-gregory-va-posix", "en_US_POSIX@calendar=gregorian",   FULL_LENGTH},
5806     {"en-us-posix-u-va-posix",   "en_US_POSIX@va=posix",    FULL_LENGTH},
5807     {"en-us-u-va-posix2",        "en_US@va=posix2",         FULL_LENGTH},
5808     {"en-us-vari1-u-va-posix",   "en_US_VARI1@va=posix",    FULL_LENGTH},
5809     {"ar-x-1-2-3",          "ar@x=1-2-3",           FULL_LENGTH},
5810     {"fr-u-nu-latn-cu-eur", "fr@currency=eur;numbers=latn", FULL_LENGTH},
5811     {"de-k-kext-u-co-phonebk-nu-latn",  "de@collation=phonebook;k=kext;numbers=latn",   FULL_LENGTH},
5812     {"ja-u-cu-jpy-ca-jp",   "ja@calendar=yes;currency=jpy;jp=yes",  FULL_LENGTH},
5813     {"en-us-u-tz-usnyc",    "en_US@timezone=America/New_York",  FULL_LENGTH},
5814     {"und-a-abc-def",       "und@a=abc-def",        FULL_LENGTH},
5815     {"zh-u-ca-chinese-x-u-ca-chinese",  "zh@calendar=chinese;x=u-ca-chinese",   FULL_LENGTH},
5816     {"x-elmer",             "@x=elmer",             FULL_LENGTH},
5817     {"en-US-u-attr1-attr2-ca-gregory", "en_US@attribute=attr1-attr2;calendar=gregorian",    FULL_LENGTH},
5818     {"sr-u-kn",             "sr@colnumeric=yes",    FULL_LENGTH},
5819     {"de-u-kn-co-phonebk",  "de@collation=phonebook;colnumeric=yes",    FULL_LENGTH},
5820     {"en-u-attr2-attr1-kn-kb",  "en@attribute=attr1-attr2;colbackwards=yes;colnumeric=yes", FULL_LENGTH},
5821     {"ja-u-ijkl-efgh-abcd-ca-japanese-xx-yyy-zzz-kn",   "ja@attribute=abcd-efgh-ijkl;calendar=japanese;colnumeric=yes;xx=yyy-zzz",  FULL_LENGTH},
5822 
5823     {"de-u-xc-xphonebk-co-phonebk-ca-buddhist-mo-very-lo-extensi-xd-that-de-should-vc-probably-xz-killthebuffer",
5824      "de@calendar=buddhist;collation=phonebook;de=should;lo=extensi;mo=very;vc=probably;xc=xphonebk;xd=that;xz=yes", 91},
5825     {NULL,          NULL,           0}
5826 };
5827 
TestForLanguageTag(void)5828 static void TestForLanguageTag(void) {
5829     char locale[256];
5830     int32_t i;
5831     UErrorCode status;
5832     int32_t parsedLen;
5833     int32_t expParsedLen;
5834 
5835     for (i = 0; langtag_to_locale[i].bcpID != NULL; i++) {
5836         status = U_ZERO_ERROR;
5837         locale[0] = 0;
5838         expParsedLen = langtag_to_locale[i].len;
5839         if (expParsedLen == FULL_LENGTH) {
5840             expParsedLen = uprv_strlen(langtag_to_locale[i].bcpID);
5841         }
5842         uloc_forLanguageTag(langtag_to_locale[i].bcpID, locale, sizeof(locale), &parsedLen, &status);
5843         if (U_FAILURE(status)) {
5844             log_err_status(status, "Error returned by uloc_forLanguageTag for language tag [%s] - error: %s\n",
5845                 langtag_to_locale[i].bcpID, u_errorName(status));
5846         } else {
5847             if (uprv_strcmp(langtag_to_locale[i].locID, locale) != 0) {
5848                 log_err("uloc_forLanguageTag returned locale [%s] for input language tag [%s] - expected: [%s]\n",
5849                     locale, langtag_to_locale[i].bcpID, langtag_to_locale[i].locID);
5850             }
5851             if (parsedLen != expParsedLen) {
5852                 log_err("uloc_forLanguageTag parsed length of %d for input language tag [%s] - expected parsed length: %d\n",
5853                     parsedLen, langtag_to_locale[i].bcpID, expParsedLen);
5854             }
5855         }
5856     }
5857 }
5858 
test_unicode_define(const char * namech,char ch,const char * nameu,UChar uch)5859 static void test_unicode_define(const char *namech, char ch, const char *nameu, UChar uch)
5860 {
5861   UChar asUch[1];
5862   asUch[0]=0;
5863   log_verbose("Testing whether %s[\\x%02x,'%c'] == %s[U+%04X]\n", namech, ch,(int)ch, nameu, (int) uch);
5864   u_charsToUChars(&ch, asUch, 1);
5865   if(asUch[0] != uch) {
5866     log_err("FAIL:  %s[\\x%02x,'%c'] maps to U+%04X, but %s = U+%04X\n", namech, ch, (int)ch, (int)asUch[0], nameu, (int)uch);
5867   } else {
5868     log_verbose(" .. OK, == U+%04X\n", (int)asUch[0]);
5869   }
5870 }
5871 
5872 #define TEST_UNICODE_DEFINE(x,y) test_unicode_define(#x, (char)(x), #y, (UChar)(y))
5873 
TestUnicodeDefines(void)5874 static void TestUnicodeDefines(void) {
5875   TEST_UNICODE_DEFINE(ULOC_KEYWORD_SEPARATOR, ULOC_KEYWORD_SEPARATOR_UNICODE);
5876   TEST_UNICODE_DEFINE(ULOC_KEYWORD_ASSIGN, ULOC_KEYWORD_ASSIGN_UNICODE);
5877   TEST_UNICODE_DEFINE(ULOC_KEYWORD_ITEM_SEPARATOR, ULOC_KEYWORD_ITEM_SEPARATOR_UNICODE);
5878 }
5879