| /* | 
 |  * 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. | 
 |  */ | 
 |  | 
 | #include "CacheItem.h" | 
 |  | 
 | #include <inttypes.h> | 
 | #include <stdint.h> | 
 | #include <sys/xattr.h> | 
 |  | 
 | #include <android-base/logging.h> | 
 | #include <android-base/stringprintf.h> | 
 |  | 
 | #include "utils.h" | 
 |  | 
 | using android::base::StringPrintf; | 
 |  | 
 | namespace android { | 
 | namespace installd { | 
 |  | 
 | CacheItem::CacheItem(FTSENT* p) { | 
 |     level = p->fts_level; | 
 |     directory = S_ISDIR(p->fts_statp->st_mode); | 
 |     size = p->fts_statp->st_blocks * 512; | 
 |     modified = p->fts_statp->st_mtime; | 
 |  | 
 |     mParent = static_cast<CacheItem*>(p->fts_parent->fts_pointer); | 
 |     if (mParent) { | 
 |         group = mParent->group; | 
 |         tombstone = mParent->tombstone; | 
 |         mName = p->fts_name; | 
 |         mName.insert(0, "/"); | 
 |     } else { | 
 |         group = false; | 
 |         tombstone = false; | 
 |         mName = p->fts_path; | 
 |     } | 
 | } | 
 |  | 
 | CacheItem::~CacheItem() { | 
 | } | 
 |  | 
 | std::string CacheItem::toString() { | 
 |     return StringPrintf("%s size=%" PRId64 " mod=%ld", buildPath().c_str(), size, modified); | 
 | } | 
 |  | 
 | std::string CacheItem::buildPath() { | 
 |     std::string res = mName; | 
 |     CacheItem* parent = mParent; | 
 |     while (parent) { | 
 |         res.insert(0, parent->mName); | 
 |         parent = parent->mParent; | 
 |     } | 
 |     return res; | 
 | } | 
 |  | 
 | int CacheItem::purge() { | 
 |     int res = 0; | 
 |     auto path = buildPath(); | 
 |     if (directory) { | 
 |         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 -1; | 
 |         } | 
 |         while ((p = fts_read(fts)) != nullptr) { | 
 |             switch (p->fts_info) { | 
 |             case FTS_D: | 
 |                 if (p->fts_level == 0) { | 
 |                     p->fts_number = tombstone; | 
 |                 } else { | 
 |                     p->fts_number = p->fts_parent->fts_number | 
 |                             | (getxattr(p->fts_path, kXattrCacheTombstone, nullptr, 0) >= 0); | 
 |                 } | 
 |                 break; | 
 |             case FTS_F: | 
 |                 if (p->fts_parent->fts_number) { | 
 |                     if (truncate(p->fts_path, 0) != 0) { | 
 |                         PLOG(WARNING) << "Failed to truncate " << p->fts_path; | 
 |                         res = -1; | 
 |                     } | 
 |                 } else { | 
 |                     if (unlink(p->fts_path) != 0) { | 
 |                         PLOG(WARNING) << "Failed to unlink " << p->fts_path; | 
 |                         res = -1; | 
 |                     } | 
 |                 } | 
 |                 break; | 
 |             case FTS_DEFAULT: | 
 |             case FTS_SL: | 
 |             case FTS_SLNONE: | 
 |                 if (unlink(p->fts_path) != 0) { | 
 |                     PLOG(WARNING) << "Failed to unlink " << p->fts_path; | 
 |                     res = -1; | 
 |                 } | 
 |                 break; | 
 |             case FTS_DP: | 
 |                 if (rmdir(p->fts_path) != 0) { | 
 |                     PLOG(WARNING) << "Failed to rmdir " << p->fts_path; | 
 |                     res = -1; | 
 |                 } | 
 |                 break; | 
 |             } | 
 |         } | 
 |     } else { | 
 |         if (tombstone) { | 
 |             if (truncate(path.c_str(), 0) != 0) { | 
 |                 PLOG(WARNING) << "Failed to truncate " << path; | 
 |                 res = -1; | 
 |             } | 
 |         } else { | 
 |             if (unlink(path.c_str()) != 0) { | 
 |                 PLOG(WARNING) << "Failed to unlink " << path; | 
 |                 res = -1; | 
 |             } | 
 |         } | 
 |     } | 
 |     return res; | 
 | } | 
 |  | 
 | }  // namespace installd | 
 | }  // namespace android |