1 /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* ***** BEGIN LICENSE BLOCK *****
3 * Version: MPL 1.1/GPL 2.0/LGPL 2.1
5 * The contents of this file are subject to the Mozilla Public License Version
6 * 1.1 (the "License"); you may not use this file except in compliance with
7 * the License. You may obtain a copy of the License at
8 * http://www.mozilla.org/MPL/
10 * Software distributed under the License is distributed on an "AS IS" basis,
11 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
12 * for the specific language governing rights and limitations under the
15 * The Original Code is Mozilla Communicator client code.
17 * The Initial Developer of the Original Code is
18 * Netscape Communications Corporation.
19 * Portions created by the Initial Developer are Copyright (C) 1998
20 * the Initial Developer. All Rights Reserved.
24 * Alternatively, the contents of this file may be used under the terms of
25 * either of the GNU General Public License Version 2 or later (the "GPL"),
26 * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
27 * in which case the provisions of the GPL or the LGPL are applicable instead
28 * of those above. If you wish to allow use of your version of this file only
29 * under the terms of either the GPL or the LGPL, and not to allow others to
30 * use your version of this file under the terms of the MPL, indicate your
31 * decision by deleting the provisions above and replace them with the notice
32 * and other provisions required by the GPL or the LGPL. If you do not delete
33 * the provisions above, a recipient may use your version of this file under
34 * the terms of any one of the MPL, the GPL or the LGPL.
36 * ***** END LICENSE BLOCK ***** */
39 #include "nsIPlatformCharset.h"
41 #include "nsGREResProperties.h"
43 #include "nsReadableUtils.h"
44 #include "nsLocaleCID.h"
45 #include "nsUConvDll.h"
46 #include "nsIComponentManager.h"
47 #include "nsIServiceManager.h"
48 #include "nsIUnicodeDecoder.h"
49 #include "nsIUnicodeEncoder.h"
50 #include "nsICharsetConverterManager.h"
51 #include "nsEncoderDecoderUtils.h"
52 #if HAVE_GNU_LIBC_VERSION_H
53 #include <gnu/libc-version.h>
55 #ifdef HAVE_NL_TYPES_H
58 #if HAVE_LANGINFO_CODESET
61 #include "nsPlatformCharset.h"
62 #include "nsAutoLock.h"
64 #include "nsUnicharUtils.h"
66 NS_IMPL_THREADSAFE_ISUPPORTS1(nsPlatformCharset
, nsIPlatformCharset
)
68 static nsGREResProperties
*gNLInfo
= nsnull
;
69 static nsGREResProperties
*gInfo_deprecated
= nsnull
;
70 static PRInt32 gCnt
=0;
72 //this lock is for protecting above static variable operation
73 static PRLock
*gLock
= nsnull
;
75 static PRStatus
InitLock(void)
83 nsPlatformCharset::nsPlatformCharset()
85 PR_AtomicIncrement(&gCnt
);
86 static PRCallOnceType once
;
87 PR_CallOnce(&once
, InitLock
);
88 NS_ASSERTION(gLock
, "Can't allocate a lock?!");
92 nsPlatformCharset::ConvertLocaleToCharsetUsingDeprecatedConfig(nsAString
& locale
, nsACString
& oResult
)
95 // locked for thread safety
97 nsAutoLock
guard(gLock
);
98 if (!gInfo_deprecated
) {
99 nsGREResProperties
*info
=
100 new nsGREResProperties(NS_LITERAL_CSTRING("unixcharset.properties"));
101 NS_ASSERTION(info
, "cannot create nsGREResProperties");
102 gInfo_deprecated
= info
;
106 if (gInfo_deprecated
&& !(locale
.IsEmpty())) {
107 nsAutoString platformLocaleKey
;
108 // note: NS_LITERAL_STRING("locale." OSTYPE ".") does not compile on AIX
109 platformLocaleKey
.AssignLiteral("locale.");
110 platformLocaleKey
.AppendWithConversion(OSTYPE
);
111 platformLocaleKey
.AppendLiteral(".");
112 platformLocaleKey
.Append(locale
);
114 nsAutoString charset
;
115 nsresult res
= gInfo_deprecated
->Get(platformLocaleKey
, charset
);
116 if (NS_SUCCEEDED(res
)) {
117 LossyCopyUTF16toASCII(charset
, oResult
);
120 nsAutoString localeKey
;
121 localeKey
.AssignLiteral("locale.all.");
122 localeKey
.Append(locale
);
123 res
= gInfo_deprecated
->Get(localeKey
, charset
);
124 if (NS_SUCCEEDED(res
)) {
125 LossyCopyUTF16toASCII(charset
, oResult
);
129 NS_ASSERTION(0, "unable to convert locale to charset using deprecated config");
130 mCharset
.AssignLiteral("ISO-8859-1");
131 oResult
.AssignLiteral("ISO-8859-1");
132 return NS_SUCCESS_USING_FALLBACK_LOCALE
;
135 nsPlatformCharset::~nsPlatformCharset()
137 PR_AtomicDecrement(&gCnt
);
142 PR_DestroyLock(gLock
);
145 if (gInfo_deprecated
) {
146 delete gInfo_deprecated
;
147 gInfo_deprecated
= nsnull
;
153 nsPlatformCharset::GetCharset(nsPlatformCharsetSel selector
, nsACString
& oResult
)
160 nsPlatformCharset::GetDefaultCharsetForLocale(const nsAString
& localeName
, nsACString
&oResult
)
163 // if this locale is the user's locale then use the charset
164 // we already determined at initialization
166 if (mLocale
.Equals(localeName
) ||
167 // support the 4.x behavior
168 (mLocale
.LowerCaseEqualsLiteral("en_us") &&
169 localeName
.LowerCaseEqualsLiteral("c"))) {
174 #if HAVE_LANGINFO_CODESET
176 // This locale appears to be a different locale from the user's locale.
177 // To do this we would need to lock the global resource we are currently
178 // using or use a library that provides multi locale support.
179 // ICU is a possible example of a multi locale library.
180 // http://oss.software.ibm.com/icu/
182 // A more common cause of hitting this warning than the above is that
183 // Mozilla is launched under an ll_CC.UTF-8 locale. In xpLocale,
184 // we only store the language and the region (ll-CC) losing 'UTF-8', which
185 // leads |mLocale| to be different from |localeName|. Although we lose
186 // 'UTF-8', we init'd |mCharset| with the value obtained via
187 // |nl_langinfo(CODESET)| so that we're all right here.
189 NS_WARNING("GetDefaultCharsetForLocale: need to add multi locale support");
190 #ifdef DEBUG_jungshik
191 printf("localeName=%s mCharset=%s\n", NS_ConvertUTF16toUTF8(localeName
).get(),
194 // until we add multi locale support: use the the charset of the user's locale
196 return NS_SUCCESS_USING_FALLBACK_LOCALE
;
200 // convert from locale to charset
201 // using the deprecated locale to charset mapping
203 nsAutoString
localeStr(localeName
);
204 nsresult res
= ConvertLocaleToCharsetUsingDeprecatedConfig(localeStr
, oResult
);
205 if (NS_SUCCEEDED(res
))
208 NS_ASSERTION(0, "unable to convert locale to charset using deprecated config");
209 oResult
.AssignLiteral("ISO-8859-1");
210 return NS_SUCCESS_USING_FALLBACK_LOCALE
;
214 nsPlatformCharset::InitGetCharset(nsACString
&oString
)
216 char* nl_langinfo_codeset
= nsnull
;
220 #if HAVE_LANGINFO_CODESET
221 nl_langinfo_codeset
= nl_langinfo(CODESET
);
222 NS_ASSERTION(nl_langinfo_codeset
, "cannot get nl_langinfo(CODESET)");
225 // see if we can use nl_langinfo(CODESET) directly
227 if (nl_langinfo_codeset
) {
228 aCharset
.Assign(nl_langinfo_codeset
);
229 res
= VerifyCharset(aCharset
);
230 if (NS_SUCCEEDED(res
)) {
236 // locked for thread safety
238 nsAutoLock
guard(gLock
);
241 nsCAutoString propertyFile
;
242 // note: NS_LITERAL_CSTRING("unixcharset." OSARCH ".properties") does not compile on AIX
243 propertyFile
.AssignLiteral("unixcharset.");
244 propertyFile
.AppendLiteral(NS_STRINGIFY(OSARCH
));
245 propertyFile
.AppendLiteral(".properties");
246 nsGREResProperties
*info
= new nsGREResProperties(propertyFile
);
247 NS_ASSERTION(info
, "cannot create nsGREResProperties");
249 PRBool didLoad
= info
->DidLoad();
260 // See if we are remapping nl_langinfo(CODESET)
262 if (gNLInfo
&& nl_langinfo_codeset
) {
263 nsAutoString localeKey
;
265 #if HAVE_GNU_GET_LIBC_VERSION
267 // look for an glibc version specific charset remap
269 const char *glibc_version
= gnu_get_libc_version();
270 if ((glibc_version
!= nsnull
) && (strlen(glibc_version
))) {
271 localeKey
.AssignLiteral("nllic.");
272 localeKey
.AppendWithConversion(glibc_version
);
273 localeKey
.AppendLiteral(".");
274 localeKey
.AppendWithConversion(nl_langinfo_codeset
);
275 nsAutoString uCharset
;
276 res
= gNLInfo
->Get(localeKey
, uCharset
);
277 if (NS_SUCCEEDED(res
)) {
278 aCharset
.AssignWithConversion(uCharset
);
279 res
= VerifyCharset(aCharset
);
280 if (NS_SUCCEEDED(res
)) {
289 // look for a charset specific charset remap
291 localeKey
.AssignLiteral("nllic.");
292 localeKey
.AppendWithConversion(nl_langinfo_codeset
);
293 nsAutoString uCharset
;
294 res
= gNLInfo
->Get(localeKey
, uCharset
);
295 if (NS_SUCCEEDED(res
)) {
296 aCharset
.AssignWithConversion(uCharset
);
297 res
= VerifyCharset(aCharset
);
298 if (NS_SUCCEEDED(res
)) {
305 NS_ASSERTION(0, "unable to use nl_langinfo(CODESET)");
309 // try falling back on a deprecated (locale based) name
311 char* locale
= setlocale(LC_CTYPE
, nsnull
);
312 nsAutoString localeStr
;
313 localeStr
.AssignWithConversion(locale
);
314 res
= ConvertLocaleToCharsetUsingDeprecatedConfig(localeStr
, oString
);
315 if (NS_SUCCEEDED(res
)) {
316 return res
; // succeeded
324 nsPlatformCharset::Init()
326 nsCAutoString charset
;
327 nsresult res
= NS_OK
;
330 // remember default locale so we can use the
331 // same charset when asked for the same locale
333 char* locale
= setlocale(LC_CTYPE
, nsnull
);
334 NS_ASSERTION(locale
, "cannot setlocale");
336 CopyASCIItoUTF16(locale
, mLocale
);
338 mLocale
.AssignLiteral("en_US");
341 res
= InitGetCharset(charset
);
342 if (NS_SUCCEEDED(res
)) {
344 return res
; // succeeded
347 // last resort fallback
348 NS_ASSERTION(0, "unable to convert locale to charset using deprecated config");
349 mCharset
.AssignLiteral("ISO-8859-1");
350 return NS_SUCCESS_USING_FALLBACK_LOCALE
;
354 nsPlatformCharset::VerifyCharset(nsCString
&aCharset
)
358 // get the convert manager
360 nsCOMPtr
<nsICharsetConverterManager
> charsetConverterManager
;
361 charsetConverterManager
= do_GetService(NS_CHARSETCONVERTERMANAGER_CONTRACTID
, &res
);
366 // check if we can get an input converter
368 nsCOMPtr
<nsIUnicodeEncoder
> enc
;
369 res
= charsetConverterManager
->GetUnicodeEncoder(aCharset
.get(), getter_AddRefs(enc
));
370 if (NS_FAILED(res
)) {
371 NS_ASSERTION(0, "failed to create encoder");
376 // check if we can get an output converter
378 nsCOMPtr
<nsIUnicodeDecoder
> dec
;
379 res
= charsetConverterManager
->GetUnicodeDecoder(aCharset
.get(), getter_AddRefs(dec
));
380 if (NS_FAILED(res
)) {
381 NS_ASSERTION(0, "failed to create decoder");
386 // check if we recognize the charset string
389 nsCAutoString result
;
390 res
= charsetConverterManager
->GetCharsetAlias(aCharset
.get(), result
);
391 if (NS_FAILED(res
)) {
396 // return the preferred string
399 aCharset
.Assign(result
);
400 NS_ASSERTION(NS_SUCCEEDED(res
), "failed to get preferred charset name, using non-preferred");
405 nsPlatformCharset::InitInfo()