1 /**
2  * FreeRDP: A Remote Desktop Protocol Implementation
3  * Microsoft Locales
4  *
5  * Copyright 2009-2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
6  * Copyright 2021 Thincast Technologies GmbH
7  * Copyright 2021 Martin Fleisz <martin.fleisz@thincast.com>
8  *
9  * Licensed under the Apache License, Version 2.0 (the "License");
10  * you may not use this file except in compliance with the License.
11  * You may obtain a copy of the License at
12  *
13  *     http://www.apache.org/licenses/LICENSE-2.0
14  *
15  * Unless required by applicable law or agreed to in writing, software
16  * distributed under the License is distributed on an "AS IS" BASIS,
17  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18  * See the License for the specific language governing permissions and
19  * limitations under the License.
20  */
21 
22 #ifdef HAVE_CONFIG_H
23 #include "config.h"
24 #endif
25 
26 #if defined(__APPLE__)
27 #include <CoreFoundation/CFString.h>
28 #include <CoreFoundation/CFLocale.h>
29 #endif
30 
31 #include <stdio.h>
32 #include <stdlib.h>
33 #include <string.h>
34 
35 #include <winpr/crt.h>
36 #include <assert.h>
37 #include <winpr/environment.h>
38 
39 #include "liblocale.h"
40 
41 #include <freerdp/locale/locale.h>
42 
43 #define LOCALE_LANGUAGE_LEN 6
44 #define LOCALE_COUNTRY_LEN 10
45 
46 struct _SYSTEM_LOCALE
47 {
48 	char language[LOCALE_LANGUAGE_LEN]; /* Two or three letter language code */
49 	char country[LOCALE_COUNTRY_LEN];   /* Two or three letter country code (Sometimes with Cyrl_
50 	                                       prefix) */
51 	DWORD code;                         /* 32-bit unsigned integer corresponding to the locale */
52 };
53 typedef struct _SYSTEM_LOCALE SYSTEM_LOCALE;
54 
55 /*
56  * Refer to MSDN article "Locale Identifier Constants and Strings":
57  * http://msdn.microsoft.com/en-us/library/ms776260.aspx
58  */
59 
60 static const SYSTEM_LOCALE SYSTEM_LOCALE_TABLE[] = {
61 	{ "af", "ZA", AFRIKAANS },           /* Afrikaans (South Africa) */
62 	{ "sq", "AL", ALBANIAN },            /* Albanian (Albania) */
63 	{ "gsw", "FR", ALSATIAN },           /* Windows Vista and later: Alsatian (France) */
64 	{ "am", "ET", AMHARIC },             /* Windows Vista and later: Amharic (Ethiopia) */
65 	{ "ar", "DZ", ARABIC_ALGERIA },      /* Arabic (Algeria) */
66 	{ "ar", "BH", ARABIC_BAHRAIN },      /* Arabic (Bahrain) */
67 	{ "ar", "EG", ARABIC_EGYPT },        /* Arabic (Egypt) */
68 	{ "ar", "IQ", ARABIC_IRAQ },         /* Arabic (Iraq) */
69 	{ "ar", "JO", ARABIC_JORDAN },       /* Arabic (Jordan) */
70 	{ "ar", "KW", ARABIC_KUWAIT },       /* Arabic (Kuwait) */
71 	{ "ar", "LB", ARABIC_LEBANON },      /* Arabic (Lebanon) */
72 	{ "ar", "LY", ARABIC_LIBYA },        /* Arabic (Libya) */
73 	{ "ar", "MA", ARABIC_MOROCCO },      /* Arabic (Morocco) */
74 	{ "ar", "OM", ARABIC_OMAN },         /* Arabic (Oman) */
75 	{ "ar", "QA", ARABIC_QATAR },        /* Arabic (Qatar) */
76 	{ "ar", "SA", ARABIC_SAUDI_ARABIA }, /* Arabic (Saudi Arabia) */
77 	{ "ar", "SY", ARABIC_SYRIA },        /* Arabic (Syria) */
78 	{ "ar", "TN", ARABIC_TUNISIA },      /* Arabic (Tunisia) */
79 	{ "ar", "AE", ARABIC_UAE },          /* Arabic (U.A.E.) */
80 	{ "ar", "YE", ARABIC_YEMEN },        /* Arabic (Yemen) */
81 	{ "az", "AZ", AZERI_LATIN },         /* Azeri (Latin) */
82 	{ "az", "Cyrl_AZ", AZERI_CYRILLIC }, /* Azeri (Cyrillic) */
83 	{ "hy", "AM", ARMENIAN },            /* Windows 2000 and later: Armenian (Armenia) */
84 	{ "as", "IN", ASSAMESE },            /* Windows Vista and later: Assamese (India) */
85 	{ "ba", "RU", BASHKIR },             /* Windows Vista and later: Bashkir (Russia) */
86 	{ "eu", "ES", BASQUE },              /* Basque (Basque) */
87 	{ "be", "BY", BELARUSIAN },          /* Belarusian (Belarus) */
88 	{ "bn", "IN", BENGALI_INDIA },       /* Windows XP SP2 and later: Bengali (India) */
89 	{ "br", "FR", BRETON },              /* Breton (France) */
90 	{ "bs", "BA", BOSNIAN_LATIN },       /* Bosnian (Latin) */
91 	{ "bg", "BG", BULGARIAN },           /* Bulgarian (Bulgaria) */
92 	{ "ca", "ES", CATALAN },             /* Catalan (Catalan) */
93 	{ "zh", "HK", CHINESE_HONG_KONG },   /* Chinese (Hong Kong SAR, PRC) */
94 	{ "zh", "MO", CHINESE_MACAU }, /* Windows 98/Me, Windows XP and later: Chinese (Macao SAR) */
95 	{ "zh", "CN", CHINESE_PRC },   /* Chinese (PRC) */
96 	{ "zh", "SG", CHINESE_SINGAPORE },           /* Chinese (Singapore) */
97 	{ "zh", "TW", CHINESE_TAIWAN },              /* Chinese (Taiwan) */
98 	{ "hr", "BA", CROATIAN_BOSNIA_HERZEGOVINA }, /* Windows XP SP2 and later: Croatian (Bosnia and
99 	                                                Herzegovina, Latin) */
100 	{ "hr", "HR", CROATIAN },                    /* Croatian (Croatia) */
101 	{ "cs", "CZ", CZECH },                       /* Czech (Czech Republic) */
102 	{ "da", "DK", DANISH },                      /* Danish (Denmark) */
103 	{ "prs", "AF", DARI },                       /* Windows XP and later: Dari (Afghanistan) */
104 	{ "dv", "MV", DIVEHI },                      /* Windows XP and later: Divehi (Maldives) */
105 	{ "nl", "BE", DUTCH_BELGIAN },               /* Dutch (Belgium) */
106 	{ "nl", "NL", DUTCH_STANDARD },              /* Dutch (Netherlands) */
107 	{ "en", "AU", ENGLISH_AUSTRALIAN },          /* English (Australia) */
108 	{ "en", "BZ", ENGLISH_BELIZE },              /* English (Belize) */
109 	{ "en", "CA", ENGLISH_CANADIAN },            /* English (Canada) */
110 	{ "en", "CB", ENGLISH_CARIBBEAN },           /* English (Carribean) */
111 	{ "en", "IN", ENGLISH_INDIA },               /* Windows Vista and later: English (India) */
112 	{ "en", "IE", ENGLISH_IRELAND },             /* English (Ireland) */
113 	{ "en", "JM", ENGLISH_JAMAICA },             /* English (Jamaica) */
114 	{ "en", "MY", ENGLISH_MALAYSIA },            /* Windows Vista and later: English (Malaysia) */
115 	{ "en", "NZ", ENGLISH_NEW_ZEALAND },         /* English (New Zealand) */
116 	{ "en", "PH",
117 	  ENGLISH_PHILIPPINES }, /* Windows 98/Me, Windows 2000 and later: English (Philippines) */
118 	{ "en", "SG", ENGLISH_SINGAPORE },      /* Windows Vista and later: English (Singapore) */
119 	{ "en", "ZA", ENGLISH_SOUTH_AFRICA },   /* English (South Africa) */
120 	{ "en", "TT", ENGLISH_TRINIDAD },       /* English (Trinidad and Tobago) */
121 	{ "en", "GB", ENGLISH_UNITED_KINGDOM }, /* English (United Kingdom) */
122 	{ "en", "US", ENGLISH_UNITED_STATES },  /* English (United States) */
123 	{ "en", "ZW",
124 	  ENGLISH_ZIMBABWE },      /* Windows 98/Me, Windows 2000 and later: English (Zimbabwe) */
125 	{ "et", "EE", ESTONIAN },  /* Estonian (Estonia) */
126 	{ "fo", "FO", FAEROESE },  /* Faroese (Faroe Islands) */
127 	{ "fil", "PH", FILIPINO }, /* Windows XP SP2 and later (downloadable); Windows Vista and later:
128 	                              Filipino (Philippines) */
129 	{ "fi", "FI", FINNISH },   /* Finnish (Finland) */
130 	{ "fr", "BE", FRENCH_BELGIAN },    /* French (Belgium) */
131 	{ "fr", "CA", FRENCH_CANADIAN },   /* French (Canada) */
132 	{ "fr", "FR", FRENCH_STANDARD },   /* French (France) */
133 	{ "fr", "LU", FRENCH_LUXEMBOURG }, /* French (Luxembourg) */
134 	{ "fr", "MC", FRENCH_MONACO },     /* French (Monaco) */
135 	{ "fr", "CH", FRENCH_SWISS },      /* French (Switzerland) */
136 	{ "fy", "NL", FRISIAN },  /* Windows XP SP2 and later (downloadable); Windows Vista and later:
137 	                             Frisian (Netherlands) */
138 	{ "gl", "ES", GALICIAN }, /* Windows XP and later: Galician (Spain) */
139 	{ "ka", "GE", GEORGIAN }, /* Windows 2000 and later: Georgian (Georgia) */
140 	{ "de", "AT", GERMAN_AUSTRIAN },      /* German (Austria) */
141 	{ "de", "DE", GERMAN_STANDARD },      /* German (Germany) */
142 	{ "de", "LI", GERMAN_LIECHTENSTEIN }, /* German (Liechtenstein) */
143 	{ "de", "LU", GERMAN_LUXEMBOURG },    /* German (Luxembourg) */
144 	{ "de", "CH", GERMAN_SWISS },         /* German (Switzerland) */
145 	{ "el", "GR", GREEK },                /* Greek (Greece) */
146 	{ "kl", "GL", GREENLANDIC },          /* Windows Vista and later: Greenlandic (Greenland) */
147 	{ "gu", "IN", GUJARATI },             /* Windows XP and later: Gujarati (India) */
148 	{ "he", "IL", HEBREW },               /* Hebrew (Israel) */
149 	{ "hi", "IN", HINDI },                /* Windows 2000 and later: Hindi (India) */
150 	{ "hu", "HU", HUNGARIAN },            /* Hungarian (Hungary) */
151 	{ "is", "IS", ICELANDIC },            /* Icelandic (Iceland) */
152 	{ "ig", "NG", IGBO },                 /* Igbo (Nigeria) */
153 	{ "id", "ID", INDONESIAN },           /* Indonesian (Indonesia) */
154 	{ "ga", "IE", IRISH }, /* Windows XP SP2 and later (downloadable); Windows Vista and later:
155 	                          Irish (Ireland) */
156 	{ "it", "IT", ITALIAN_STANDARD }, /* Italian (Italy) */
157 	{ "it", "CH", ITALIAN_SWISS },    /* Italian (Switzerland) */
158 	{ "ja", "JP", JAPANESE },         /* Japanese (Japan) */
159 	{ "kn", "IN", KANNADA },          /* Windows XP and later: Kannada (India) */
160 	{ "kk", "KZ", KAZAKH },           /* Windows 2000 and later: Kazakh (Kazakhstan) */
161 	{ "kh", "KH", KHMER },            /* Windows Vista and later: Khmer (Cambodia) */
162 	{ "qut", "GT", KICHE },           /* Windows Vista and later: K'iche (Guatemala) */
163 	{ "rw", "RW", KINYARWANDA },      /* Windows Vista and later: Kinyarwanda (Rwanda) */
164 	{ "kok", "IN", KONKANI },         /* Windows 2000 and later: Konkani (India) */
165 	{ "ko", "KR", KOREAN },           /* Korean (Korea) */
166 	{ "ky", "KG", KYRGYZ },           /* Windows XP and later: Kyrgyz (Kyrgyzstan) */
167 	{ "lo", "LA", LAO },              /* Windows Vista and later: Lao (Lao PDR) */
168 	{ "lv", "LV", LATVIAN },          /* Latvian (Latvia) */
169 	{ "lt", "LT", LITHUANIAN },       /* Lithuanian (Lithuania) */
170 	{ "dsb", "DE", LOWER_SORBIAN },   /* Windows Vista and later: Lower Sorbian (Germany) */
171 	{ "lb", "LU", LUXEMBOURGISH },    /* Windows XP SP2 and later (downloadable); Windows Vista and
172 	                                     later: Luxembourgish (Luxembourg) */
173 	{ "mk", "MK", MACEDONIAN },       /* Windows 2000 and later: Macedonian (Macedonia, FYROM) */
174 	{ "ms", "BN", MALAY_BRUNEI_DARUSSALAM }, /* Windows 2000 and later: Malay (Brunei Darussalam) */
175 	{ "ms", "MY", MALAY_MALAYSIA },          /* Windows 2000 and later: Malay (Malaysia) */
176 	{ "ml", "IN", MALAYALAM },               /* Windows XP SP2 and later: Malayalam (India) */
177 	{ "mt", "MT", MALTESE },                 /* Windows XP SP2 and later: Maltese (Malta) */
178 	{ "mi", "NZ", MAORI },                   /* Windows XP SP2 and later: Maori (New Zealand) */
179 	{ "arn", "CL", MAPUDUNGUN }, /* Windows XP SP2 and later (downloadable); Windows Vista and
180 	                                later: Mapudungun (Chile) */
181 	{ "mr", "IN", MARATHI },     /* Windows 2000 and later: Marathi (India) */
182 	{ "moh", "CA", MOHAWK },   /* Windows XP SP2 and later (downloadable); Windows Vista and later:
183 	                              Mohawk (Canada) */
184 	{ "mn", "MN", MONGOLIAN }, /* Mongolian */
185 	{ "ne", "NP", NEPALI },    /* Windows XP SP2 and later (downloadable); Windows Vista and later:
186 	                              Nepali (Nepal) */
187 	{ "nb", "NO", NORWEGIAN_BOKMAL },  /* Norwegian (Bokmal, Norway) */
188 	{ "nn", "NO", NORWEGIAN_NYNORSK }, /* Norwegian (Nynorsk, Norway) */
189 	{ "oc", "FR", OCCITAN },           /* Occitan (France) */
190 	{ "or", "IN", ORIYA },             /* Oriya (India) */
191 	{ "ps", "AF", PASHTO }, /* Windows XP SP2 and later (downloadable); Windows Vista and later:
192 	                           Pashto (Afghanistan) */
193 	{ "fa", "IR", FARSI },  /* Persian (Iran) */
194 	{ "pl", "PL", POLISH }, /* Polish (Poland) */
195 	{ "pt", "BR", PORTUGUESE_BRAZILIAN }, /* Portuguese (Brazil) */
196 	{ "pt", "PT", PORTUGUESE_STANDARD },  /* Portuguese (Portugal) */
197 	{ "pa", "IN", PUNJABI },              /* Windows XP and later: Punjabi (India) */
198 	{ "quz", "BO", QUECHUA_BOLIVIA },     /* Windows XP SP2 and later: Quechua (Bolivia) */
199 	{ "quz", "EC", QUECHUA_ECUADOR },     /* Windows XP SP2 and later: Quechua (Ecuador) */
200 	{ "quz", "PE", QUECHUA_PERU },        /* Windows XP SP2 and later: Quechua (Peru) */
201 	{ "ro", "RO", ROMANIAN },             /* Romanian (Romania) */
202 	{ "rm", "CH", ROMANSH }, /* Windows XP SP2 and later (downloadable); Windows Vista and later:
203 	                            Romansh (Switzerland) */
204 	{ "ru", "RU", RUSSIAN }, /* Russian (Russia) */
205 	{ "smn", "FI", SAMI_INARI },           /* Windows XP SP2 and later: Sami (Inari, Finland) */
206 	{ "smj", "NO", SAMI_LULE_NORWAY },     /* Windows XP SP2 and later: Sami (Lule, Norway) */
207 	{ "smj", "SE", SAMI_LULE_SWEDEN },     /* Windows XP SP2 and later: Sami (Lule, Sweden) */
208 	{ "se", "FI", SAMI_NORTHERN_FINLAND }, /* Windows XP SP2 and later: Sami (Northern, Finland) */
209 	{ "se", "NO", SAMI_NORTHERN_NORWAY },  /* Windows XP SP2 and later: Sami (Northern, Norway) */
210 	{ "se", "SE", SAMI_NORTHERN_SWEDEN },  /* Windows XP SP2 and later: Sami (Northern, Sweden) */
211 	{ "sms", "FI", SAMI_SKOLT },           /* Windows XP SP2 and later: Sami (Skolt, Finland) */
212 	{ "sma", "NO", SAMI_SOUTHERN_NORWAY }, /* Windows XP SP2 and later: Sami (Southern, Norway) */
213 	{ "sma", "SE", SAMI_SOUTHERN_SWEDEN }, /* Windows XP SP2 and later: Sami (Southern, Sweden) */
214 	{ "sa", "IN", SANSKRIT },              /* Windows 2000 and later: Sanskrit (India) */
215 	{ "sr", "SP", SERBIAN_LATIN },         /* Serbian (Latin) */
216 	{ "sr", "SIH",
217 	  SERBIAN_LATIN_BOSNIA_HERZEGOVINA },  /* Serbian (Latin) (Bosnia and Herzegovina) */
218 	{ "sr", "Cyrl_SP", SERBIAN_CYRILLIC }, /* Serbian (Cyrillic) */
219 	{ "sr", "Cyrl_SIH",
220 	  SERBIAN_CYRILLIC_BOSNIA_HERZEGOVINA }, /* Serbian (Cyrillic) (Bosnia and Herzegovina) */
221 	{ "ns", "ZA", SESOTHO_SA_LEBOA }, /* Windows XP SP2 and later: Sesotho sa Leboa/Northern Sotho
222 	                                     (South Africa) */
223 	{ "tn", "ZA", TSWANA },           /* Windows XP SP2 and later: Setswana/Tswana (South Africa) */
224 	{ "si", "LK", SINHALA },          /* Windows Vista and later: Sinhala (Sri Lanka) */
225 	{ "sk", "SK", SLOVAK },           /* Slovak (Slovakia) */
226 	{ "sl", "SI", SLOVENIAN },        /* Slovenian (Slovenia) */
227 	{ "es", "AR", SPANISH_ARGENTINA },          /* Spanish (Argentina) */
228 	{ "es", "BO", SPANISH_BOLIVIA },            /* Spanish (Bolivia) */
229 	{ "es", "CL", SPANISH_CHILE },              /* Spanish (Chile) */
230 	{ "es", "CO", SPANISH_COLOMBIA },           /* Spanish (Colombia) */
231 	{ "es", "CR", SPANISH_COSTA_RICA },         /* Spanish (Costa Rica) */
232 	{ "es", "DO", SPANISH_DOMINICAN_REPUBLIC }, /* Spanish (Dominican Republic) */
233 	{ "es", "EC", SPANISH_ECUADOR },            /* Spanish (Ecuador) */
234 	{ "es", "SV", SPANISH_EL_SALVADOR },        /* Spanish (El Salvador) */
235 	{ "es", "GT", SPANISH_GUATEMALA },          /* Spanish (Guatemala) */
236 	{ "es", "HN", SPANISH_HONDURAS },           /* Spanish (Honduras) */
237 	{ "es", "MX", SPANISH_MEXICAN },            /* Spanish (Mexico) */
238 	{ "es", "NI", SPANISH_NICARAGUA },          /* Spanish (Nicaragua) */
239 	{ "es", "PA", SPANISH_PANAMA },             /* Spanish (Panama) */
240 	{ "es", "PY", SPANISH_PARAGUAY },           /* Spanish (Paraguay) */
241 	{ "es", "PE", SPANISH_PERU },               /* Spanish (Peru) */
242 	{ "es", "PR", SPANISH_PUERTO_RICO },        /* Spanish (Puerto Rico) */
243 	{ "es", "ES", SPANISH_MODERN_SORT },        /* Spanish (Spain) */
244 	{ "es", "ES", SPANISH_TRADITIONAL_SORT },   /* Spanish (Spain, Traditional Sort) */
245 	{ "es", "US", SPANISH_UNITED_STATES }, /* Windows Vista and later: Spanish (United States) */
246 	{ "es", "UY", SPANISH_URUGUAY },       /* Spanish (Uruguay) */
247 	{ "es", "VE", SPANISH_VENEZUELA },     /* Spanish (Venezuela) */
248 	{ "sw", "KE", SWAHILI },               /* Windows 2000 and later: Swahili (Kenya) */
249 	{ "sv", "FI", SWEDISH_FINLAND },       /* Swedish (Finland) */
250 	{ "sv", "SE", SWEDISH },               /* Swedish (Sweden) */
251 	{ "syr", "SY", SYRIAC },               /* Windows XP and later: Syriac (Syria) */
252 	{ "ta", "IN", TAMIL },                 /* Windows 2000 and later: Tamil (India) */
253 	{ "tt", "RU", TATAR },                 /* Windows XP and later: Tatar (Russia) */
254 	{ "te", "IN", TELUGU },                /* Windows XP and later: Telugu (India) */
255 	{ "th", "TH", THAI },                  /* Thai (Thailand) */
256 	{ "bo", "BT", TIBETAN_BHUTAN },        /* Windows Vista and later: Tibetan (Bhutan) */
257 	{ "bo", "CN", TIBETAN_PRC },           /* Windows Vista and later: Tibetan (PRC) */
258 	{ "tr", "TR", TURKISH },               /* Turkish (Turkey) */
259 	{ "tk", "TM", TURKMEN },               /* Windows Vista and later: Turkmen (Turkmenistan) */
260 	{ "ug", "CN", UIGHUR },                /* Windows Vista and later: Uighur (PRC) */
261 	{ "uk", "UA", UKRAINIAN },             /* Ukrainian (Ukraine) */
262 	{ "wen", "DE", UPPER_SORBIAN },        /* Windows Vista and later: Upper Sorbian (Germany) */
263 	{ "tr", "IN", URDU_INDIA },            /* Urdu (India) */
264 	{ "ur", "PK", URDU },        /* Windows 98/Me, Windows 2000 and later: Urdu (Pakistan) */
265 	{ "uz", "UZ", UZBEK_LATIN }, /* Uzbek (Latin) */
266 	{ "uz", "Cyrl_UZ", UZBEK_CYRILLIC }, /* Uzbek (Cyrillic) */
267 	{ "vi", "VN", VIETNAMESE }, /* Windows 98/Me, Windows NT 4.0 and later: Vietnamese (Vietnam) */
268 	{ "cy", "GB", WELSH },      /* Windows XP SP2 and later: Welsh (United Kingdom) */
269 	{ "wo", "SN", WOLOF },      /* Windows Vista and later: Wolof (Senegal) */
270 	{ "xh", "ZA", XHOSA },      /* Windows XP SP2 and later: Xhosa/isiXhosa (South Africa) */
271 	{ "sah", "RU", YAKUT },     /* Windows Vista and later: Yakut (Russia) */
272 	{ "ii", "CN", YI },         /* Windows Vista and later: Yi (PRC) */
273 	{ "yo", "NG", YORUBA },     /* Windows Vista and later: Yoruba (Nigeria) */
274 	{ "zu", "ZA", ZULU }        /* Windows XP SP2 and later: Zulu/isiZulu (South Africa) */
275 };
276 
277 struct _LOCALE_NAME
278 {
279 	DWORD localeId;
280 	const char* name;
281 };
282 typedef struct _LOCALE_NAME LOCALE_NAME;
283 
284 static const LOCALE_NAME LOCALE_NAME_TABLE[] = {
285 	{ AFRIKAANS, "AFRIKAANS" },
286 	{ ALBANIAN, "ALBANIAN" },
287 	{ ALSATIAN, "ALSATIAN" },
288 	{ AMHARIC, "AMHARIC" },
289 	{ ARABIC_SAUDI_ARABIA, "ARABIC_SAUDI_ARABIA" },
290 	{ ARABIC_IRAQ, "ARABIC_IRAQ" },
291 	{ ARABIC_EGYPT, "ARABIC_EGYPT" },
292 	{ ARABIC_LIBYA, "ARABIC_LIBYA" },
293 	{ ARABIC_ALGERIA, "ARABIC_ALGERIA" },
294 	{ ARABIC_MOROCCO, "ARABIC_MOROCCO" },
295 	{ ARABIC_TUNISIA, "ARABIC_TUNISIA" },
296 	{ ARABIC_OMAN, "ARABIC_OMAN" },
297 	{ ARABIC_YEMEN, "ARABIC_YEMEN" },
298 	{ ARABIC_SYRIA, "ARABIC_SYRIA" },
299 	{ ARABIC_JORDAN, "ARABIC_JORDAN" },
300 	{ ARABIC_LEBANON, "ARABIC_LEBANON" },
301 	{ ARABIC_KUWAIT, "ARABIC_KUWAIT" },
302 	{ ARABIC_UAE, "ARABIC_UAE" },
303 	{ ARABIC_BAHRAIN, "ARABIC_BAHRAIN" },
304 	{ ARABIC_QATAR, "ARABIC_QATAR" },
305 	{ ARMENIAN, "ARMENIAN" },
306 	{ ASSAMESE, "ASSAMESE" },
307 	{ AZERI_LATIN, "AZERI_LATIN" },
308 	{ AZERI_CYRILLIC, "AZERI_CYRILLIC" },
309 	{ BASHKIR, "BASHKIR" },
310 	{ BASQUE, "BASQUE" },
311 	{ BELARUSIAN, "BELARUSIAN" },
312 	{ BENGALI_INDIA, "BENGALI_INDIA" },
313 	{ BOSNIAN_LATIN, "BOSNIAN_LATIN" },
314 	{ BRETON, "BRETON" },
315 	{ BULGARIAN, "BULGARIAN" },
316 	{ CATALAN, "CATALAN" },
317 	{ CHINESE_TAIWAN, "CHINESE_TAIWAN" },
318 	{ CHINESE_PRC, "CHINESE_PRC" },
319 	{ CHINESE_HONG_KONG, "CHINESE_HONG_KONG" },
320 	{ CHINESE_SINGAPORE, "CHINESE_SINGAPORE" },
321 	{ CHINESE_MACAU, "CHINESE_MACAU" },
322 	{ CROATIAN, "CROATIAN" },
323 	{ CROATIAN_BOSNIA_HERZEGOVINA, "CROATIAN_BOSNIA_HERZEGOVINA" },
324 	{ CZECH, "CZECH" },
325 	{ DANISH, "DANISH" },
326 	{ DARI, "DARI" },
327 	{ DIVEHI, "DIVEHI" },
328 	{ DUTCH_STANDARD, "DUTCH_STANDARD" },
329 	{ DUTCH_BELGIAN, "DUTCH_BELGIAN" },
330 	{ ENGLISH_UNITED_STATES, "ENGLISH_UNITED_STATES" },
331 	{ ENGLISH_UNITED_KINGDOM, "ENGLISH_UNITED_KINGDOM" },
332 	{ ENGLISH_AUSTRALIAN, "ENGLISH_AUSTRALIAN" },
333 	{ ENGLISH_CANADIAN, "ENGLISH_CANADIAN" },
334 	{ ENGLISH_NEW_ZEALAND, "ENGLISH_NEW_ZEALAND" },
335 	{ ENGLISH_INDIA, "ENGLISH_INDIA" },
336 	{ ENGLISH_IRELAND, "ENGLISH_IRELAND" },
337 	{ ENGLISH_MALAYSIA, "ENGLISH_MALAYSIA" },
338 	{ ENGLISH_SOUTH_AFRICA, "ENGLISH_SOUTH_AFRICA" },
339 	{ ENGLISH_JAMAICA, "ENGLISH_JAMAICA" },
340 	{ ENGLISH_CARIBBEAN, "ENGLISH_CARIBBEAN" },
341 	{ ENGLISH_BELIZE, "ENGLISH_BELIZE" },
342 	{ ENGLISH_TRINIDAD, "ENGLISH_TRINIDAD" },
343 	{ ENGLISH_ZIMBABWE, "ENGLISH_ZIMBABWE" },
344 	{ ENGLISH_PHILIPPINES, "ENGLISH_PHILIPPINES" },
345 	{ ENGLISH_SINGAPORE, "ENGLISH_SINGAPORE" },
346 	{ ESTONIAN, "ESTONIAN" },
347 	{ FAEROESE, "FAEROESE" },
348 	{ FARSI, "FARSI" },
349 	{ FILIPINO, "FILIPINO" },
350 	{ FINNISH, "FINNISH" },
351 	{ FRENCH_STANDARD, "FRENCH_STANDARD" },
352 	{ FRENCH_BELGIAN, "FRENCH_BELGIAN" },
353 	{ FRENCH_CANADIAN, "FRENCH_CANADIAN" },
354 	{ FRENCH_SWISS, "FRENCH_SWISS" },
355 	{ FRENCH_LUXEMBOURG, "FRENCH_LUXEMBOURG" },
356 	{ FRENCH_MONACO, "FRENCH_MONACO" },
357 	{ FRISIAN, "FRISIAN" },
358 	{ GEORGIAN, "GEORGIAN" },
359 	{ GALICIAN, "GALICIAN" },
360 	{ GERMAN_STANDARD, "GERMAN_STANDARD" },
361 	{ GERMAN_SWISS, "GERMAN_SWISS" },
362 	{ GERMAN_AUSTRIAN, "GERMAN_AUSTRIAN" },
363 	{ GERMAN_LUXEMBOURG, "GERMAN_LUXEMBOURG" },
364 	{ GERMAN_LIECHTENSTEIN, "GERMAN_LIECHTENSTEIN" },
365 	{ GREEK, "GREEK" },
366 	{ GREENLANDIC, "GREENLANDIC" },
367 	{ GUJARATI, "GUJARATI" },
368 	{ HEBREW, "HEBREW" },
369 	{ HINDI, "HINDI" },
370 	{ HUNGARIAN, "HUNGARIAN" },
371 	{ ICELANDIC, "ICELANDIC" },
372 	{ IGBO, "IGBO" },
373 	{ INDONESIAN, "INDONESIAN" },
374 	{ IRISH, "IRISH" },
375 	{ ITALIAN_STANDARD, "ITALIAN_STANDARD" },
376 	{ ITALIAN_SWISS, "ITALIAN_SWISS" },
377 	{ JAPANESE, "JAPANESE" },
378 	{ KANNADA, "KANNADA" },
379 	{ KAZAKH, "KAZAKH" },
380 	{ KHMER, "KHMER" },
381 	{ KICHE, "KICHE" },
382 	{ KINYARWANDA, "KINYARWANDA" },
383 	{ KONKANI, "KONKANI" },
384 	{ KOREAN, "KOREAN" },
385 	{ KYRGYZ, "KYRGYZ" },
386 	{ LAO, "LAO" },
387 	{ LATVIAN, "LATVIAN" },
388 	{ LITHUANIAN, "LITHUANIAN" },
389 	{ LOWER_SORBIAN, "LOWER_SORBIAN" },
390 	{ LUXEMBOURGISH, "LUXEMBOURGISH" },
391 	{ MACEDONIAN, "MACEDONIAN" },
392 	{ MALAY_MALAYSIA, "MALAY_MALAYSIA" },
393 	{ MALAY_BRUNEI_DARUSSALAM, "MALAY_BRUNEI_DARUSSALAM" },
394 	{ MALAYALAM, "MALAYALAM" },
395 	{ MALTESE, "MALTESE" },
396 	{ MAPUDUNGUN, "MAPUDUNGUN" },
397 	{ MAORI, "MAORI" },
398 	{ MARATHI, "MARATHI" },
399 	{ MOHAWK, "MOHAWK" },
400 	{ MONGOLIAN, "MONGOLIAN" },
401 	{ NEPALI, "NEPALI" },
402 	{ NORWEGIAN_BOKMAL, "NORWEGIAN_BOKMAL" },
403 	{ NORWEGIAN_NYNORSK, "NORWEGIAN_NYNORSK" },
404 	{ OCCITAN, "OCCITAN" },
405 	{ ORIYA, "ORIYA" },
406 	{ PASHTO, "PASHTO" },
407 	{ POLISH, "POLISH" },
408 	{ PORTUGUESE_BRAZILIAN, "PORTUGUESE_BRAZILIAN" },
409 	{ PORTUGUESE_STANDARD, "PORTUGUESE_STANDARD" },
410 	{ PUNJABI, "PUNJABI" },
411 	{ QUECHUA_BOLIVIA, "QUECHUA_BOLIVIA" },
412 	{ QUECHUA_ECUADOR, "QUECHUA_ECUADOR" },
413 	{ QUECHUA_PERU, "QUECHUA_PERU" },
414 	{ ROMANIAN, "ROMANIAN" },
415 	{ ROMANSH, "ROMANSH" },
416 	{ RUSSIAN, "RUSSIAN" },
417 	{ SAMI_INARI, "SAMI_INARI" },
418 	{ SAMI_LULE_NORWAY, "SAMI_LULE_NORWAY" },
419 	{ SAMI_LULE_SWEDEN, "SAMI_LULE_SWEDEN" },
420 	{ SAMI_NORTHERN_FINLAND, "SAMI_NORTHERN_FINLAND" },
421 	{ SAMI_NORTHERN_NORWAY, "SAMI_NORTHERN_NORWAY" },
422 	{ SAMI_NORTHERN_SWEDEN, "SAMI_NORTHERN_SWEDEN" },
423 	{ SAMI_SKOLT, "SAMI_SKOLT" },
424 	{ SAMI_SOUTHERN_NORWAY, "SAMI_SOUTHERN_NORWAY" },
425 	{ SAMI_SOUTHERN_SWEDEN, "SAMI_SOUTHERN_SWEDEN" },
426 	{ SANSKRIT, "SANSKRIT" },
427 	{ SERBIAN_LATIN, "SERBIAN_LATIN" },
428 	{ SERBIAN_LATIN_BOSNIA_HERZEGOVINA, "SERBIAN_LATIN_BOSNIA_HERZEGOVINA" },
429 	{ SERBIAN_CYRILLIC, "SERBIAN_CYRILLIC" },
430 	{ SERBIAN_CYRILLIC_BOSNIA_HERZEGOVINA, "SERBIAN_CYRILLIC_BOSNIA_HERZEGOVINA" },
431 	{ SESOTHO_SA_LEBOA, "SESOTHO_SA_LEBOA" },
432 	{ SINHALA, "SINHALA" },
433 	{ SLOVAK, "SLOVAK" },
434 	{ SLOVENIAN, "SLOVENIAN" },
435 	{ SPANISH_TRADITIONAL_SORT, "SPANISH_TRADITIONAL_SORT" },
436 	{ SPANISH_MEXICAN, "SPANISH_MEXICAN" },
437 	{ SPANISH_MODERN_SORT, "SPANISH_MODERN_SORT" },
438 	{ SPANISH_GUATEMALA, "SPANISH_GUATEMALA" },
439 	{ SPANISH_COSTA_RICA, "SPANISH_COSTA_RICA" },
440 	{ SPANISH_PANAMA, "SPANISH_PANAMA" },
441 	{ SPANISH_DOMINICAN_REPUBLIC, "SPANISH_DOMINICAN_REPUBLIC" },
442 	{ SPANISH_VENEZUELA, "SPANISH_VENEZUELA" },
443 	{ SPANISH_COLOMBIA, "SPANISH_COLOMBIA" },
444 	{ SPANISH_PERU, "SPANISH_PERU" },
445 	{ SPANISH_ARGENTINA, "SPANISH_ARGENTINA" },
446 	{ SPANISH_ECUADOR, "SPANISH_ECUADOR" },
447 	{ SPANISH_CHILE, "SPANISH_CHILE" },
448 	{ SPANISH_UNITED_STATES, "SPANISH_UNITED_STATES" },
449 	{ SPANISH_URUGUAY, "SPANISH_URUGUAY" },
450 	{ SPANISH_PARAGUAY, "SPANISH_PARAGUAY" },
451 	{ SPANISH_BOLIVIA, "SPANISH_BOLIVIA" },
452 	{ SPANISH_EL_SALVADOR, "SPANISH_EL_SALVADOR" },
453 	{ SPANISH_HONDURAS, "SPANISH_HONDURAS" },
454 	{ SPANISH_NICARAGUA, "SPANISH_NICARAGUA" },
455 	{ SPANISH_PUERTO_RICO, "SPANISH_PUERTO_RICO" },
456 	{ SWAHILI, "SWAHILI" },
457 	{ SWEDISH, "SWEDISH" },
458 	{ SWEDISH_FINLAND, "SWEDISH_FINLAND" },
459 	{ SYRIAC, "SYRIAC" },
460 	{ TAMIL, "TAMIL" },
461 	{ TATAR, "TATAR" },
462 	{ TELUGU, "TELUGU" },
463 	{ THAI, "THAI" },
464 	{ TIBETAN_BHUTAN, "TIBETAN_BHUTAN" },
465 	{ TIBETAN_PRC, "TIBETAN_PRC" },
466 	{ TSWANA, "TSWANA" },
467 	{ UKRAINIAN, "UKRAINIAN" },
468 	{ TURKISH, "TURKISH" },
469 	{ TURKMEN, "TURKMEN" },
470 	{ UIGHUR, "UIGHUR" },
471 	{ UPPER_SORBIAN, "UPPER_SORBIAN" },
472 	{ URDU, "URDU" },
473 	{ URDU_INDIA, "URDU_INDIA" },
474 	{ UZBEK_LATIN, "UZBEK_LATIN" },
475 	{ UZBEK_CYRILLIC, "UZBEK_CYRILLIC" },
476 	{ VIETNAMESE, "VIETNAMESE" },
477 	{ WELSH, "WELSH" },
478 	{ WOLOF, "WOLOF" },
479 	{ XHOSA, "XHOSA" },
480 	{ YAKUT, "YAKUT" },
481 	{ YI, "YI" },
482 	{ YORUBA, "YORUBA" },
483 	{ ZULU, "ZULU" }
484 };
485 
486 struct _LOCALE_KEYBOARD_LAYOUTS
487 {
488 	DWORD locale;             /* Locale ID */
489 	DWORD keyboardLayouts[5]; /* array of associated keyboard layouts */
490 };
491 typedef struct _LOCALE_KEYBOARD_LAYOUTS LOCALE_KEYBOARD_LAYOUTS;
492 
493 /* TODO: Use KBD_* defines instead of hardcoded values */
494 
495 static const LOCALE_KEYBOARD_LAYOUTS LOCALE_KEYBOARD_LAYOUTS_TABLE[] = {
496 	{ AFRIKAANS, { 0x00000409, 0x00000409, 0x0, 0x0, 0x0 } },
497 	{ ALBANIAN, { 0x0000041c, 0x00000409, 0x0, 0x0, 0x0 } },
498 	{ ARABIC_SAUDI_ARABIA, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
499 	{ ARABIC_IRAQ, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
500 	{ ARABIC_EGYPT, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
501 	{ ARABIC_LIBYA, { 0x0000040c, 0x00020401, 0x0, 0x0, 0x0 } },
502 	{ ARABIC_ALGERIA, { 0x0000040c, 0x00020401, 0x0, 0x0, 0x0 } },
503 	{ ARABIC_MOROCCO, { 0x0000040c, 0x00020401, 0x0, 0x0, 0x0 } },
504 	{ ARABIC_TUNISIA, { 0x0000040c, 0x00020401, 0x0, 0x0, 0x0 } },
505 	{ ARABIC_OMAN, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
506 	{ ARABIC_YEMEN, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
507 	{ ARABIC_SYRIA, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
508 	{ ARABIC_JORDAN, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
509 	{ ARABIC_LEBANON, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
510 	{ ARABIC_KUWAIT, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
511 	{ ARABIC_UAE, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
512 	{ ARABIC_BAHRAIN, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
513 	{ ARABIC_QATAR, { 0x00000409, 0x00000401, 0x0, 0x0, 0x0 } },
514 	{ ARMENIAN, { 0x0000042b, 0x00000409, 0x00000419, 0x0, 0x0 } },
515 	{ AZERI_LATIN, { 0x0000042c, 0x0000082c, 0x00000419, 0x0, 0x0 } },
516 	{ AZERI_CYRILLIC, { 0x0000082c, 0x0000042c, 0x00000419, 0x0, 0x0 } },
517 	{ BASQUE, { 0x0000040a, 0x00000409, 0x0, 0x0, 0x0 } },
518 	{ BELARUSIAN, { 0x00000423, 0x00000409, 0x00000419, 0x0, 0x0 } },
519 	{ BENGALI_INDIA, { 0x00000445, 0x00000409, 0x0, 0x0, 0x0 } },
520 	{ BOSNIAN_LATIN, { 0x0000141A, 0x00000409, 0x0, 0x0, 0x0 } },
521 	{ BULGARIAN, { 0x00000402, 0x00000409, 0x0, 0x0, 0x0 } },
522 	{ CATALAN, { 0x0000040a, 0x00000409, 0x0, 0x0, 0x0 } },
523 	{ CHINESE_TAIWAN, { 0x00000404, 0xe0080404, 0xE0010404, 0x0, 0x0 } },
524 	{ CHINESE_PRC, { 0x00000804, 0xe00e0804, 0xe0010804, 0xe0030804, 0xe0040804 } },
525 	{ CHINESE_HONG_KONG, { 0x00000409, 0xe0080404, 0x0, 0x0, 0x0 } },
526 	{ CHINESE_SINGAPORE, { 0x00000409, 0xe00e0804, 0xe0010804, 0xe0030804, 0xe0040804 } },
527 	{ CHINESE_MACAU, { 0x00000409, 0xe00e0804, 0xe0020404, 0xe0080404 } },
528 	{ CROATIAN, { 0x0000041a, 0x00000409, 0x0, 0x0, 0x0 } },
529 	{ CROATIAN_BOSNIA_HERZEGOVINA, { 0x0000041a, 0x00000409, 0x0, 0x0, 0x0 } },
530 	{ CZECH, { 0x00000405, 0x00000409, 0x0, 0x0, 0x0 } },
531 	{ DANISH, { 0x00000406, 0x00000409, 0x0, 0x0, 0x0 } },
532 	{ DIVEHI, { 0x00000409, 0x00000465, 0x0, 0x0, 0x0 } },
533 	{ DUTCH_STANDARD, { 0x00020409, 0x00000413, 0x00000409, 0x0, 0x0 } },
534 	{ DUTCH_BELGIAN, { 0x00000813, 0x00000409, 0x0, 0x0, 0x0 } },
535 	{ ENGLISH_UNITED_STATES, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
536 	{ ENGLISH_UNITED_KINGDOM, { 0x00000809, 0x0, 0x0, 0x0, 0x0 } },
537 	{ ENGLISH_AUSTRALIAN, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
538 	{ ENGLISH_CANADIAN, { 0x00000409, 0x00011009, 0x00001009, 0x0, 0x0 } },
539 	{ ENGLISH_NEW_ZEALAND, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
540 	{ ENGLISH_IRELAND, { 0x00001809, 0x00011809, 0x0, 0x0, 0x0 } },
541 	{ ENGLISH_SOUTH_AFRICA, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
542 	{ ENGLISH_JAMAICA, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
543 	{ ENGLISH_CARIBBEAN, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
544 	{ ENGLISH_BELIZE, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
545 	{ ENGLISH_TRINIDAD, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
546 	{ ENGLISH_ZIMBABWE, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
547 	{ ENGLISH_PHILIPPINES, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
548 	{ ESTONIAN, { 0x00000425, 0x0, 0x0, 0x0, 0x0 } },
549 	{ FAEROESE, { 0x00000406, 0x00000409, 0x0, 0x0, 0x0 } },
550 	{ FARSI, { 0x00000409, 0x00000429, 0x00000401, 0x0, 0x0 } },
551 	{ FINNISH, { 0x0000040b, 0x00000409, 0x0, 0x0, 0x0 } },
552 	{ FRENCH_STANDARD, { 0x0000040c, 0x00000409, 0x0, 0x0, 0x0 } },
553 	{ FRENCH_BELGIAN, { 0x0000080c, 0x00000409, 0x0, 0x0, 0x0 } },
554 	{ FRENCH_CANADIAN, { 0x00000C0C, 0x00011009, 0x00000409, 0x0, 0x0 } },
555 	{ FRENCH_SWISS, { 0x0000100c, 0x00000409, 0x0, 0x0, 0x0 } },
556 	{ FRENCH_LUXEMBOURG, { 0x0000040c, 0x00000409, 0x0, 0x0, 0x0 } },
557 	{ FRENCH_MONACO, { 0x0000040c, 0x00000409, 0x0, 0x0, 0x0 } },
558 	{ GEORGIAN, { 0x00000437, 0x00000409, 0x00000419, 0x0, 0x0 } },
559 	{ GALICIAN, { 0x0000040a, 0x00000409, 0x0, 0x0, 0x0 } },
560 	{ GERMAN_STANDARD, { 0x00000407, 0x00000409, 0x0, 0x0, 0x0 } },
561 	{ GERMAN_SWISS, { 0x00000807, 0x00000409, 0x0, 0x0, 0x0 } },
562 	{ GERMAN_AUSTRIAN, { 0x00000407, 0x00000409, 0x0, 0x0, 0x0 } },
563 	{ GERMAN_LUXEMBOURG, { 0x00000407, 0x00000409, 0x0, 0x0, 0x0 } },
564 	{ GERMAN_LIECHTENSTEIN, { 0x00000407, 0x00000409, 0x0, 0x0, 0x0 } },
565 	{ GREEK, { 0x00000408, 0x00000409, 0x0, 0x0, 0x0 } },
566 	{ GUJARATI, { 0x00000409, 0x00000447, 0x00010439, 0x0, 0x0 } },
567 	{ HEBREW, { 0x00000409, 0x0000040d, 0x0, 0x0, 0x0 } },
568 	{ HINDI, { 0x00000409, 0x00010439, 0x00000439, 0x0, 0x0 } },
569 	{ HUNGARIAN, { 0x0000040e, 0x00000409, 0x0, 0x0, 0x0 } },
570 	{ ICELANDIC, { 0x0000040f, 0x00000409, 0x0, 0x0, 0x0 } },
571 	{ INDONESIAN, { 0x00000409, 0x00000409, 0x0, 0x0, 0x0 } },
572 	{ ITALIAN_STANDARD, { 0x00000410, 0x00000409, 0x0, 0x0, 0x0 } },
573 	{ ITALIAN_SWISS, { 0x00000410, 0x00000409, 0x0, 0x0, 0x0 } },
574 	{ JAPANESE, { 0xe0010411, 0x0, 0x0, 0x0, 0x0 } },
575 	{ KANNADA, { 0x00000409, 0x0000044b, 0x00010439, 0x0, 0x0 } },
576 	{ KAZAKH, { 0x0000043f, 0x00000409, 0x00000419, 0x0, 0x0 } },
577 	{ KONKANI, { 0x00000409, 0x00000439, 0x0, 0x0, 0x0 } },
578 	{ KOREAN, { 0xE0010412, 0x0, 0x0, 0x0, 0x0 } },
579 	{ KYRGYZ, { 0x00000440, 0x00000409, 0x0, 0x0, 0x0 } },
580 	{ LATVIAN, { 0x00010426, 0x0, 0x0, 0x0, 0x0 } },
581 	{ LITHUANIAN, { 0x00010427, 0x0, 0x0, 0x0, 0x0 } },
582 	{ MACEDONIAN, { 0x0000042f, 0x00000409, 0x0, 0x0, 0x0 } },
583 	{ MALAY_MALAYSIA, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
584 	{ MALAY_BRUNEI_DARUSSALAM, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
585 	{ MALAYALAM, { 0x00000409, 0x0000044c, 0x0, 0x0, 0x0 } },
586 	{ MALTESE, { 0x00000409, 0x0000043a, 0x0, 0x0, 0x0 } },
587 	{ MAORI, { 0x00000409, 0x00000481, 0x0, 0x0, 0x0 } },
588 	{ MARATHI, { 0x00000409, 0x0000044e, 0x00000439, 0x0, 0x0 } },
589 	{ MONGOLIAN, { 0x00000450, 0x00000409, 0x0, 0x0, 0x0 } },
590 	{ NORWEGIAN_BOKMAL, { 0x00000414, 0x00000409, 0x0, 0x0, 0x0 } },
591 	{ NORWEGIAN_NYNORSK, { 0x00000414, 0x00000409, 0x0, 0x0, 0x0 } },
592 	{ POLISH, { 0x00010415, 0x00000415, 0x00000409, 0x0, 0x0 } },
593 	{ PORTUGUESE_BRAZILIAN, { 0x00000416, 0x00000409, 0x0, 0x0, 0x0 } },
594 	{ PORTUGUESE_STANDARD, { 0x00000816, 0x00000409, 0x0, 0x0, 0x0 } },
595 	{ PUNJABI, { 0x00000409, 0x00000446, 0x00010439, 0x0, 0x0 } },
596 	{ QUECHUA_BOLIVIA, { 0x00000409, 0x0000080A, 0x0, 0x0, 0x0 } },
597 	{ QUECHUA_ECUADOR, { 0x00000409, 0x0000080A, 0x0, 0x0, 0x0 } },
598 	{ QUECHUA_PERU, { 0x00000409, 0x0000080A, 0x0, 0x0, 0x0 } },
599 	{ ROMANIAN, { 0x00000418, 0x00000409, 0x0, 0x0, 0x0 } },
600 	{ RUSSIAN, { 0x00000419, 0x00000409, 0x0, 0x0, 0x0 } },
601 	{ SAMI_INARI, { 0x0001083b, 0x00000409, 0x0, 0x0, 0x0 } },
602 	{ SAMI_LULE_NORWAY, { 0x0000043b, 0x00000409, 0x0, 0x0, 0x0 } },
603 	{ SAMI_LULE_SWEDEN, { 0x0000083b, 0x00000409, 0x0, 0x0, 0x0 } },
604 	{ SAMI_NORTHERN_FINLAND, { 0x0001083b, 0x00000409, 0x0, 0x0, 0x0 } },
605 	{ SAMI_NORTHERN_NORWAY, { 0x0000043b, 0x00000409, 0x0, 0x0, 0x0 } },
606 	{ SAMI_NORTHERN_SWEDEN, { 0x0000083b, 0x00000409, 0x0, 0x0, 0x0 } },
607 	{ SAMI_SKOLT, { 0x0001083b, 0x00000409, 0x0, 0x0, 0x0 } },
608 	{ SAMI_SOUTHERN_NORWAY, { 0x0000043b, 0x00000409, 0x0, 0x0, 0x0 } },
609 	{ SAMI_SOUTHERN_SWEDEN, { 0x0000083b, 0x00000409, 0x0, 0x0, 0x0 } },
610 	{ SANSKRIT, { 0x00000409, 0x00000439, 0x0, 0x0, 0x0 } },
611 	{ SERBIAN_LATIN, { 0x0000081a, 0x00000409, 0x0, 0x0, 0x0 } },
612 	{ SERBIAN_LATIN_BOSNIA_HERZEGOVINA, { 0x0000081a, 0x00000409, 0x0, 0x0, 0x0 } },
613 	{ SERBIAN_CYRILLIC, { 0x00000c1a, 0x00000409, 0x0, 0x0, 0x0 } },
614 	{ SERBIAN_CYRILLIC_BOSNIA_HERZEGOVINA, { 0x00000c1a, 0x00000409, 0x0, 0x0, 0x0 } },
615 	{ SLOVAK, { 0x0000041b, 0x00000409, 0x0, 0x0, 0x0 } },
616 	{ SLOVENIAN, { 0x00000424, 0x00000409, 0x0, 0x0, 0x0 } },
617 	{ SPANISH_TRADITIONAL_SORT, { 0x0000040a, 0x00000409, 0x0, 0x0, 0x0 } },
618 	{ SPANISH_MEXICAN, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
619 	{ SPANISH_MODERN_SORT, { 0x0000040a, 0x00000409, 0x0, 0x0, 0x0 } },
620 	{ SPANISH_GUATEMALA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
621 	{ SPANISH_COSTA_RICA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
622 	{ SPANISH_PANAMA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
623 	{ SPANISH_DOMINICAN_REPUBLIC, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
624 	{ SPANISH_VENEZUELA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
625 	{ SPANISH_COLOMBIA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
626 	{ SPANISH_PERU, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
627 	{ SPANISH_ARGENTINA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
628 	{ SPANISH_ECUADOR, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
629 	{ SPANISH_CHILE, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
630 	{ SPANISH_URUGUAY, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
631 	{ SPANISH_PARAGUAY, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
632 	{ SPANISH_BOLIVIA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
633 	{ SPANISH_EL_SALVADOR, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
634 	{ SPANISH_HONDURAS, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
635 	{ SPANISH_NICARAGUA, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
636 	{ SPANISH_PUERTO_RICO, { 0x0000080a, 0x00000409, 0x0, 0x0, 0x0 } },
637 	{ SWAHILI, { 0x00000409, 0x0, 0x0, 0x0, 0x0 } },
638 	{ SWEDISH, { 0x0000041d, 0x00000409, 0x0, 0x0, 0x0 } },
639 	{ SWEDISH_FINLAND, { 0x0000041d, 0x00000409, 0x0, 0x0, 0x0 } },
640 	{ SYRIAC, { 0x00000409, 0x0000045a, 0x0, 0x0, 0x0 } },
641 	{ TAMIL, { 0x00000409, 0x00000449, 0x0, 0x0, 0x0 } },
642 	{ TATAR, { 0x00000444, 0x00000409, 0x00000419, 0x0, 0x0 } },
643 	{ TELUGU, { 0x00000409, 0x0000044a, 0x00010439, 0x0, 0x0 } },
644 	{ THAI, { 0x00000409, 0x0000041e, 0x0, 0x0, 0x0 } },
645 	{ TSWANA, { 0x00000409, 0x0000041f, 0x0, 0x0, 0x0 } },
646 	{ UKRAINIAN, { 0x00000422, 0x00000409, 0x0, 0x0, 0x0 } },
647 	{ TURKISH, { 0x0000041f, 0x0000041f, 0x0, 0x0, 0x0 } },
648 	{ UKRAINIAN, { 0x00000422, 0x00000409, 0x0, 0x0, 0x0 } },
649 	{ URDU, { 0x00000401, 0x00000409, 0x0, 0x0, 0x0 } },
650 	{ UZBEK_LATIN, { 0x00000409, 0x00000843, 0x00000419, 0x0, 0x0 } },
651 	{ UZBEK_CYRILLIC, { 0x00000843, 0x00000409, 0x00000419, 0x0, 0x0 } },
652 	{ VIETNAMESE, { 0x00000409, 0x0000042a, 0x0, 0x0, 0x0 } },
653 	{ WELSH, { 0x00000452, 0x00000809, 0x0, 0x0, 0x0 } },
654 	{ XHOSA, { 0x00000409, 0x00000409, 0x0, 0x0, 0x0 } },
655 };
656 
freerdp_get_system_language_and_country_codes(char * language,size_t languageLen,char * country,size_t countryLen)657 static BOOL freerdp_get_system_language_and_country_codes(char* language, size_t languageLen,
658                                                           char* country, size_t countryLen)
659 {
660 	assert(language);
661 	assert(languageLen > 0);
662 	assert(country);
663 	assert(countryLen);
664 
665 #if defined(__APPLE__)
666 	{
667 		CFIndex strSize;
668 		CFStringRef langRef, countryRef;
669 		CFLocaleRef localeRef = CFLocaleCopyCurrent();
670 		if (!localeRef)
671 			return FALSE;
672 
673 		langRef = (CFStringRef)CFLocaleGetValue(localeRef, kCFLocaleLanguageCode);
674 		countryRef = (CFStringRef)CFLocaleGetValue(localeRef, kCFLocaleCountryCode);
675 		if (!langRef || !countryRef)
676 		{
677 			CFRelease(localeRef);
678 			return FALSE;
679 		}
680 
681 		if (!CFStringGetCString(langRef, language, languageLen, kCFStringEncodingUTF8) ||
682 		    !CFStringGetCString(countryRef, country, countryLen, kCFStringEncodingUTF8))
683 		{
684 			CFRelease(localeRef);
685 			return FALSE;
686 		}
687 
688 		CFRelease(localeRef);
689 		return TRUE;
690 	}
691 #else
692 	{
693 		int dot;
694 		DWORD nSize;
695 		int underscore;
696 		char* env_lang = NULL;
697 		LPCSTR lang = "LANG";
698 		/* LANG = <language>_<country>.<encoding> */
699 		nSize = GetEnvironmentVariableA(lang, NULL, 0);
700 
701 		if (!nSize)
702 			return FALSE; /* LANG environment variable was not set */
703 
704 		env_lang = (char*)malloc(nSize);
705 
706 		if (!env_lang)
707 			return FALSE;
708 
709 		if (GetEnvironmentVariableA(lang, env_lang, nSize) !=
710 		    nSize - 1) /* Get locale from environment variable LANG */
711 		{
712 			free(env_lang);
713 			return FALSE;
714 		}
715 
716 		underscore = strcspn(env_lang, "_");
717 
718 		if (underscore > 3)
719 		{
720 			free(env_lang);
721 			return FALSE; /* The language name should not be more than 3 letters long */
722 		}
723 		else
724 		{
725 			/* Get language code */
726 			size_t len = MIN(languageLen - 1, underscore);
727 			strncpy(language, env_lang, len);
728 			language[len] = '\0';
729 		}
730 
731 		dot = strcspn(env_lang, ".");
732 
733 		/* Get country code */
734 		if (dot > underscore)
735 		{
736 			size_t len = MIN(countryLen - 1, dot - underscore - 1);
737 			strncpy(country, &env_lang[underscore + 1], len);
738 			country[len] = '\0';
739 		}
740 		else
741 		{
742 			free(env_lang);
743 			return FALSE; /* Invalid locale */
744 		}
745 
746 		free(env_lang);
747 		return TRUE;
748 	}
749 #endif
750 }
751 
freerdp_detect_system_locale(void)752 static const SYSTEM_LOCALE* freerdp_detect_system_locale(void)
753 {
754 	size_t i;
755 	char language[LOCALE_LANGUAGE_LEN] = { 0 };
756 	char country[LOCALE_COUNTRY_LEN] = { 0 };
757 	const SYSTEM_LOCALE* locale = NULL;
758 
759 	freerdp_get_system_language_and_country_codes(language, ARRAYSIZE(language), country,
760 	                                              ARRAYSIZE(country));
761 
762 	for (i = 0; i < ARRAYSIZE(SYSTEM_LOCALE_TABLE); i++)
763 	{
764 		const SYSTEM_LOCALE* current = &SYSTEM_LOCALE_TABLE[i];
765 
766 		if ((strcmp(language, current->language) == 0) && (strcmp(country, current->country) == 0))
767 		{
768 			locale = current;
769 			break;
770 		}
771 	}
772 
773 	return locale;
774 }
775 
freerdp_get_system_locale_id(void)776 DWORD freerdp_get_system_locale_id(void)
777 {
778 	const SYSTEM_LOCALE* locale;
779 	locale = freerdp_detect_system_locale();
780 
781 	if (locale != NULL)
782 		return locale->code;
783 
784 	return 0;
785 }
786 
freerdp_get_system_locale_name_from_id(DWORD localeId)787 const char* freerdp_get_system_locale_name_from_id(DWORD localeId)
788 {
789 	size_t index;
790 
791 	for (index = 0; index < ARRAYSIZE(LOCALE_NAME_TABLE); index++)
792 	{
793 		const LOCALE_NAME* const current = &LOCALE_NAME_TABLE[index];
794 
795 		if (localeId == current->localeId)
796 			return current->name;
797 	}
798 
799 	return NULL;
800 }
801 
freerdp_detect_keyboard_layout_from_system_locale(DWORD * keyboardLayoutId)802 int freerdp_detect_keyboard_layout_from_system_locale(DWORD* keyboardLayoutId)
803 {
804 	size_t i, j;
805 	char language[LOCALE_LANGUAGE_LEN] = { 0 };
806 	char country[LOCALE_COUNTRY_LEN] = { 0 };
807 	const SYSTEM_LOCALE* locale;
808 
809 	freerdp_get_system_language_and_country_codes(language, ARRAYSIZE(language), country,
810 	                                              ARRAYSIZE(country));
811 
812 	if ((strcmp(language, "C") == 0) || (strcmp(language, "POSIX") == 0))
813 	{
814 		*keyboardLayoutId = ENGLISH_UNITED_STATES; /* U.S. Keyboard Layout */
815 		return 0;
816 	}
817 
818 	locale = freerdp_detect_system_locale();
819 
820 	if (!locale)
821 		return -1;
822 
823 	DEBUG_KBD("Found locale : %s_%s", locale->language, locale->country);
824 
825 	for (i = 0; i < ARRAYSIZE(LOCALE_KEYBOARD_LAYOUTS_TABLE); i++)
826 	{
827 		const LOCALE_KEYBOARD_LAYOUTS* const current = &LOCALE_KEYBOARD_LAYOUTS_TABLE[i];
828 
829 		if (current->locale == locale->code)
830 		{
831 			/* Locale found in list of default keyboard layouts */
832 			for (j = 0; j < 5; j++)
833 			{
834 				if (current->keyboardLayouts[j] == ENGLISH_UNITED_STATES)
835 				{
836 					continue; /* Skip, try to get a more localized keyboard layout */
837 				}
838 				else if (current->keyboardLayouts[j] == 0)
839 				{
840 					break; /* No more keyboard layouts */
841 				}
842 				else
843 				{
844 					*keyboardLayoutId = current->keyboardLayouts[j];
845 					return 0;
846 				}
847 			}
848 
849 			/*
850 			 * If we skip the ENGLISH_UNITED_STATES keyboard layout but there are no
851 			 * other possible keyboard layout for the locale, we end up here with k > 1
852 			 */
853 
854 			if (j >= 1)
855 			{
856 				*keyboardLayoutId = ENGLISH_UNITED_STATES;
857 				return 0;
858 			}
859 			else
860 			{
861 				return -1;
862 			}
863 		}
864 	}
865 
866 	return -1; /* Could not detect the current keyboard layout from locale */
867 }
868