xref: /aosp_15_r20/external/libchrome/base/i18n/case_conversion.cc (revision 635a864187cb8b6c713ff48b7e790a6b21769273)
1*635a8641SAndroid Build Coastguard Worker // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2*635a8641SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*635a8641SAndroid Build Coastguard Worker // found in the LICENSE file.
4*635a8641SAndroid Build Coastguard Worker 
5*635a8641SAndroid Build Coastguard Worker #include "base/i18n/case_conversion.h"
6*635a8641SAndroid Build Coastguard Worker 
7*635a8641SAndroid Build Coastguard Worker #include <stdint.h>
8*635a8641SAndroid Build Coastguard Worker 
9*635a8641SAndroid Build Coastguard Worker #include "base/numerics/safe_conversions.h"
10*635a8641SAndroid Build Coastguard Worker #include "base/strings/string16.h"
11*635a8641SAndroid Build Coastguard Worker #include "base/strings/string_util.h"
12*635a8641SAndroid Build Coastguard Worker #include "third_party/icu/source/common/unicode/uchar.h"
13*635a8641SAndroid Build Coastguard Worker #include "third_party/icu/source/common/unicode/unistr.h"
14*635a8641SAndroid Build Coastguard Worker #include "third_party/icu/source/common/unicode/ustring.h"
15*635a8641SAndroid Build Coastguard Worker 
16*635a8641SAndroid Build Coastguard Worker namespace base {
17*635a8641SAndroid Build Coastguard Worker namespace i18n {
18*635a8641SAndroid Build Coastguard Worker 
19*635a8641SAndroid Build Coastguard Worker namespace {
20*635a8641SAndroid Build Coastguard Worker 
21*635a8641SAndroid Build Coastguard Worker // Provides a uniform interface for upper/lower/folding which take take
22*635a8641SAndroid Build Coastguard Worker // slightly varying parameters.
23*635a8641SAndroid Build Coastguard Worker typedef int32_t (*CaseMapperFunction)(UChar* dest, int32_t dest_capacity,
24*635a8641SAndroid Build Coastguard Worker                                       const UChar* src, int32_t src_length,
25*635a8641SAndroid Build Coastguard Worker                                       UErrorCode* error);
26*635a8641SAndroid Build Coastguard Worker 
ToUpperMapper(UChar * dest,int32_t dest_capacity,const UChar * src,int32_t src_length,UErrorCode * error)27*635a8641SAndroid Build Coastguard Worker int32_t ToUpperMapper(UChar* dest, int32_t dest_capacity,
28*635a8641SAndroid Build Coastguard Worker                       const UChar* src, int32_t src_length,
29*635a8641SAndroid Build Coastguard Worker                       UErrorCode* error) {
30*635a8641SAndroid Build Coastguard Worker   // Use default locale.
31*635a8641SAndroid Build Coastguard Worker   return u_strToUpper(dest, dest_capacity, src, src_length, nullptr, error);
32*635a8641SAndroid Build Coastguard Worker }
33*635a8641SAndroid Build Coastguard Worker 
ToLowerMapper(UChar * dest,int32_t dest_capacity,const UChar * src,int32_t src_length,UErrorCode * error)34*635a8641SAndroid Build Coastguard Worker int32_t ToLowerMapper(UChar* dest, int32_t dest_capacity,
35*635a8641SAndroid Build Coastguard Worker                       const UChar* src, int32_t src_length,
36*635a8641SAndroid Build Coastguard Worker                       UErrorCode* error) {
37*635a8641SAndroid Build Coastguard Worker   // Use default locale.
38*635a8641SAndroid Build Coastguard Worker   return u_strToLower(dest, dest_capacity, src, src_length, nullptr, error);
39*635a8641SAndroid Build Coastguard Worker }
40*635a8641SAndroid Build Coastguard Worker 
FoldCaseMapper(UChar * dest,int32_t dest_capacity,const UChar * src,int32_t src_length,UErrorCode * error)41*635a8641SAndroid Build Coastguard Worker int32_t FoldCaseMapper(UChar* dest, int32_t dest_capacity,
42*635a8641SAndroid Build Coastguard Worker                        const UChar* src, int32_t src_length,
43*635a8641SAndroid Build Coastguard Worker                        UErrorCode* error) {
44*635a8641SAndroid Build Coastguard Worker   return u_strFoldCase(dest, dest_capacity, src, src_length,
45*635a8641SAndroid Build Coastguard Worker                        U_FOLD_CASE_DEFAULT, error);
46*635a8641SAndroid Build Coastguard Worker }
47*635a8641SAndroid Build Coastguard Worker 
48*635a8641SAndroid Build Coastguard Worker // Provides similar functionality as UnicodeString::caseMap but on string16.
CaseMap(StringPiece16 string,CaseMapperFunction case_mapper)49*635a8641SAndroid Build Coastguard Worker string16 CaseMap(StringPiece16 string, CaseMapperFunction case_mapper) {
50*635a8641SAndroid Build Coastguard Worker   string16 dest;
51*635a8641SAndroid Build Coastguard Worker   if (string.empty())
52*635a8641SAndroid Build Coastguard Worker     return dest;
53*635a8641SAndroid Build Coastguard Worker 
54*635a8641SAndroid Build Coastguard Worker   // Provide an initial guess that the string length won't change. The typical
55*635a8641SAndroid Build Coastguard Worker   // strings we use will very rarely change length in this process, so don't
56*635a8641SAndroid Build Coastguard Worker   // optimize for that case.
57*635a8641SAndroid Build Coastguard Worker   dest.resize(string.size());
58*635a8641SAndroid Build Coastguard Worker 
59*635a8641SAndroid Build Coastguard Worker   UErrorCode error;
60*635a8641SAndroid Build Coastguard Worker   do {
61*635a8641SAndroid Build Coastguard Worker     error = U_ZERO_ERROR;
62*635a8641SAndroid Build Coastguard Worker 
63*635a8641SAndroid Build Coastguard Worker     // ICU won't terminate the string if there's not enough room for the null
64*635a8641SAndroid Build Coastguard Worker     // terminator, but will otherwise. So we don't need to save room for that.
65*635a8641SAndroid Build Coastguard Worker     // Don't use WriteInto, which assumes null terminators.
66*635a8641SAndroid Build Coastguard Worker     int32_t new_length = case_mapper(
67*635a8641SAndroid Build Coastguard Worker         &dest[0], saturated_cast<int32_t>(dest.size()),
68*635a8641SAndroid Build Coastguard Worker         string.data(), saturated_cast<int32_t>(string.size()),
69*635a8641SAndroid Build Coastguard Worker         &error);
70*635a8641SAndroid Build Coastguard Worker     dest.resize(new_length);
71*635a8641SAndroid Build Coastguard Worker   } while (error == U_BUFFER_OVERFLOW_ERROR);
72*635a8641SAndroid Build Coastguard Worker   return dest;
73*635a8641SAndroid Build Coastguard Worker }
74*635a8641SAndroid Build Coastguard Worker 
75*635a8641SAndroid Build Coastguard Worker }  // namespace
76*635a8641SAndroid Build Coastguard Worker 
ToLower(StringPiece16 string)77*635a8641SAndroid Build Coastguard Worker string16 ToLower(StringPiece16 string) {
78*635a8641SAndroid Build Coastguard Worker   return CaseMap(string, &ToLowerMapper);
79*635a8641SAndroid Build Coastguard Worker }
80*635a8641SAndroid Build Coastguard Worker 
ToUpper(StringPiece16 string)81*635a8641SAndroid Build Coastguard Worker string16 ToUpper(StringPiece16 string) {
82*635a8641SAndroid Build Coastguard Worker   return CaseMap(string, &ToUpperMapper);
83*635a8641SAndroid Build Coastguard Worker }
84*635a8641SAndroid Build Coastguard Worker 
FoldCase(StringPiece16 string)85*635a8641SAndroid Build Coastguard Worker string16 FoldCase(StringPiece16 string) {
86*635a8641SAndroid Build Coastguard Worker   return CaseMap(string, &FoldCaseMapper);
87*635a8641SAndroid Build Coastguard Worker }
88*635a8641SAndroid Build Coastguard Worker 
89*635a8641SAndroid Build Coastguard Worker }  // namespace i18n
90*635a8641SAndroid Build Coastguard Worker }  // namespace base
91