/* * Copyright (C) 2018 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 // umask #include // umask #include #include #include #include #include #include #include #include "android-base/macros.h" #include "utils/String8.h" #include "utils/Trace.h" #include "idmap2/BinaryStreamVisitor.h" #include "idmap2/FileUtils.h" #include "idmap2/Idmap.h" #include "idmap2d/Idmap2Service.h" using android::binder::Status; using android::idmap2::BinaryStreamVisitor; using android::idmap2::Idmap; using android::idmap2::IdmapHeader; namespace { static constexpr const char* kIdmapCacheDir = "/data/resource-cache"; Status ok() { return Status::ok(); } Status error(const std::string& msg) { LOG(ERROR) << msg; return Status::fromExceptionCode(Status::EX_NONE, msg.c_str()); } } // namespace namespace android { namespace os { Status Idmap2Service::getIdmapPath(const std::string& overlay_apk_path, int32_t user_id ATTRIBUTE_UNUSED, std::string* _aidl_return) { assert(_aidl_return); *_aidl_return = Idmap::CanonicalIdmapPathFor(kIdmapCacheDir, overlay_apk_path); return ok(); } Status Idmap2Service::removeIdmap(const std::string& overlay_apk_path, int32_t user_id ATTRIBUTE_UNUSED, bool* _aidl_return) { assert(_aidl_return); const std::string idmap_path = Idmap::CanonicalIdmapPathFor(kIdmapCacheDir, overlay_apk_path); if (unlink(idmap_path.c_str()) == 0) { *_aidl_return = true; return ok(); } else { *_aidl_return = false; return error("failed to unlink " + idmap_path + ": " + strerror(errno)); } } Status Idmap2Service::createIdmap(const std::string& target_apk_path, const std::string& overlay_apk_path, int32_t user_id, std::unique_ptr* _aidl_return) { assert(_aidl_return); std::stringstream trace; trace << __FUNCTION__ << " " << target_apk_path << " " << overlay_apk_path << " " << std::to_string(user_id); ATRACE_NAME(trace.str().c_str()); std::cout << trace.str() << std::endl; _aidl_return->reset(nullptr); const std::string idmap_path = Idmap::CanonicalIdmapPathFor(kIdmapCacheDir, overlay_apk_path); std::ifstream fin(idmap_path); const std::unique_ptr header = IdmapHeader::FromBinaryStream(fin); fin.close(); // do not reuse error stream from IsUpToDate below, or error messages will be // polluted with irrelevant data std::stringstream dev_null; if (header && header->IsUpToDate(dev_null)) { return ok(); } const std::unique_ptr target_apk = ApkAssets::Load(target_apk_path); if (!target_apk) { return error("failed to load apk " + target_apk_path); } const std::unique_ptr overlay_apk = ApkAssets::Load(overlay_apk_path); if (!overlay_apk) { return error("failed to load apk " + overlay_apk_path); } std::stringstream err; const std::unique_ptr idmap = Idmap::FromApkAssets(target_apk_path, *target_apk, overlay_apk_path, *overlay_apk, err); if (!idmap) { return error(err.str()); } umask(0133); // u=rw,g=r,o=r std::ofstream fout(idmap_path); if (fout.fail()) { return error("failed to open idmap path " + idmap_path); } BinaryStreamVisitor visitor(fout); idmap->accept(&visitor); fout.close(); if (fout.fail()) { return error("failed to write to idmap path " + idmap_path); } _aidl_return->reset(new std::string(idmap_path)); return ok(); } } // namespace os } // namespace android