/*
* 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 "AaptAssets.h"
#include "ApkBuilder.h"
using namespace android;
ApkBuilder::ApkBuilder(const sp<WeakResourceFilter>& configFilter)
: mConfigFilter(configFilter)
, mDefaultFilter(new AndResourceFilter()) {
// Add the default split, which is present for all APKs.
mDefaultFilter->addFilter(mConfigFilter);
mSplits.add(new ApkSplit(std::set<ConfigDescription>(), mDefaultFilter, true));
}
status_t ApkBuilder::createSplitForConfigs(const std::set<ConfigDescription>& configs) {
const size_t N = mSplits.size();
for (size_t i = 0; i < N; i++) {
const std::set<ConfigDescription>& splitConfigs = mSplits[i]->getConfigs();
std::set<ConfigDescription>::const_iterator iter = configs.begin();
for (; iter != configs.end(); iter++) {
if (splitConfigs.count(*iter) > 0) {
// Can't have overlapping configurations.
fprintf(stderr, "ERROR: Split configuration '%s' is already defined "
"in another split.\n", iter->toString().string());
return ALREADY_EXISTS;
}
}
}
sp<StrongResourceFilter> splitFilter = new StrongResourceFilter(configs);
// Add the inverse filter of this split filter to the base apk filter so it will
// omit resources that belong in this split.
mDefaultFilter->addFilter(new InverseResourceFilter(splitFilter));
// Now add the apk-wide config filter to our split filter.
sp<AndResourceFilter> filter = new AndResourceFilter();
filter->addFilter(splitFilter);
filter->addFilter(mConfigFilter);
mSplits.add(new ApkSplit(configs, filter));
return NO_ERROR;
}
status_t ApkBuilder::addEntry(const String8& path, const sp<AaptFile>& file) {
const size_t N = mSplits.size();
for (size_t i = 0; i < N; i++) {
if (mSplits[i]->matches(file)) {
return mSplits.editItemAt(i)->addEntry(path, file);
}
}
// Entry can be dropped if it doesn't match any split. This will only happen
// if the enry doesn't mConfigFilter.
return NO_ERROR;
}
void ApkBuilder::print() const {
fprintf(stderr, "APK Builder\n");
fprintf(stderr, "-----------\n");
const size_t N = mSplits.size();
for (size_t i = 0; i < N; i++) {
mSplits[i]->print();
fprintf(stderr, "\n");
}
}
ApkSplit::ApkSplit(const std::set<ConfigDescription>& configs, const sp<ResourceFilter>& filter, bool isBase)
: mConfigs(configs), mFilter(filter), mIsBase(isBase) {
std::set<ConfigDescription>::const_iterator iter = configs.begin();
for (; iter != configs.end(); iter++) {
if (mName.size() > 0) {
mName.append(",");
mDirName.append("_");
mPackageSafeName.append(".");
}
String8 configStr = iter->toString();
String8 packageConfigStr(configStr);
size_t len = packageConfigStr.length();
if (len > 0) {
char* buf = packageConfigStr.lockBuffer(len);
for (char* end = buf + len; buf < end; ++buf) {
if (*buf == '-') {
*buf = '_';
}
}
packageConfigStr.unlockBuffer(len);
}
mName.append(configStr);
mDirName.append(configStr);
mPackageSafeName.append(packageConfigStr);
}
}
status_t ApkSplit::addEntry(const String8& path, const sp<AaptFile>& file) {
if (!mFiles.insert(OutputEntry(path, file)).second) {
// Duplicate file.
return ALREADY_EXISTS;
}
return NO_ERROR;
}
void ApkSplit::print() const {
fprintf(stderr, "APK Split '%s'\n", mName.string());
std::set<OutputEntry>::const_iterator iter = mFiles.begin();
for (; iter != mFiles.end(); iter++) {
fprintf(stderr, " %s (%s)\n", iter->getPath().string(), iter->getFile()->getSourceFile().string());
}
}