| // |
| // Copyright 2014 The Android Open Source Project |
| // |
| // Build resource files from raw assets. |
| // |
| |
| #include "ResourceFilter.h" |
| #include "AaptUtil.h" |
| #include "AaptConfig.h" |
| |
| status_t |
| WeakResourceFilter::parse(const String8& str) |
| { |
| Vector<String8> configStrs = AaptUtil::split(str, ','); |
| const size_t N = configStrs.size(); |
| mConfigs.clear(); |
| mConfigMask = 0; |
| mConfigs.resize(N); |
| for (size_t i = 0; i < N; i++) { |
| const String8& part = configStrs[i]; |
| if (part == "en_XA") { |
| mContainsPseudoAccented = true; |
| } else if (part == "ar_XB") { |
| mContainsPseudoBidi = true; |
| } |
| |
| std::pair<ConfigDescription, uint32_t>& entry = mConfigs.editItemAt(i); |
| |
| AaptLocaleValue val; |
| if (val.initFromFilterString(part)) { |
| // For backwards compatibility, we accept configurations that |
| // only specify locale in the standard 'en_US' format. |
| val.writeTo(&entry.first); |
| } else if (!AaptConfig::parse(part, &entry.first)) { |
| fprintf(stderr, "Invalid configuration: %s\n", part.string()); |
| return UNKNOWN_ERROR; |
| } |
| |
| entry.second = mDefault.diff(entry.first); |
| |
| // Ignore the version |
| entry.second &= ~ResTable_config::CONFIG_VERSION; |
| |
| // Ignore any densities. Those are best handled in --preferred-density |
| if ((entry.second & ResTable_config::CONFIG_DENSITY) != 0) { |
| fprintf(stderr, "warning: ignoring flag -c %s. Use --preferred-density instead.\n", entry.first.toString().string()); |
| entry.first.density = 0; |
| entry.second &= ~ResTable_config::CONFIG_DENSITY; |
| } |
| |
| mConfigMask |= entry.second; |
| } |
| |
| return NO_ERROR; |
| } |
| |
| // Returns true if the locale script of the config should be considered matching |
| // the locale script of entry. |
| // |
| // If both the scripts are empty, the scripts are considered matching for |
| // backward compatibility reasons. |
| // |
| // If only one script is empty, we try to compute it based on the provided |
| // language and country. If we could not compute it, we assume it's either a |
| // new language we don't know about, or a private use language. We return true |
| // since we don't know any better and they might as well be a match. |
| // |
| // Finally, when we have two scripts (one of which could be computed), we return |
| // true if and only if they are an exact match. |
| inline bool |
| scriptsMatch(const ResTable_config& config, const ResTable_config& entry) { |
| const char* configScript = config.localeScript; |
| const char* entryScript = entry.localeScript; |
| if (configScript[0] == '\0' && entryScript[0] == '\0') { |
| return true; // both scripts are empty. We match for backward compatibility reasons. |
| } |
| |
| char scriptBuffer[sizeof(config.localeScript)]; |
| if (configScript[0] == '\0') { |
| localeDataComputeScript(scriptBuffer, config.language, config.country); |
| if (scriptBuffer[0] == '\0') { // We can't compute the script, so we match. |
| return true; |
| } |
| configScript = scriptBuffer; |
| } else if (entryScript[0] == '\0') { |
| localeDataComputeScript( |
| scriptBuffer, entry.language, entry.country); |
| if (scriptBuffer[0] == '\0') { // We can't compute the script, so we match. |
| return true; |
| } |
| entryScript = scriptBuffer; |
| } |
| return (memcmp(configScript, entryScript, sizeof(config.localeScript)) == 0); |
| } |
| |
| |
| bool |
| WeakResourceFilter::match(const ResTable_config& config) const |
| { |
| uint32_t mask = mDefault.diff(config); |
| if ((mConfigMask & mask) == 0) { |
| // The two configurations don't have any common axis. |
| return true; |
| } |
| |
| uint32_t matchedAxis = 0x0; |
| const size_t N = mConfigs.size(); |
| for (size_t i = 0; i < N; i++) { |
| const std::pair<ConfigDescription, uint32_t>& entry = mConfigs[i]; |
| uint32_t diff = entry.first.diff(config); |
| if ((diff & entry.second) == 0) { |
| // Mark the axis that was matched. |
| matchedAxis |= entry.second; |
| } else if ((diff & entry.second) == ResTable_config::CONFIG_LOCALE) { |
| // If the locales differ, but the languages are the same and |
| // the locale we are matching only has a language specified, |
| // we match. |
| // |
| // Exception: we won't match if a script is specified for at least |
| // one of the locales and it's different from the other locale's |
| // script. (We will compute the other script if at least one of the |
| // scripts were explicitly set. In cases we can't compute an script, |
| // we match.) |
| if (config.language[0] != '\0' && |
| config.country[0] == '\0' && |
| config.localeVariant[0] == '\0' && |
| config.language[0] == entry.first.language[0] && |
| config.language[1] == entry.first.language[1] && |
| scriptsMatch(config, entry.first)) { |
| matchedAxis |= ResTable_config::CONFIG_LOCALE; |
| } |
| } else if ((diff & entry.second) == ResTable_config::CONFIG_SMALLEST_SCREEN_SIZE) { |
| // Special case if the smallest screen width doesn't match. We check that the |
| // config being matched has a smaller screen width than the filter specified. |
| if (config.smallestScreenWidthDp != 0 && |
| config.smallestScreenWidthDp < entry.first.smallestScreenWidthDp) { |
| matchedAxis |= ResTable_config::CONFIG_SMALLEST_SCREEN_SIZE; |
| } |
| } |
| } |
| return matchedAxis == (mConfigMask & mask); |
| } |
| |
| status_t |
| StrongResourceFilter::parse(const String8& str) { |
| Vector<String8> configStrs = AaptUtil::split(str, ','); |
| ConfigDescription config; |
| mConfigs.clear(); |
| for (size_t i = 0; i < configStrs.size(); i++) { |
| if (!AaptConfig::parse(configStrs[i], &config)) { |
| fprintf(stderr, "Invalid configuration: %s\n", configStrs[i].string()); |
| return UNKNOWN_ERROR; |
| } |
| mConfigs.insert(config); |
| } |
| return NO_ERROR; |
| } |