Home | History | Annotate | Download | only in androidfw
      1 /*
      2  * Copyright (C) 2015 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 #include "android-base/macros.h"
     18 #include "androidfw/Locale.h"
     19 #include "androidfw/Util.h"
     20 
     21 #include <ctype.h>
     22 
     23 #include <algorithm>
     24 #include <string>
     25 #include <vector>
     26 
     27 using ::android::ResTable_config;
     28 using ::android::StringPiece;
     29 
     30 namespace android {
     31 
     32 void LocaleValue::set_language(const char* language_chars) {
     33   size_t i = 0;
     34   while ((*language_chars) != '\0') {
     35     language[i++] = ::tolower(*language_chars);
     36     language_chars++;
     37   }
     38 }
     39 
     40 void LocaleValue::set_region(const char* region_chars) {
     41   size_t i = 0;
     42   while ((*region_chars) != '\0') {
     43     region[i++] = ::toupper(*region_chars);
     44     region_chars++;
     45   }
     46 }
     47 
     48 void LocaleValue::set_script(const char* script_chars) {
     49   size_t i = 0;
     50   while ((*script_chars) != '\0') {
     51     if (i == 0) {
     52       script[i++] = ::toupper(*script_chars);
     53     } else {
     54       script[i++] = ::tolower(*script_chars);
     55     }
     56     script_chars++;
     57   }
     58 }
     59 
     60 void LocaleValue::set_variant(const char* variant_chars) {
     61   size_t i = 0;
     62   while ((*variant_chars) != '\0') {
     63     variant[i++] = *variant_chars;
     64     variant_chars++;
     65   }
     66 }
     67 
     68 static inline bool is_alpha(const std::string& str) {
     69   return std::all_of(std::begin(str), std::end(str), ::isalpha);
     70 }
     71 
     72 static inline bool is_number(const std::string& str) {
     73   return std::all_of(std::begin(str), std::end(str), ::isdigit);
     74 }
     75 
     76 bool LocaleValue::InitFromFilterString(const StringPiece& str) {
     77   // A locale (as specified in the filter) is an underscore separated name such
     78   // as "en_US", "en_Latn_US", or "en_US_POSIX".
     79   std::vector<std::string> parts = util::SplitAndLowercase(str, '_');
     80 
     81   const int num_tags = parts.size();
     82   bool valid = false;
     83   if (num_tags >= 1) {
     84     const std::string& lang = parts[0];
     85     if (is_alpha(lang) && (lang.length() == 2 || lang.length() == 3)) {
     86       set_language(lang.c_str());
     87       valid = true;
     88     }
     89   }
     90 
     91   if (!valid || num_tags == 1) {
     92     return valid;
     93   }
     94 
     95   // At this point, valid == true && numTags > 1.
     96   const std::string& part2 = parts[1];
     97   if ((part2.length() == 2 && is_alpha(part2)) ||
     98       (part2.length() == 3 && is_number(part2))) {
     99     set_region(part2.c_str());
    100   } else if (part2.length() == 4 && is_alpha(part2)) {
    101     set_script(part2.c_str());
    102   } else if (part2.length() >= 4 && part2.length() <= 8) {
    103     set_variant(part2.c_str());
    104   } else {
    105     valid = false;
    106   }
    107 
    108   if (!valid || num_tags == 2) {
    109     return valid;
    110   }
    111 
    112   // At this point, valid == true && numTags > 1.
    113   const std::string& part3 = parts[2];
    114   if (((part3.length() == 2 && is_alpha(part3)) ||
    115        (part3.length() == 3 && is_number(part3))) &&
    116       script[0]) {
    117     set_region(part3.c_str());
    118   } else if (part3.length() >= 4 && part3.length() <= 8) {
    119     set_variant(part3.c_str());
    120   } else {
    121     valid = false;
    122   }
    123 
    124   if (!valid || num_tags == 3) {
    125     return valid;
    126   }
    127 
    128   const std::string& part4 = parts[3];
    129   if (part4.length() >= 4 && part4.length() <= 8) {
    130     set_variant(part4.c_str());
    131   } else {
    132     valid = false;
    133   }
    134 
    135   if (!valid || num_tags > 4) {
    136     return false;
    137   }
    138 
    139   return true;
    140 }
    141 
    142 bool LocaleValue::InitFromBcp47Tag(const StringPiece& bcp47tag) {
    143   return InitFromBcp47TagImpl(bcp47tag, '-');
    144 }
    145 
    146 bool LocaleValue::InitFromBcp47TagImpl(const StringPiece& bcp47tag, const char separator) {
    147   std::vector<std::string> subtags = util::SplitAndLowercase(bcp47tag, separator);
    148   if (subtags.size() == 1) {
    149     set_language(subtags[0].c_str());
    150   } else if (subtags.size() == 2) {
    151     set_language(subtags[0].c_str());
    152 
    153     // The second tag can either be a region, a variant or a script.
    154     switch (subtags[1].size()) {
    155       case 2:
    156       case 3:
    157         set_region(subtags[1].c_str());
    158         break;
    159       case 4:
    160         if ('0' <= subtags[1][0] && subtags[1][0] <= '9') {
    161           // This is a variant: fall through
    162         } else {
    163           set_script(subtags[1].c_str());
    164           break;
    165         }
    166         FALLTHROUGH_INTENDED;
    167       case 5:
    168       case 6:
    169       case 7:
    170       case 8:
    171         set_variant(subtags[1].c_str());
    172         break;
    173       default:
    174         return false;
    175     }
    176   } else if (subtags.size() == 3) {
    177     // The language is always the first subtag.
    178     set_language(subtags[0].c_str());
    179 
    180     // The second subtag can either be a script or a region code.
    181     // If its size is 4, it's a script code, else it's a region code.
    182     if (subtags[1].size() == 4) {
    183       set_script(subtags[1].c_str());
    184     } else if (subtags[1].size() == 2 || subtags[1].size() == 3) {
    185       set_region(subtags[1].c_str());
    186     } else {
    187       return false;
    188     }
    189 
    190     // The third tag can either be a region code (if the second tag was
    191     // a script), else a variant code.
    192     if (subtags[2].size() >= 4) {
    193       set_variant(subtags[2].c_str());
    194     } else {
    195       set_region(subtags[2].c_str());
    196     }
    197   } else if (subtags.size() == 4) {
    198     set_language(subtags[0].c_str());
    199     set_script(subtags[1].c_str());
    200     set_region(subtags[2].c_str());
    201     set_variant(subtags[3].c_str());
    202   } else {
    203     return false;
    204   }
    205   return true;
    206 }
    207 
    208 ssize_t LocaleValue::InitFromParts(std::vector<std::string>::iterator iter,
    209                                    std::vector<std::string>::iterator end) {
    210   const std::vector<std::string>::iterator start_iter = iter;
    211 
    212   std::string& part = *iter;
    213   if (part[0] == 'b' && part[1] == '+') {
    214     // This is a "modified" BCP 47 language tag. Same semantics as BCP 47 tags,
    215     // except that the separator is "+" and not "-". Skip the prefix 'b+'.
    216     if (!InitFromBcp47TagImpl(StringPiece(part).substr(2), '+')) {
    217       return -1;
    218     }
    219     ++iter;
    220   } else {
    221     if ((part.length() == 2 || part.length() == 3) && is_alpha(part) && part != "car") {
    222       set_language(part.c_str());
    223       ++iter;
    224 
    225       if (iter != end) {
    226         const std::string& region_part = *iter;
    227         if (region_part.c_str()[0] == 'r' && region_part.length() == 3) {
    228           set_region(region_part.c_str() + 1);
    229           ++iter;
    230         }
    231       }
    232     }
    233   }
    234   return static_cast<ssize_t>(iter - start_iter);
    235 }
    236 
    237 void LocaleValue::InitFromResTable(const ResTable_config& config) {
    238   config.unpackLanguage(language);
    239   config.unpackRegion(region);
    240   if (config.localeScript[0] && !config.localeScriptWasComputed) {
    241     memcpy(script, config.localeScript, sizeof(config.localeScript));
    242   }
    243 
    244   if (config.localeVariant[0]) {
    245     memcpy(variant, config.localeVariant, sizeof(config.localeVariant));
    246   }
    247 }
    248 
    249 void LocaleValue::WriteTo(ResTable_config* out) const {
    250   out->packLanguage(language);
    251   out->packRegion(region);
    252 
    253   if (script[0]) {
    254     memcpy(out->localeScript, script, sizeof(out->localeScript));
    255   }
    256 
    257   if (variant[0]) {
    258     memcpy(out->localeVariant, variant, sizeof(out->localeVariant));
    259   }
    260 }
    261 
    262 }  // namespace android
    263