| /* |
| * Copyright (C) 2017 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. |
| */ |
| |
| #define ATRACE_TAG ATRACE_TAG_PACKAGE_MANAGER |
| |
| #include "CacheTracker.h" |
| |
| #include <fts.h> |
| #include <sys/xattr.h> |
| #include <utils/Trace.h> |
| |
| #include <android-base/logging.h> |
| #include <android-base/stringprintf.h> |
| |
| #include "QuotaUtils.h" |
| #include "utils.h" |
| |
| using android::base::StringPrintf; |
| |
| namespace android { |
| namespace installd { |
| |
| CacheTracker::CacheTracker(userid_t userId, appid_t appId, const std::string& uuid) |
| : cacheUsed(0), |
| cacheQuota(0), |
| mUserId(userId), |
| mAppId(appId), |
| mItemsLoaded(false), |
| mUuid(uuid) { |
| } |
| |
| CacheTracker::~CacheTracker() { |
| } |
| |
| std::string CacheTracker::toString() { |
| return StringPrintf("UID=%d used=%" PRId64 " quota=%" PRId64 " ratio=%d", |
| multiuser_get_uid(mUserId, mAppId), cacheUsed, cacheQuota, getCacheRatio()); |
| } |
| |
| void CacheTracker::addDataPath(const std::string& dataPath) { |
| mDataPaths.push_back(dataPath); |
| } |
| |
| void CacheTracker::loadStats() { |
| ATRACE_BEGIN("loadStats quota"); |
| cacheUsed = 0; |
| if (loadQuotaStats()) { |
| return; |
| } |
| ATRACE_END(); |
| |
| ATRACE_BEGIN("loadStats tree"); |
| cacheUsed = 0; |
| for (const auto& path : mDataPaths) { |
| auto cachePath = read_path_inode(path, "cache", kXattrInodeCache); |
| auto codeCachePath = read_path_inode(path, "code_cache", kXattrInodeCodeCache); |
| calculate_tree_size(cachePath, &cacheUsed); |
| calculate_tree_size(codeCachePath, &cacheUsed); |
| } |
| ATRACE_END(); |
| } |
| |
| bool CacheTracker::loadQuotaStats() { |
| int cacheGid = multiuser_get_cache_gid(mUserId, mAppId); |
| int extCacheGid = multiuser_get_ext_cache_gid(mUserId, mAppId); |
| if (IsQuotaSupported(mUuid) && cacheGid != -1 && extCacheGid != -1) { |
| int64_t space; |
| if ((space = GetOccupiedSpaceForGid(mUuid, cacheGid)) != -1) { |
| cacheUsed += space; |
| } else { |
| return false; |
| } |
| |
| if ((space = GetOccupiedSpaceForGid(mUuid, extCacheGid)) != -1) { |
| cacheUsed += space; |
| } else { |
| return false; |
| } |
| return true; |
| } else { |
| return false; |
| } |
| } |
| |
| void CacheTracker::loadItemsFrom(const std::string& path) { |
| FTS *fts; |
| FTSENT *p; |
| char *argv[] = { (char*) path.c_str(), nullptr }; |
| if (!(fts = fts_open(argv, FTS_PHYSICAL | FTS_NOCHDIR | FTS_XDEV, nullptr))) { |
| PLOG(WARNING) << "Failed to fts_open " << path; |
| return; |
| } |
| while ((p = fts_read(fts)) != nullptr) { |
| if (p->fts_level == 0) continue; |
| |
| // Create tracking nodes for everything we encounter |
| switch (p->fts_info) { |
| case FTS_D: |
| case FTS_DEFAULT: |
| case FTS_F: |
| case FTS_SL: |
| case FTS_SLNONE: { |
| auto item = std::shared_ptr<CacheItem>(new CacheItem(p)); |
| p->fts_pointer = static_cast<void*>(item.get()); |
| items.push_back(item); |
| } |
| } |
| |
| switch (p->fts_info) { |
| case FTS_D: { |
| auto item = static_cast<CacheItem*>(p->fts_pointer); |
| item->group |= (getxattr(p->fts_path, kXattrCacheGroup, nullptr, 0) >= 0); |
| item->tombstone |= (getxattr(p->fts_path, kXattrCacheTombstone, nullptr, 0) >= 0); |
| |
| // When group, immediately collect all files under tree |
| if (item->group) { |
| while ((p = fts_read(fts)) != nullptr) { |
| if (p->fts_info == FTS_DP && p->fts_level == item->level) break; |
| switch (p->fts_info) { |
| case FTS_D: |
| case FTS_DEFAULT: |
| case FTS_F: |
| case FTS_SL: |
| case FTS_SLNONE: |
| item->size += p->fts_statp->st_blocks * 512; |
| item->modified = std::max(item->modified, p->fts_statp->st_mtime); |
| } |
| } |
| } |
| } |
| } |
| |
| // Bubble up modified time to parent |
| CHECK(p != nullptr); |
| switch (p->fts_info) { |
| case FTS_DP: |
| case FTS_DEFAULT: |
| case FTS_F: |
| case FTS_SL: |
| case FTS_SLNONE: { |
| auto item = static_cast<CacheItem*>(p->fts_pointer); |
| auto parent = static_cast<CacheItem*>(p->fts_parent->fts_pointer); |
| if (parent) { |
| parent->modified = std::max(parent->modified, item->modified); |
| } |
| } |
| } |
| } |
| fts_close(fts); |
| } |
| |
| void CacheTracker::loadItems() { |
| items.clear(); |
| |
| ATRACE_BEGIN("loadItems"); |
| for (const auto& path : mDataPaths) { |
| loadItemsFrom(read_path_inode(path, "cache", kXattrInodeCache)); |
| loadItemsFrom(read_path_inode(path, "code_cache", kXattrInodeCodeCache)); |
| } |
| ATRACE_END(); |
| |
| ATRACE_BEGIN("sortItems"); |
| auto cmp = [](std::shared_ptr<CacheItem> left, std::shared_ptr<CacheItem> right) { |
| // TODO: sort dotfiles last |
| // TODO: sort code_cache last |
| if (left->modified != right->modified) { |
| return (left->modified > right->modified); |
| } |
| if (left->level != right->level) { |
| return (left->level < right->level); |
| } |
| return left->directory; |
| }; |
| std::stable_sort(items.begin(), items.end(), cmp); |
| ATRACE_END(); |
| } |
| |
| void CacheTracker::ensureItems() { |
| if (mItemsLoaded) { |
| return; |
| } else { |
| loadItems(); |
| mItemsLoaded = true; |
| } |
| } |
| |
| int CacheTracker::getCacheRatio() { |
| if (cacheQuota == 0) { |
| return 0; |
| } else { |
| return (cacheUsed * 10000) / cacheQuota; |
| } |
| } |
| |
| } // namespace installd |
| } // namespace android |