| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2017 The Android Open Source Project | 
|  | 3 | * | 
|  | 4 | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | 5 | * you may not use this file except in compliance with the License. | 
|  | 6 | * You may obtain a copy of the License at | 
|  | 7 | * | 
|  | 8 | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | 9 | * | 
|  | 10 | * Unless required by applicable law or agreed to in writing, software | 
|  | 11 | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 13 | * See the License for the specific language governing permissions and | 
|  | 14 | * limitations under the License. | 
|  | 15 | */ | 
|  | 16 |  | 
|  | 17 | #include "CacheItem.h" | 
|  | 18 |  | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 19 | #include <inttypes.h> | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 20 | #include <stdint.h> | 
|  | 21 | #include <sys/xattr.h> | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 22 |  | 
|  | 23 | #include <android-base/logging.h> | 
|  | 24 | #include <android-base/stringprintf.h> | 
|  | 25 |  | 
|  | 26 | #include "utils.h" | 
|  | 27 |  | 
|  | 28 | using android::base::StringPrintf; | 
|  | 29 |  | 
|  | 30 | namespace android { | 
|  | 31 | namespace installd { | 
|  | 32 |  | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 33 | CacheItem::CacheItem(FTSENT* p) { | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 34 | level = p->fts_level; | 
|  | 35 | directory = S_ISDIR(p->fts_statp->st_mode); | 
|  | 36 | size = p->fts_statp->st_blocks * 512; | 
|  | 37 | modified = p->fts_statp->st_mtime; | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 38 |  | 
|  | 39 | mParent = static_cast<CacheItem*>(p->fts_parent->fts_pointer); | 
|  | 40 | if (mParent) { | 
| Jeff Sharkey | ed909ae | 2017-03-22 21:27:40 -0600 | [diff] [blame] | 41 | group = mParent->group; | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 42 | tombstone = mParent->tombstone; | 
|  | 43 | mName = p->fts_name; | 
|  | 44 | mName.insert(0, "/"); | 
|  | 45 | } else { | 
| Jeff Sharkey | ed909ae | 2017-03-22 21:27:40 -0600 | [diff] [blame] | 46 | group = false; | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 47 | tombstone = false; | 
|  | 48 | mName = p->fts_path; | 
|  | 49 | } | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 50 | } | 
|  | 51 |  | 
|  | 52 | CacheItem::~CacheItem() { | 
|  | 53 | } | 
|  | 54 |  | 
|  | 55 | std::string CacheItem::toString() { | 
|  | 56 | return StringPrintf("%s size=%" PRId64 " mod=%ld", buildPath().c_str(), size, modified); | 
|  | 57 | } | 
|  | 58 |  | 
|  | 59 | std::string CacheItem::buildPath() { | 
|  | 60 | std::string res = mName; | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 61 | CacheItem* parent = mParent; | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 62 | while (parent) { | 
|  | 63 | res.insert(0, parent->mName); | 
|  | 64 | parent = parent->mParent; | 
|  | 65 | } | 
|  | 66 | return res; | 
|  | 67 | } | 
|  | 68 |  | 
|  | 69 | int CacheItem::purge() { | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 70 | int res = 0; | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 71 | auto path = buildPath(); | 
|  | 72 | if (directory) { | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 73 | FTS *fts; | 
|  | 74 | FTSENT *p; | 
|  | 75 | char *argv[] = { (char*) path.c_str(), nullptr }; | 
| Yi Kong | 954cf64 | 2018-07-17 16:16:24 -0700 | [diff] [blame] | 76 | if (!(fts = fts_open(argv, FTS_PHYSICAL | FTS_NOCHDIR | FTS_XDEV, nullptr))) { | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 77 | PLOG(WARNING) << "Failed to fts_open " << path; | 
|  | 78 | return -1; | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 79 | } | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 80 | while ((p = fts_read(fts)) != nullptr) { | 
|  | 81 | switch (p->fts_info) { | 
|  | 82 | case FTS_D: | 
|  | 83 | if (p->fts_level == 0) { | 
|  | 84 | p->fts_number = tombstone; | 
|  | 85 | } else { | 
|  | 86 | p->fts_number = p->fts_parent->fts_number | 
|  | 87 | | (getxattr(p->fts_path, kXattrCacheTombstone, nullptr, 0) >= 0); | 
|  | 88 | } | 
|  | 89 | break; | 
|  | 90 | case FTS_F: | 
|  | 91 | if (p->fts_parent->fts_number) { | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 92 | if (truncate(p->fts_path, 0) != 0) { | 
|  | 93 | PLOG(WARNING) << "Failed to truncate " << p->fts_path; | 
|  | 94 | res = -1; | 
|  | 95 | } | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 96 | } else { | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 97 | if (unlink(p->fts_path) != 0) { | 
|  | 98 | PLOG(WARNING) << "Failed to unlink " << p->fts_path; | 
|  | 99 | res = -1; | 
|  | 100 | } | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 101 | } | 
|  | 102 | break; | 
|  | 103 | case FTS_DEFAULT: | 
|  | 104 | case FTS_SL: | 
|  | 105 | case FTS_SLNONE: | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 106 | if (unlink(p->fts_path) != 0) { | 
|  | 107 | PLOG(WARNING) << "Failed to unlink " << p->fts_path; | 
|  | 108 | res = -1; | 
|  | 109 | } | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 110 | break; | 
|  | 111 | case FTS_DP: | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 112 | if (rmdir(p->fts_path) != 0) { | 
|  | 113 | PLOG(WARNING) << "Failed to rmdir " << p->fts_path; | 
|  | 114 | res = -1; | 
|  | 115 | } | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 116 | break; | 
|  | 117 | } | 
|  | 118 | } | 
| Alex Buynytskyy | 61fc874 | 2021-11-17 19:01:32 -0800 | [diff] [blame] | 119 | fts_close(fts); | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 120 | } else { | 
|  | 121 | if (tombstone) { | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 122 | if (truncate(path.c_str(), 0) != 0) { | 
|  | 123 | PLOG(WARNING) << "Failed to truncate " << path; | 
|  | 124 | res = -1; | 
|  | 125 | } | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 126 | } else { | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 127 | if (unlink(path.c_str()) != 0) { | 
|  | 128 | PLOG(WARNING) << "Failed to unlink " << path; | 
|  | 129 | res = -1; | 
|  | 130 | } | 
| Jeff Sharkey | 871a8f2 | 2017-02-21 18:30:28 -0700 | [diff] [blame] | 131 | } | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 132 | } | 
| Jeff Sharkey | d712f0c | 2017-05-03 11:36:42 -0600 | [diff] [blame] | 133 | return res; | 
| Jeff Sharkey | 88ddd94 | 2017-01-17 18:05:54 -0700 | [diff] [blame] | 134 | } | 
|  | 135 |  | 
|  | 136 | }  // namespace installd | 
|  | 137 | }  // namespace android |