C++程序  |  164行  |  5.52 KB

/*
 * Copyright (C) 2014 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include "RuleGenerator.h"
#include "aapt/SdkConstants.h"

#include <algorithm>
#include <cmath>
#include <vector>
#include <androidfw/ResourceTypes.h>

using namespace android;

namespace split {

// Calculate the point at which the density selection changes between l and h.
static inline int findMid(int l, int h) {
    double root = sqrt((h*h) + (8*l*h));
    return (double(-h) + root) / 2.0;
}

sp<Rule> RuleGenerator::generateDensity(const Vector<int>& allDensities, size_t index) {
    if (allDensities[index] != ResTable_config::DENSITY_ANY) {
        sp<Rule> densityRule = new Rule();
        densityRule->op = Rule::AND_SUBRULES;

        const bool hasAnyDensity = std::find(allDensities.begin(),
                allDensities.end(), (int) ResTable_config::DENSITY_ANY) != allDensities.end();

        if (hasAnyDensity) {
            sp<Rule> version = new Rule();
            version->op = Rule::LESS_THAN;
            version->key = Rule::SDK_VERSION;
            version->longArgs.add((long) SDK_LOLLIPOP);
            densityRule->subrules.add(version);
        }

        if (index > 0) {
            sp<Rule> gt = new Rule();
            gt->op = Rule::GREATER_THAN;
            gt->key = Rule::SCREEN_DENSITY;
            gt->longArgs.add(findMid(allDensities[index - 1], allDensities[index]) - 1);
            densityRule->subrules.add(gt);
        }

        if (index + 1 < allDensities.size() && allDensities[index + 1] != ResTable_config::DENSITY_ANY) {
            sp<Rule> lt = new Rule();
            lt->op = Rule::LESS_THAN;
            lt->key = Rule::SCREEN_DENSITY;
            lt->longArgs.add(findMid(allDensities[index], allDensities[index + 1]));
            densityRule->subrules.add(lt);
        }
        return densityRule;
    } else {
        // SDK_VERSION is handled elsewhere, so we always pick DENSITY_ANY if it's
        // available.
        sp<Rule> always = new Rule();
        always->op = Rule::ALWAYS_TRUE;
        return always;
    }
}

sp<Rule> RuleGenerator::generateAbi(const Vector<abi::Variant>& splitAbis, size_t index) {
    const abi::Variant thisAbi = splitAbis[index];
    const Vector<abi::Variant>& familyVariants = abi::getVariants(abi::getFamily(thisAbi));

    Vector<abi::Variant>::const_iterator start =
            std::find(familyVariants.begin(), familyVariants.end(), thisAbi);

    Vector<abi::Variant>::const_iterator end = familyVariants.end();
    if (index + 1 < splitAbis.size()) {
        end = std::find(start, familyVariants.end(), splitAbis[index + 1]);
    }

    sp<Rule> abiRule = new Rule();
    abiRule->op = Rule::CONTAINS_ANY;
    abiRule->key = Rule::NATIVE_PLATFORM;
    while (start != end) {
        abiRule->stringArgs.add(String8(abi::toString(*start)));
        ++start;
    }
    return abiRule;
}

sp<Rule> RuleGenerator::generate(const SortedVector<SplitDescription>& group, size_t index) {
    sp<Rule> rootRule = new Rule();
    rootRule->op = Rule::AND_SUBRULES;

    if (group[index].config.locale != 0) {
        sp<Rule> locale = new Rule();
        locale->op = Rule::EQUALS;
        locale->key = Rule::LANGUAGE;
        char str[RESTABLE_MAX_LOCALE_LEN];
        group[index].config.getBcp47Locale(str);
        locale->stringArgs.add(String8(str));
        rootRule->subrules.add(locale);
    }

    if (group[index].config.sdkVersion != 0) {
        sp<Rule> sdk = new Rule();
        sdk->op = Rule::GREATER_THAN;
        sdk->key = Rule::SDK_VERSION;
        sdk->longArgs.add(group[index].config.sdkVersion - 1);
        rootRule->subrules.add(sdk);
    }

    if (group[index].config.density != 0) {
        size_t densityIndex = 0;
        Vector<int> allDensities;
        allDensities.add(group[index].config.density);

        const size_t groupSize = group.size();
        for (size_t i = 0; i < groupSize; i++) {
            if (group[i].config.density != group[index].config.density) {
                // This group differs by density.
                allDensities.clear();
                for (size_t j = 0; j < groupSize; j++) {
                    allDensities.add(group[j].config.density);
                }
                densityIndex = index;
                break;
            }
        }
        rootRule->subrules.add(generateDensity(allDensities, densityIndex));
    }

    if (group[index].abi != abi::Variant_none) {
        size_t abiIndex = 0;
        Vector<abi::Variant> allVariants;
        allVariants.add(group[index].abi);

        const size_t groupSize = group.size();
        for (size_t i = 0; i < groupSize; i++) {
            if (group[i].abi != group[index].abi) {
                // This group differs by ABI.
                allVariants.clear();
                for (size_t j = 0; j < groupSize; j++) {
                    allVariants.add(group[j].abi);
                }
                abiIndex = index;
                break;
            }
        }
        rootRule->subrules.add(generateAbi(allVariants, abiIndex));
    }

    return rootRule;
}

} // namespace split