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