1 /*
2  * Copyright (C) 2019 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 <android-base/file.h>
18 #include <android-base/logging.h>
19 #include <android-base/unique_fd.h>
20 #include <binder/ParcelFileDescriptor.h>
21 #include <gmock/gmock.h>
22 #include <gtest/gtest.h>
23 #include <utils/Log.h>
24 #include <utils/String16.h>
25 
26 #include <chrono>
27 #include <future>
28 
29 #include "IncrementalService.h"
30 #include "IncrementalServiceValidation.h"
31 #include "Metadata.pb.h"
32 #include "ServiceWrappers.h"
33 
34 using namespace testing;
35 using namespace android::incremental;
36 using namespace std::literals;
37 using testing::_;
38 using testing::Invoke;
39 using testing::NiceMock;
40 
41 #undef LOG_TAG
42 #define LOG_TAG "IncrementalServiceTest"
43 
44 using namespace android::incfs;
45 using namespace android::content::pm;
46 using PerUidReadTimeouts = android::os::incremental::PerUidReadTimeouts;
47 
48 namespace android::os::incremental {
49 
50 class MockVoldService : public VoldServiceWrapper {
51 public:
52     MOCK_CONST_METHOD5(mountIncFs,
53                        binder::Status(const std::string& backingPath, const std::string& targetDir,
54                                       int32_t flags, const std::string& sysfsName,
55                                       IncrementalFileSystemControlParcel* _aidl_return));
56     MOCK_CONST_METHOD1(unmountIncFs, binder::Status(const std::string& dir));
57     MOCK_CONST_METHOD2(bindMount,
58                        binder::Status(const std::string& sourceDir, const std::string& argetDir));
59     MOCK_CONST_METHOD4(
60             setIncFsMountOptions,
61             binder::Status(const ::android::os::incremental::IncrementalFileSystemControlParcel&,
62                            bool, bool, const std::string&));
63 
mountIncFsFails()64     void mountIncFsFails() {
65         ON_CALL(*this, mountIncFs(_, _, _, _, _))
66                 .WillByDefault(
67                         Return(binder::Status::fromExceptionCode(1, String8("failed to mount"))));
68     }
mountIncFsInvalidControlParcel()69     void mountIncFsInvalidControlParcel() {
70         ON_CALL(*this, mountIncFs(_, _, _, _, _))
71                 .WillByDefault(Invoke(this, &MockVoldService::getInvalidControlParcel));
72     }
mountIncFsSuccess()73     void mountIncFsSuccess() {
74         ON_CALL(*this, mountIncFs(_, _, _, _, _))
75                 .WillByDefault(Invoke(this, &MockVoldService::incFsSuccess));
76     }
bindMountFails()77     void bindMountFails() {
78         ON_CALL(*this, bindMount(_, _))
79                 .WillByDefault(Return(
80                         binder::Status::fromExceptionCode(1, String8("failed to bind-mount"))));
81     }
bindMountSuccess()82     void bindMountSuccess() {
83         ON_CALL(*this, bindMount(_, _)).WillByDefault(Return(binder::Status::ok()));
84     }
setIncFsMountOptionsFails() const85     void setIncFsMountOptionsFails() const {
86         ON_CALL(*this, setIncFsMountOptions(_, _, _, _))
87                 .WillByDefault(Return(
88                         binder::Status::fromExceptionCode(1, String8("failed to set options"))));
89     }
setIncFsMountOptionsSuccess()90     void setIncFsMountOptionsSuccess() {
91         ON_CALL(*this, setIncFsMountOptions(_, _, _, _))
92                 .WillByDefault(Invoke(this, &MockVoldService::setIncFsMountOptionsOk));
93     }
getInvalidControlParcel(const std::string & imagePath,const std::string & targetDir,int32_t flags,const std::string & sysfsName,IncrementalFileSystemControlParcel * _aidl_return)94     binder::Status getInvalidControlParcel(const std::string& imagePath,
95                                            const std::string& targetDir, int32_t flags,
96                                            const std::string& sysfsName,
97                                            IncrementalFileSystemControlParcel* _aidl_return) {
98         _aidl_return = {};
99         return binder::Status::ok();
100     }
incFsSuccess(const std::string & imagePath,const std::string & targetDir,int32_t flags,const std::string & sysfsName,IncrementalFileSystemControlParcel * _aidl_return)101     binder::Status incFsSuccess(const std::string& imagePath, const std::string& targetDir,
102                                 int32_t flags, const std::string& sysfsName,
103                                 IncrementalFileSystemControlParcel* _aidl_return) {
104         _aidl_return->pendingReads.reset(base::unique_fd(dup(STDIN_FILENO)));
105         _aidl_return->cmd.reset(base::unique_fd(dup(STDIN_FILENO)));
106         _aidl_return->log.reset(base::unique_fd(dup(STDIN_FILENO)));
107         return binder::Status::ok();
108     }
setIncFsMountOptionsOk(const::android::os::incremental::IncrementalFileSystemControlParcel & control,bool enableReadLogs,bool enableReadTimeouts,const std::string & sysfsName)109     binder::Status setIncFsMountOptionsOk(
110             const ::android::os::incremental::IncrementalFileSystemControlParcel& control,
111             bool enableReadLogs, bool enableReadTimeouts, const std::string& sysfsName) {
112         mReadLogsEnabled = enableReadLogs;
113         mReadTimeoutsEnabled = enableReadTimeouts;
114         return binder::Status::ok();
115     }
116 
readLogsEnabled() const117     bool readLogsEnabled() const { return mReadLogsEnabled; }
readTimeoutsEnabled() const118     bool readTimeoutsEnabled() const { return mReadTimeoutsEnabled; }
119 
120 private:
121     TemporaryFile cmdFile;
122     TemporaryFile logFile;
123 
124     bool mReadLogsEnabled = false;
125     bool mReadTimeoutsEnabled = true;
126 };
127 
128 class MockDataLoader : public IDataLoader {
129 public:
MockDataLoader()130     MockDataLoader() {
131         initializeCreateOk();
132         ON_CALL(*this, start(_)).WillByDefault(Invoke(this, &MockDataLoader::startOk));
133         ON_CALL(*this, stop(_)).WillByDefault(Invoke(this, &MockDataLoader::stopOk));
134         ON_CALL(*this, destroy(_)).WillByDefault(Invoke(this, &MockDataLoader::destroyOk));
135         ON_CALL(*this, prepareImage(_, _, _))
136                 .WillByDefault(Invoke(this, &MockDataLoader::prepareImageOk));
137     }
onAsBinder()138     IBinder* onAsBinder() override { return nullptr; }
139     MOCK_METHOD4(create,
140                  binder::Status(int32_t id, const DataLoaderParamsParcel& params,
141                                 const FileSystemControlParcel& control,
142                                 const sp<IDataLoaderStatusListener>& listener));
143     MOCK_METHOD1(start, binder::Status(int32_t id));
144     MOCK_METHOD1(stop, binder::Status(int32_t id));
145     MOCK_METHOD1(destroy, binder::Status(int32_t id));
146     MOCK_METHOD3(prepareImage,
147                  binder::Status(int32_t id, const std::vector<InstallationFileParcel>& addedFiles,
148                                 const std::vector<std::string>& removedFiles));
149 
initializeCreateOk()150     void initializeCreateOk() {
151         ON_CALL(*this, create(_, _, _, _)).WillByDefault(Invoke(this, &MockDataLoader::createOk));
152     }
153 
initializeCreateOkNoStatus()154     void initializeCreateOkNoStatus() {
155         ON_CALL(*this, create(_, _, _, _))
156                 .WillByDefault(Invoke(this, &MockDataLoader::createOkNoStatus));
157     }
158 
createOk(int32_t id,const content::pm::DataLoaderParamsParcel & params,const content::pm::FileSystemControlParcel & control,const sp<content::pm::IDataLoaderStatusListener> & listener)159     binder::Status createOk(int32_t id, const content::pm::DataLoaderParamsParcel& params,
160                             const content::pm::FileSystemControlParcel& control,
161                             const sp<content::pm::IDataLoaderStatusListener>& listener) {
162         createOkNoStatus(id, params, control, listener);
163         reportStatus(id);
164         return binder::Status::ok();
165     }
createOkNoStatus(int32_t id,const content::pm::DataLoaderParamsParcel & params,const content::pm::FileSystemControlParcel & control,const sp<content::pm::IDataLoaderStatusListener> & listener)166     binder::Status createOkNoStatus(int32_t id, const content::pm::DataLoaderParamsParcel& params,
167                                     const content::pm::FileSystemControlParcel& control,
168                                     const sp<content::pm::IDataLoaderStatusListener>& listener) {
169         mServiceConnector = control.service;
170         mListener = listener;
171         mStatus = IDataLoaderStatusListener::DATA_LOADER_CREATED;
172         return binder::Status::ok();
173     }
startOk(int32_t id)174     binder::Status startOk(int32_t id) {
175         setAndReportStatus(id, IDataLoaderStatusListener::DATA_LOADER_STARTED);
176         return binder::Status::ok();
177     }
stopOk(int32_t id)178     binder::Status stopOk(int32_t id) {
179         setAndReportStatus(id, IDataLoaderStatusListener::DATA_LOADER_STOPPED);
180         return binder::Status::ok();
181     }
destroyOk(int32_t id)182     binder::Status destroyOk(int32_t id) {
183         setAndReportStatus(id, IDataLoaderStatusListener::DATA_LOADER_DESTROYED);
184         mListener = nullptr;
185         return binder::Status::ok();
186     }
prepareImageOk(int32_t id,const::std::vector<content::pm::InstallationFileParcel> &,const::std::vector<::std::string> &)187     binder::Status prepareImageOk(int32_t id,
188                                   const ::std::vector<content::pm::InstallationFileParcel>&,
189                                   const ::std::vector<::std::string>&) {
190         setAndReportStatus(id, IDataLoaderStatusListener::DATA_LOADER_IMAGE_READY);
191         return binder::Status::ok();
192     }
setStorageParams(bool enableReadLogs)193     int32_t setStorageParams(bool enableReadLogs) {
194         int32_t result = -1;
195         EXPECT_NE(mServiceConnector.get(), nullptr);
196         EXPECT_TRUE(mServiceConnector->setStorageParams(enableReadLogs, &result).isOk());
197         return result;
198     }
status() const199     int status() const { return mStatus; }
200 
201 private:
setAndReportStatus(int id,int status)202     void setAndReportStatus(int id, int status) {
203         mStatus = status;
204         reportStatus(id);
205     }
reportStatus(int id)206     void reportStatus(int id) {
207         if (mListener) {
208             mListener->onStatusChanged(id, mStatus);
209         }
210     }
211 
212     sp<IIncrementalServiceConnector> mServiceConnector;
213     sp<IDataLoaderStatusListener> mListener;
214     int mStatus = IDataLoaderStatusListener::DATA_LOADER_DESTROYED;
215 };
216 
217 class MockDataLoaderManager : public DataLoaderManagerWrapper {
218 public:
MockDataLoaderManager(sp<IDataLoader> dataLoader)219     MockDataLoaderManager(sp<IDataLoader> dataLoader) : mDataLoaderHolder(std::move(dataLoader)) {
220         EXPECT_TRUE(mDataLoaderHolder != nullptr);
221     }
222 
223     MOCK_CONST_METHOD5(bindToDataLoader,
224                        binder::Status(int32_t mountId, const DataLoaderParamsParcel& params,
225                                       int bindDelayMs,
226                                       const sp<IDataLoaderStatusListener>& listener,
227                                       bool* _aidl_return));
228     MOCK_CONST_METHOD2(getDataLoader,
229                        binder::Status(int32_t mountId, sp<IDataLoader>* _aidl_return));
230     MOCK_CONST_METHOD1(unbindFromDataLoader, binder::Status(int32_t mountId));
231 
bindToDataLoaderSuccess()232     void bindToDataLoaderSuccess() {
233         ON_CALL(*this, bindToDataLoader(_, _, _, _, _))
234                 .WillByDefault(Invoke(this, &MockDataLoaderManager::bindToDataLoaderOk));
235     }
bindToDataLoaderFails()236     void bindToDataLoaderFails() {
237         ON_CALL(*this, bindToDataLoader(_, _, _, _, _))
238                 .WillByDefault(Return(
239                         (binder::Status::fromExceptionCode(1, String8("failed to prepare")))));
240     }
getDataLoaderSuccess()241     void getDataLoaderSuccess() {
242         ON_CALL(*this, getDataLoader(_, _))
243                 .WillByDefault(Invoke(this, &MockDataLoaderManager::getDataLoaderOk));
244     }
unbindFromDataLoaderSuccess()245     void unbindFromDataLoaderSuccess() {
246         ON_CALL(*this, unbindFromDataLoader(_))
247                 .WillByDefault(Invoke(this, &MockDataLoaderManager::unbindFromDataLoaderOk));
248     }
bindToDataLoaderOk(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)249     binder::Status bindToDataLoaderOk(int32_t mountId, const DataLoaderParamsParcel& params,
250                                       int bindDelayMs,
251                                       const sp<IDataLoaderStatusListener>& listener,
252                                       bool* _aidl_return) {
253         mId = mountId;
254         mListener = listener;
255         mDataLoader = mDataLoaderHolder;
256         mBindDelayMs = bindDelayMs;
257         *_aidl_return = true;
258         if (mListener) {
259             mListener->onStatusChanged(mId, IDataLoaderStatusListener::DATA_LOADER_BOUND);
260         }
261         return binder::Status::ok();
262     }
bindToDataLoaderNotOkWithNoDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)263     binder::Status bindToDataLoaderNotOkWithNoDelay(int32_t mountId,
264                                                     const DataLoaderParamsParcel& params,
265                                                     int bindDelayMs,
266                                                     const sp<IDataLoaderStatusListener>& listener,
267                                                     bool* _aidl_return) {
268         CHECK(bindDelayMs == 0) << bindDelayMs;
269         *_aidl_return = false;
270         return binder::Status::ok();
271     }
bindToDataLoaderBindingWithNoDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)272     binder::Status bindToDataLoaderBindingWithNoDelay(int32_t mountId,
273                                                       const DataLoaderParamsParcel& params,
274                                                       int bindDelayMs,
275                                                       const sp<IDataLoaderStatusListener>& listener,
276                                                       bool* _aidl_return) {
277         CHECK(bindDelayMs == 0) << bindDelayMs;
278         *_aidl_return = true;
279         if (listener) {
280             listener->onStatusChanged(mId, IDataLoaderStatusListener::DATA_LOADER_BINDING);
281         }
282         return binder::Status::ok();
283     }
bindToDataLoaderOkWithNoDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)284     binder::Status bindToDataLoaderOkWithNoDelay(int32_t mountId,
285                                                  const DataLoaderParamsParcel& params,
286                                                  int bindDelayMs,
287                                                  const sp<IDataLoaderStatusListener>& listener,
288                                                  bool* _aidl_return) {
289         CHECK(bindDelayMs == 0) << bindDelayMs;
290         return bindToDataLoaderOk(mountId, params, bindDelayMs, listener, _aidl_return);
291     }
bindToDataLoaderOkWith1sDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)292     binder::Status bindToDataLoaderOkWith1sDelay(int32_t mountId,
293                                                  const DataLoaderParamsParcel& params,
294                                                  int bindDelayMs,
295                                                  const sp<IDataLoaderStatusListener>& listener,
296                                                  bool* _aidl_return) {
297         CHECK(100 * 9 <= bindDelayMs && bindDelayMs <= 100 * 11) << bindDelayMs;
298         return bindToDataLoaderOk(mountId, params, bindDelayMs, listener, _aidl_return);
299     }
bindToDataLoaderOkWith10sDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)300     binder::Status bindToDataLoaderOkWith10sDelay(int32_t mountId,
301                                                   const DataLoaderParamsParcel& params,
302                                                   int bindDelayMs,
303                                                   const sp<IDataLoaderStatusListener>& listener,
304                                                   bool* _aidl_return) {
305         CHECK(100 * 9 * 9 <= bindDelayMs && bindDelayMs <= 100 * 11 * 11) << bindDelayMs;
306         return bindToDataLoaderOk(mountId, params, bindDelayMs, listener, _aidl_return);
307     }
bindToDataLoaderOkWith100sDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)308     binder::Status bindToDataLoaderOkWith100sDelay(int32_t mountId,
309                                                    const DataLoaderParamsParcel& params,
310                                                    int bindDelayMs,
311                                                    const sp<IDataLoaderStatusListener>& listener,
312                                                    bool* _aidl_return) {
313         CHECK(100 * 9 * 9 * 9 < bindDelayMs && bindDelayMs < 100 * 11 * 11 * 11) << bindDelayMs;
314         return bindToDataLoaderOk(mountId, params, bindDelayMs, listener, _aidl_return);
315     }
bindToDataLoaderOkWith1000sDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)316     binder::Status bindToDataLoaderOkWith1000sDelay(int32_t mountId,
317                                                     const DataLoaderParamsParcel& params,
318                                                     int bindDelayMs,
319                                                     const sp<IDataLoaderStatusListener>& listener,
320                                                     bool* _aidl_return) {
321         CHECK(100 * 9 * 9 * 9 * 9 < bindDelayMs && bindDelayMs < 100 * 11 * 11 * 11 * 11)
322                 << bindDelayMs;
323         return bindToDataLoaderOk(mountId, params, bindDelayMs, listener, _aidl_return);
324     }
bindToDataLoaderOkWith10000sDelay(int32_t mountId,const DataLoaderParamsParcel & params,int bindDelayMs,const sp<IDataLoaderStatusListener> & listener,bool * _aidl_return)325     binder::Status bindToDataLoaderOkWith10000sDelay(int32_t mountId,
326                                                      const DataLoaderParamsParcel& params,
327                                                      int bindDelayMs,
328                                                      const sp<IDataLoaderStatusListener>& listener,
329                                                      bool* _aidl_return) {
330         CHECK(100 * 9 * 9 * 9 * 9 * 9 < bindDelayMs && bindDelayMs < 100 * 11 * 11 * 11 * 11 * 11)
331                 << bindDelayMs;
332         return bindToDataLoaderOk(mountId, params, bindDelayMs, listener, _aidl_return);
333     }
334 
getDataLoaderOk(int32_t mountId,sp<IDataLoader> * _aidl_return)335     binder::Status getDataLoaderOk(int32_t mountId, sp<IDataLoader>* _aidl_return) {
336         *_aidl_return = mDataLoader;
337         return binder::Status::ok();
338     }
setDataLoaderStatusCreated()339     void setDataLoaderStatusCreated() {
340         mListener->onStatusChanged(mId, IDataLoaderStatusListener::DATA_LOADER_CREATED);
341     }
setDataLoaderStatusStarted()342     void setDataLoaderStatusStarted() {
343         mListener->onStatusChanged(mId, IDataLoaderStatusListener::DATA_LOADER_STARTED);
344     }
setDataLoaderStatusDestroyed()345     void setDataLoaderStatusDestroyed() {
346         mListener->onStatusChanged(mId, IDataLoaderStatusListener::DATA_LOADER_DESTROYED);
347     }
setDataLoaderStatusUnavailable()348     void setDataLoaderStatusUnavailable() {
349         mListener->onStatusChanged(mId, IDataLoaderStatusListener::DATA_LOADER_UNAVAILABLE);
350     }
setDataLoaderStatusUnrecoverable()351     void setDataLoaderStatusUnrecoverable() {
352         mListener->onStatusChanged(mId, IDataLoaderStatusListener::DATA_LOADER_UNRECOVERABLE);
353     }
unbindFromDataLoaderOk(int32_t id)354     binder::Status unbindFromDataLoaderOk(int32_t id) {
355         mBindDelayMs = -1;
356         if (mDataLoader) {
357             if (auto status = mDataLoader->destroy(id); !status.isOk()) {
358                 return status;
359             }
360             mDataLoader = nullptr;
361         } else if (mListener) {
362             mListener->onStatusChanged(id, IDataLoaderStatusListener::DATA_LOADER_DESTROYED);
363         }
364         return binder::Status::ok();
365     }
366 
bindDelayMs() const367     int bindDelayMs() const { return mBindDelayMs; }
368 
369 private:
370     int mId = -1;
371     int mBindDelayMs = -1;
372     sp<IDataLoaderStatusListener> mListener;
373     sp<IDataLoader> mDataLoader;
374     sp<IDataLoader> mDataLoaderHolder;
375 };
376 
377 class MockIncFs : public IncFsWrapper {
378 public:
379     MOCK_CONST_METHOD0(features, Features());
380     MOCK_CONST_METHOD1(listExistingMounts, void(const ExistingMountCallback& cb));
381     MOCK_CONST_METHOD1(openMount, Control(std::string_view path));
382     MOCK_CONST_METHOD4(createControl,
383                        Control(IncFsFd cmd, IncFsFd pendingReads, IncFsFd logs,
384                                IncFsFd blocksWritten));
385     MOCK_CONST_METHOD5(makeFile,
386                        ErrorCode(const Control& control, std::string_view path, int mode, FileId id,
387                                  NewFileParams params));
388     MOCK_CONST_METHOD4(makeMappedFile,
389                        ErrorCode(const Control& control, std::string_view path, int mode,
390                                  NewMappedFileParams params));
391     MOCK_CONST_METHOD3(makeDir, ErrorCode(const Control& control, std::string_view path, int mode));
392     MOCK_CONST_METHOD3(makeDirs,
393                        ErrorCode(const Control& control, std::string_view path, int mode));
394     MOCK_CONST_METHOD2(getMetadata, RawMetadata(const Control& control, FileId fileid));
395     MOCK_CONST_METHOD2(getMetadata, RawMetadata(const Control& control, std::string_view path));
396     MOCK_CONST_METHOD2(getFileId, FileId(const Control& control, std::string_view path));
397     MOCK_CONST_METHOD2(countFilledBlocks,
398                        std::pair<IncFsBlockIndex, IncFsBlockIndex>(const Control& control,
399                                                                    std::string_view path));
400     MOCK_CONST_METHOD2(isFileFullyLoaded,
401                        incfs::LoadingState(const Control& control, std::string_view path));
402     MOCK_CONST_METHOD2(isFileFullyLoaded, incfs::LoadingState(const Control& control, FileId id));
403     MOCK_CONST_METHOD1(isEverythingFullyLoaded, incfs::LoadingState(const Control& control));
404     MOCK_CONST_METHOD3(link,
405                        ErrorCode(const Control& control, std::string_view from,
406                                  std::string_view to));
407     MOCK_CONST_METHOD2(unlink, ErrorCode(const Control& control, std::string_view path));
408     MOCK_CONST_METHOD2(openForSpecialOps, UniqueFd(const Control& control, FileId id));
409     MOCK_CONST_METHOD1(writeBlocks, ErrorCode(std::span<const DataBlock> blocks));
410     MOCK_CONST_METHOD3(reserveSpace, ErrorCode(const Control& control, FileId id, IncFsSize size));
411     MOCK_CONST_METHOD3(waitForPendingReads,
412                        WaitResult(const Control& control, std::chrono::milliseconds timeout,
413                                   std::vector<incfs::ReadInfoWithUid>* pendingReadsBuffer));
414     MOCK_CONST_METHOD2(setUidReadTimeouts,
415                        ErrorCode(const Control& control,
416                                  const std::vector<PerUidReadTimeouts>& perUidReadTimeouts));
417     MOCK_CONST_METHOD2(forEachFile, ErrorCode(const Control& control, FileCallback cb));
418     MOCK_CONST_METHOD2(forEachIncompleteFile, ErrorCode(const Control& control, FileCallback cb));
419     MOCK_CONST_METHOD1(getMetrics, std::optional<Metrics>(std::string_view path));
420     MOCK_CONST_METHOD1(getLastReadError, std::optional<LastReadError>(const Control& control));
421 
MockIncFs()422     MockIncFs() {
423         ON_CALL(*this, listExistingMounts(_)).WillByDefault(Return());
424         ON_CALL(*this, reserveSpace(_, _, _)).WillByDefault(Return(0));
425     }
426 
makeFileFails()427     void makeFileFails() { ON_CALL(*this, makeFile(_, _, _, _, _)).WillByDefault(Return(-1)); }
makeFileSuccess()428     void makeFileSuccess() { ON_CALL(*this, makeFile(_, _, _, _, _)).WillByDefault(Return(0)); }
429 
countFilledBlocksSuccess()430     void countFilledBlocksSuccess() {
431         ON_CALL(*this, countFilledBlocks(_, _)).WillByDefault(Return(std::make_pair(1, 2)));
432     }
433 
countFilledBlocksFullyLoaded()434     void countFilledBlocksFullyLoaded() {
435         ON_CALL(*this, countFilledBlocks(_, _)).WillByDefault(Return(std::make_pair(10000, 10000)));
436     }
437 
countFilledBlocksFails()438     void countFilledBlocksFails() {
439         ON_CALL(*this, countFilledBlocks(_, _)).WillByDefault(Return(std::make_pair(-1, -1)));
440     }
441 
countFilledBlocksEmpty()442     void countFilledBlocksEmpty() {
443         ON_CALL(*this, countFilledBlocks(_, _)).WillByDefault(Return(std::make_pair(0, 0)));
444     }
445 
openMountSuccess()446     void openMountSuccess() {
447         ON_CALL(*this, openMount(_)).WillByDefault(Invoke(this, &MockIncFs::openMountForHealth));
448     }
449 
450     // 1000ms
waitForPendingReadsSuccess(uint64_t ts=0)451     void waitForPendingReadsSuccess(uint64_t ts = 0) {
452         ON_CALL(*this, waitForPendingReads(_, _, _))
453                 .WillByDefault(
454                         Invoke([ts](const Control& control, std::chrono::milliseconds timeout,
455                                     std::vector<incfs::ReadInfoWithUid>* pendingReadsBuffer) {
456                             pendingReadsBuffer->push_back({.bootClockTsUs = ts});
457                             return android::incfs::WaitResult::HaveData;
458                         }));
459     }
460 
waitForPendingReadsTimeout()461     void waitForPendingReadsTimeout() {
462         ON_CALL(*this, waitForPendingReads(_, _, _))
463                 .WillByDefault(Return(android::incfs::WaitResult::Timeout));
464     }
465 
466     static constexpr auto kPendingReadsFd = 42;
openMountForHealth(std::string_view)467     Control openMountForHealth(std::string_view) {
468         return UniqueControl(IncFs_CreateControl(-1, kPendingReadsFd, -1, -1));
469     }
470 
getMountInfoMetadata(const Control & control,std::string_view path)471     RawMetadata getMountInfoMetadata(const Control& control, std::string_view path) {
472         metadata::Mount m;
473         m.mutable_storage()->set_id(100);
474         m.mutable_loader()->set_package_name("com.test");
475         m.mutable_loader()->set_arguments("com.uri");
476         const auto metadata = m.SerializeAsString();
477         static_cast<void>(m.mutable_loader()->release_arguments());
478         static_cast<void>(m.mutable_loader()->release_package_name());
479         return {metadata.begin(), metadata.end()};
480     }
getStorageMetadata(const Control & control,std::string_view path)481     RawMetadata getStorageMetadata(const Control& control, std::string_view path) {
482         metadata::Storage st;
483         st.set_id(100);
484         auto metadata = st.SerializeAsString();
485         return {metadata.begin(), metadata.end()};
486     }
getBindPointMetadata(const Control & control,std::string_view path)487     RawMetadata getBindPointMetadata(const Control& control, std::string_view path) {
488         metadata::BindPoint bp;
489         std::string destPath = "dest";
490         std::string srcPath = "src";
491         bp.set_storage_id(100);
492         bp.set_allocated_dest_path(&destPath);
493         bp.set_allocated_source_subdir(&srcPath);
494         const auto metadata = bp.SerializeAsString();
495         static_cast<void>(bp.release_source_subdir());
496         static_cast<void>(bp.release_dest_path());
497         return std::vector<char>(metadata.begin(), metadata.end());
498     }
499 };
500 
501 class MockAppOpsManager : public AppOpsManagerWrapper {
502 public:
503     MOCK_CONST_METHOD3(checkPermission, binder::Status(const char*, const char*, const char*));
504     MOCK_METHOD3(startWatchingMode, void(int32_t, const String16&, const sp<IAppOpsCallback>&));
505     MOCK_METHOD1(stopWatchingMode, void(const sp<IAppOpsCallback>&));
506 
checkPermissionSuccess()507     void checkPermissionSuccess() {
508         ON_CALL(*this, checkPermission(_, _, _)).WillByDefault(Return(android::incremental::Ok()));
509     }
checkPermissionNoCrossUsers()510     void checkPermissionNoCrossUsers() {
511         ON_CALL(*this,
512                 checkPermission("android.permission.LOADER_USAGE_STATS",
513                                 "android:loader_usage_stats", _))
514                 .WillByDefault(Return(android::incremental::Ok()));
515         ON_CALL(*this, checkPermission("android.permission.INTERACT_ACROSS_USERS", nullptr, _))
516                 .WillByDefault(
517                         Return(android::incremental::Exception(binder::Status::EX_SECURITY, {})));
518     }
checkPermissionFails()519     void checkPermissionFails() {
520         ON_CALL(*this, checkPermission(_, _, _))
521                 .WillByDefault(
522                         Return(android::incremental::Exception(binder::Status::EX_SECURITY, {})));
523     }
initializeStartWatchingMode()524     void initializeStartWatchingMode() {
525         ON_CALL(*this, startWatchingMode(_, _, _))
526                 .WillByDefault(Invoke(this, &MockAppOpsManager::storeCallback));
527     }
storeCallback(int32_t,const String16 &,const sp<IAppOpsCallback> & cb)528     void storeCallback(int32_t, const String16&, const sp<IAppOpsCallback>& cb) {
529         mStoredCallback = cb;
530     }
531 
532     sp<IAppOpsCallback> mStoredCallback;
533 };
534 
535 class MockJniWrapper : public JniWrapper {
536 public:
537     MOCK_CONST_METHOD0(initializeForCurrentThread, void());
538 
MockJniWrapper()539     MockJniWrapper() { EXPECT_CALL(*this, initializeForCurrentThread()).Times(2); }
540 };
541 
542 class MockLooperWrapper : public LooperWrapper {
543 public:
544     MOCK_METHOD5(addFd, int(int, int, int, android::Looper_callbackFunc, void*));
545     MOCK_METHOD1(removeFd, int(int));
546     MOCK_METHOD0(wake, void());
547     MOCK_METHOD1(pollAll, int(int));
548 
MockLooperWrapper()549     MockLooperWrapper() {
550         ON_CALL(*this, addFd(_, _, _, _, _))
551                 .WillByDefault(Invoke(this, &MockLooperWrapper::storeCallback));
552         ON_CALL(*this, removeFd(_)).WillByDefault(Invoke(this, &MockLooperWrapper::clearCallback));
553         ON_CALL(*this, pollAll(_)).WillByDefault(Invoke(this, &MockLooperWrapper::wait10Ms));
554     }
555 
storeCallback(int,int,int,android::Looper_callbackFunc callback,void * data)556     int storeCallback(int, int, int, android::Looper_callbackFunc callback, void* data) {
557         mCallback = callback;
558         mCallbackData = data;
559         return 0;
560     }
561 
clearCallback(int)562     int clearCallback(int) {
563         mCallback = nullptr;
564         mCallbackData = nullptr;
565         return 0;
566     }
567 
wait10Ms(int)568     int wait10Ms(int) {
569         // This is called from a loop in runCmdLooper.
570         // Sleeping for 10ms only to avoid busy looping.
571         std::this_thread::sleep_for(10ms);
572         return 0;
573     }
574 
575     android::Looper_callbackFunc mCallback = nullptr;
576     void* mCallbackData = nullptr;
577 };
578 
579 class MockTimedQueueWrapper : public TimedQueueWrapper {
580 public:
581     MOCK_METHOD3(addJob, void(MountId, Milliseconds, Job));
582     MOCK_METHOD1(removeJobs, void(MountId));
583     MOCK_METHOD0(stop, void());
584 
MockTimedQueueWrapper()585     MockTimedQueueWrapper() {
586         ON_CALL(*this, addJob(_, _, _))
587                 .WillByDefault(Invoke(this, &MockTimedQueueWrapper::storeJob));
588         ON_CALL(*this, removeJobs(_)).WillByDefault(Invoke(this, &MockTimedQueueWrapper::clearJob));
589     }
590 
storeJob(MountId id,Milliseconds after,Job what)591     void storeJob(MountId id, Milliseconds after, Job what) {
592         mId = id;
593         mAfter = after;
594         mWhat = std::move(what);
595     }
596 
clearJob(MountId id)597     void clearJob(MountId id) {
598         if (mId == id) {
599             mAfter = {};
600             mWhat = {};
601         }
602     }
603 
604     MountId mId = -1;
605     Milliseconds mAfter;
606     Job mWhat;
607 };
608 
609 class MockFsWrapper : public FsWrapper {
610 public:
611     MOCK_CONST_METHOD2(listFilesRecursive, void(std::string_view, FileCallback));
hasNoFile()612     void hasNoFile() { ON_CALL(*this, listFilesRecursive(_, _)).WillByDefault(Return()); }
hasFiles()613     void hasFiles() {
614         ON_CALL(*this, listFilesRecursive(_, _))
615                 .WillByDefault(Invoke(this, &MockFsWrapper::fakeFiles));
616     }
fakeFiles(std::string_view directoryPath,FileCallback onFile)617     void fakeFiles(std::string_view directoryPath, FileCallback onFile) {
618         for (auto file : {"base.apk", "split.apk", "lib/a.so"}) {
619             if (!onFile(file)) break;
620         }
621     }
622 };
623 
624 class MockClockWrapper : public ClockWrapper {
625 public:
626     MOCK_CONST_METHOD0(now, TimePoint());
627 
start()628     void start() { ON_CALL(*this, now()).WillByDefault(Invoke(this, &MockClockWrapper::getClock)); }
629 
630     template <class Delta>
advance(Delta delta)631     void advance(Delta delta) {
632         mClock += delta;
633     }
634 
advanceMs(int deltaMs)635     void advanceMs(int deltaMs) { mClock += std::chrono::milliseconds(deltaMs); }
636 
getClock() const637     TimePoint getClock() const { return mClock; }
getClockMono() const638     std::optional<timespec> getClockMono() const {
639         const auto nsSinceEpoch =
640                 std::chrono::duration_cast<std::chrono::nanoseconds>(mClock.time_since_epoch())
641                         .count();
642         timespec ts = {.tv_sec = static_cast<time_t>(nsSinceEpoch / 1000000000LL),
643                        .tv_nsec = static_cast<long>(nsSinceEpoch % 1000000000LL)};
644         return ts;
645     }
646 
647     TimePoint mClock = Clock::now();
648 };
649 
650 class MockStorageHealthListener : public os::incremental::BnStorageHealthListener {
651 public:
652     MOCK_METHOD2(onHealthStatus, binder::Status(int32_t storageId, int32_t status));
653 
MockStorageHealthListener()654     MockStorageHealthListener() {
655         ON_CALL(*this, onHealthStatus(_, _))
656                 .WillByDefault(Invoke(this, &MockStorageHealthListener::storeStorageIdAndStatus));
657     }
658 
storeStorageIdAndStatus(int32_t storageId,int32_t status)659     binder::Status storeStorageIdAndStatus(int32_t storageId, int32_t status) {
660         mStorageId = storageId;
661         mStatus = status;
662         return binder::Status::ok();
663     }
664 
665     int32_t mStorageId = -1;
666     int32_t mStatus = -1;
667 };
668 
669 class MockStorageLoadingProgressListener : public IStorageLoadingProgressListener {
670 public:
671     MockStorageLoadingProgressListener() = default;
672     MOCK_METHOD2(onStorageLoadingProgressChanged,
673                  binder::Status(int32_t storageId, float progress));
674     MOCK_METHOD0(onAsBinder, IBinder*());
675 };
676 
677 class MockServiceManager : public ServiceManagerWrapper {
678 public:
MockServiceManager(std::unique_ptr<MockVoldService> vold,std::unique_ptr<MockDataLoaderManager> dataLoaderManager,std::unique_ptr<MockIncFs> incfs,std::unique_ptr<MockAppOpsManager> appOpsManager,std::unique_ptr<MockJniWrapper> jni,std::unique_ptr<MockLooperWrapper> looper,std::unique_ptr<MockTimedQueueWrapper> timedQueue,std::unique_ptr<MockTimedQueueWrapper> progressUpdateJobQueue,std::unique_ptr<MockFsWrapper> fs,std::unique_ptr<MockClockWrapper> clock)679     MockServiceManager(std::unique_ptr<MockVoldService> vold,
680                        std::unique_ptr<MockDataLoaderManager> dataLoaderManager,
681                        std::unique_ptr<MockIncFs> incfs,
682                        std::unique_ptr<MockAppOpsManager> appOpsManager,
683                        std::unique_ptr<MockJniWrapper> jni,
684                        std::unique_ptr<MockLooperWrapper> looper,
685                        std::unique_ptr<MockTimedQueueWrapper> timedQueue,
686                        std::unique_ptr<MockTimedQueueWrapper> progressUpdateJobQueue,
687                        std::unique_ptr<MockFsWrapper> fs, std::unique_ptr<MockClockWrapper> clock)
688           : mVold(std::move(vold)),
689             mDataLoaderManager(std::move(dataLoaderManager)),
690             mIncFs(std::move(incfs)),
691             mAppOpsManager(std::move(appOpsManager)),
692             mJni(std::move(jni)),
693             mLooper(std::move(looper)),
694             mTimedQueue(std::move(timedQueue)),
695             mProgressUpdateJobQueue(std::move(progressUpdateJobQueue)),
696             mFs(std::move(fs)),
697             mClock(std::move(clock)) {}
getVoldService()698     std::unique_ptr<VoldServiceWrapper> getVoldService() final { return std::move(mVold); }
getDataLoaderManager()699     std::unique_ptr<DataLoaderManagerWrapper> getDataLoaderManager() final {
700         return std::move(mDataLoaderManager);
701     }
getIncFs()702     std::unique_ptr<IncFsWrapper> getIncFs() final { return std::move(mIncFs); }
getAppOpsManager()703     std::unique_ptr<AppOpsManagerWrapper> getAppOpsManager() final {
704         return std::move(mAppOpsManager);
705     }
getJni()706     std::unique_ptr<JniWrapper> getJni() final { return std::move(mJni); }
getLooper()707     std::unique_ptr<LooperWrapper> getLooper() final { return std::move(mLooper); }
getTimedQueue()708     std::unique_ptr<TimedQueueWrapper> getTimedQueue() final { return std::move(mTimedQueue); }
getProgressUpdateJobQueue()709     std::unique_ptr<TimedQueueWrapper> getProgressUpdateJobQueue() final {
710         return std::move(mProgressUpdateJobQueue);
711     }
getFs()712     std::unique_ptr<FsWrapper> getFs() final { return std::move(mFs); }
getClock()713     std::unique_ptr<ClockWrapper> getClock() final { return std::move(mClock); }
714 
715 private:
716     std::unique_ptr<MockVoldService> mVold;
717     std::unique_ptr<MockDataLoaderManager> mDataLoaderManager;
718     std::unique_ptr<MockIncFs> mIncFs;
719     std::unique_ptr<MockAppOpsManager> mAppOpsManager;
720     std::unique_ptr<MockJniWrapper> mJni;
721     std::unique_ptr<MockLooperWrapper> mLooper;
722     std::unique_ptr<MockTimedQueueWrapper> mTimedQueue;
723     std::unique_ptr<MockTimedQueueWrapper> mProgressUpdateJobQueue;
724     std::unique_ptr<MockFsWrapper> mFs;
725     std::unique_ptr<MockClockWrapper> mClock;
726 };
727 
728 // --- IncrementalServiceTest ---
729 
730 class IncrementalServiceTest : public testing::Test {
731 public:
SetUp()732     void SetUp() override {
733         auto vold = std::make_unique<NiceMock<MockVoldService>>();
734         mVold = vold.get();
735         sp<NiceMock<MockDataLoader>> dataLoader{new NiceMock<MockDataLoader>};
736         mDataLoader = dataLoader.get();
737         auto dataloaderManager = std::make_unique<NiceMock<MockDataLoaderManager>>(dataLoader);
738         mDataLoaderManager = dataloaderManager.get();
739         auto incFs = std::make_unique<NiceMock<MockIncFs>>();
740         mIncFs = incFs.get();
741         auto appOps = std::make_unique<NiceMock<MockAppOpsManager>>();
742         mAppOpsManager = appOps.get();
743         auto jni = std::make_unique<NiceMock<MockJniWrapper>>();
744         mJni = jni.get();
745         auto looper = std::make_unique<NiceMock<MockLooperWrapper>>();
746         mLooper = looper.get();
747         auto timedQueue = std::make_unique<NiceMock<MockTimedQueueWrapper>>();
748         mTimedQueue = timedQueue.get();
749         auto progressUpdateJobQueue = std::make_unique<NiceMock<MockTimedQueueWrapper>>();
750         mProgressUpdateJobQueue = progressUpdateJobQueue.get();
751         auto fs = std::make_unique<NiceMock<MockFsWrapper>>();
752         mFs = fs.get();
753         auto clock = std::make_unique<NiceMock<MockClockWrapper>>();
754         mClock = clock.get();
755         mIncrementalService = std::make_unique<
756                 IncrementalService>(MockServiceManager(std::move(vold),
757                                                        std::move(dataloaderManager),
758                                                        std::move(incFs), std::move(appOps),
759                                                        std::move(jni), std::move(looper),
760                                                        std::move(timedQueue),
761                                                        std::move(progressUpdateJobQueue),
762                                                        std::move(fs), std::move(clock)),
763                                     mRootDir.path);
764         mDataLoaderParcel.packageName = "com.test";
765         mDataLoaderParcel.arguments = "uri";
766         mDataLoaderManager->unbindFromDataLoaderSuccess();
767         mIncrementalService->onSystemReady();
768         mClock->start();
769         setupSuccess();
770     }
771 
setUpExistingMountDir(const std::string & rootDir)772     void setUpExistingMountDir(const std::string& rootDir) {
773         const auto dir = rootDir + "/dir1";
774         const auto mountDir = dir + "/mount";
775         const auto backingDir = dir + "/backing_store";
776         const auto storageDir = mountDir + "/st0";
777         ASSERT_EQ(0, mkdir(dir.c_str(), 0755));
778         ASSERT_EQ(0, mkdir(mountDir.c_str(), 0755));
779         ASSERT_EQ(0, mkdir(backingDir.c_str(), 0755));
780         ASSERT_EQ(0, mkdir(storageDir.c_str(), 0755));
781         const auto mountInfoFile = rootDir + "/dir1/mount/.info";
782         const auto mountPointsFile = rootDir + "/dir1/mount/.mountpoint.abcd";
783         ASSERT_TRUE(base::WriteStringToFile("info", mountInfoFile));
784         ASSERT_TRUE(base::WriteStringToFile("mounts", mountPointsFile));
785         ON_CALL(*mIncFs, getMetadata(_, std::string_view(mountInfoFile)))
786                 .WillByDefault(Invoke(mIncFs, &MockIncFs::getMountInfoMetadata));
787         ON_CALL(*mIncFs, getMetadata(_, std::string_view(mountPointsFile)))
788                 .WillByDefault(Invoke(mIncFs, &MockIncFs::getBindPointMetadata));
789         ON_CALL(*mIncFs, getMetadata(_, std::string_view(rootDir + "/dir1/mount/st0")))
790                 .WillByDefault(Invoke(mIncFs, &MockIncFs::getStorageMetadata));
791     }
792 
setupSuccess()793     void setupSuccess() {
794         mVold->mountIncFsSuccess();
795         mIncFs->makeFileSuccess();
796         mVold->bindMountSuccess();
797         mDataLoaderManager->bindToDataLoaderSuccess();
798         mDataLoaderManager->getDataLoaderSuccess();
799     }
800 
checkHealthMetrics(int storageId,long expectedMillisSinceOldestPendingRead,int expectedStorageHealthStatusCode)801     void checkHealthMetrics(int storageId, long expectedMillisSinceOldestPendingRead,
802                             int expectedStorageHealthStatusCode) {
803         android::os::PersistableBundle result{};
804         mIncrementalService->getMetrics(storageId, &result);
805         ASSERT_EQ(6, (int)result.size());
806         int64_t millisSinceOldestPendingRead = -1;
807         ASSERT_TRUE(result.getLong(String16(BnIncrementalService::
808                                                     METRICS_MILLIS_SINCE_OLDEST_PENDING_READ()
809                                                             .c_str()),
810                                    &millisSinceOldestPendingRead));
811         // Allow 10ms.
812         ASSERT_LE(expectedMillisSinceOldestPendingRead, millisSinceOldestPendingRead);
813         ASSERT_GE(expectedMillisSinceOldestPendingRead + 10, millisSinceOldestPendingRead);
814         int storageHealthStatusCode = -1;
815         ASSERT_TRUE(
816                 result.getInt(String16(BnIncrementalService::METRICS_STORAGE_HEALTH_STATUS_CODE()
817                                                .c_str()),
818                               &storageHealthStatusCode));
819         ASSERT_EQ(expectedStorageHealthStatusCode, storageHealthStatusCode);
820         int dataLoaderStatusCode = -1;
821         ASSERT_TRUE(result.getInt(String16(BnIncrementalService::METRICS_DATA_LOADER_STATUS_CODE()
822                                                    .c_str()),
823                                   &dataLoaderStatusCode));
824         ASSERT_EQ(IDataLoaderStatusListener::DATA_LOADER_STARTED, dataLoaderStatusCode);
825     }
826 
checkBindingMetrics(int storageId,int64_t expectedMillisSinceLastDataLoaderBind,int64_t expectedDataLoaderBindDelayMillis)827     void checkBindingMetrics(int storageId, int64_t expectedMillisSinceLastDataLoaderBind,
828                              int64_t expectedDataLoaderBindDelayMillis) {
829         android::os::PersistableBundle result{};
830         mIncrementalService->getMetrics(storageId, &result);
831         ASSERT_EQ(6, (int)result.size());
832         int dataLoaderStatus = -1;
833         ASSERT_TRUE(result.getInt(String16(BnIncrementalService::METRICS_DATA_LOADER_STATUS_CODE()
834                                                    .c_str()),
835                                   &dataLoaderStatus));
836         ASSERT_EQ(IDataLoaderStatusListener::DATA_LOADER_STARTED, dataLoaderStatus);
837         int64_t millisSinceLastDataLoaderBind = -1;
838         ASSERT_TRUE(result.getLong(String16(BnIncrementalService::
839                                                     METRICS_MILLIS_SINCE_LAST_DATA_LOADER_BIND()
840                                                             .c_str()),
841                                    &millisSinceLastDataLoaderBind));
842         ASSERT_EQ(expectedMillisSinceLastDataLoaderBind, millisSinceLastDataLoaderBind);
843         int64_t dataLoaderBindDelayMillis = -1;
844         ASSERT_TRUE(
845                 result.getLong(String16(
846                                        BnIncrementalService::METRICS_DATA_LOADER_BIND_DELAY_MILLIS()
847                                                .c_str()),
848                                &dataLoaderBindDelayMillis));
849         ASSERT_EQ(expectedDataLoaderBindDelayMillis, dataLoaderBindDelayMillis);
850     }
851 
852 protected:
853     NiceMock<MockVoldService>* mVold = nullptr;
854     NiceMock<MockIncFs>* mIncFs = nullptr;
855     NiceMock<MockDataLoaderManager>* mDataLoaderManager = nullptr;
856     NiceMock<MockAppOpsManager>* mAppOpsManager = nullptr;
857     NiceMock<MockJniWrapper>* mJni = nullptr;
858     NiceMock<MockLooperWrapper>* mLooper = nullptr;
859     NiceMock<MockTimedQueueWrapper>* mTimedQueue = nullptr;
860     NiceMock<MockTimedQueueWrapper>* mProgressUpdateJobQueue = nullptr;
861     NiceMock<MockFsWrapper>* mFs = nullptr;
862     NiceMock<MockClockWrapper>* mClock = nullptr;
863     NiceMock<MockDataLoader>* mDataLoader = nullptr;
864     std::unique_ptr<IncrementalService> mIncrementalService;
865     TemporaryDir mRootDir;
866     DataLoaderParamsParcel mDataLoaderParcel;
867 };
868 
TEST_F(IncrementalServiceTest,testCreateStorageMountIncFsFails)869 TEST_F(IncrementalServiceTest, testCreateStorageMountIncFsFails) {
870     mVold->mountIncFsFails();
871     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(0);
872     TemporaryDir tempDir;
873     int storageId =
874             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
875                                                IncrementalService::CreateOptions::CreateNew);
876     ASSERT_LT(storageId, 0);
877 }
878 
TEST_F(IncrementalServiceTest,testCreateStorageMountIncFsInvalidControlParcel)879 TEST_F(IncrementalServiceTest, testCreateStorageMountIncFsInvalidControlParcel) {
880     mVold->mountIncFsInvalidControlParcel();
881     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(0);
882     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(0);
883     TemporaryDir tempDir;
884     int storageId =
885             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
886                                                IncrementalService::CreateOptions::CreateNew);
887     ASSERT_LT(storageId, 0);
888 }
889 
TEST_F(IncrementalServiceTest,testCreateStorageMakeFileFails)890 TEST_F(IncrementalServiceTest, testCreateStorageMakeFileFails) {
891     mVold->mountIncFsSuccess();
892     mIncFs->makeFileFails();
893     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(0);
894     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(0);
895     EXPECT_CALL(*mVold, unmountIncFs(_));
896     TemporaryDir tempDir;
897     int storageId =
898             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
899                                                IncrementalService::CreateOptions::CreateNew);
900     ASSERT_LT(storageId, 0);
901 }
902 
TEST_F(IncrementalServiceTest,testCreateStorageBindMountFails)903 TEST_F(IncrementalServiceTest, testCreateStorageBindMountFails) {
904     mVold->mountIncFsSuccess();
905     mIncFs->makeFileSuccess();
906     mVold->bindMountFails();
907     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(0);
908     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(0);
909     EXPECT_CALL(*mVold, unmountIncFs(_));
910     TemporaryDir tempDir;
911     int storageId =
912             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
913                                                IncrementalService::CreateOptions::CreateNew);
914     ASSERT_LT(storageId, 0);
915 }
916 
TEST_F(IncrementalServiceTest,testCreateStoragePrepareDataLoaderFails)917 TEST_F(IncrementalServiceTest, testCreateStoragePrepareDataLoaderFails) {
918     mVold->mountIncFsSuccess();
919     mIncFs->makeFileSuccess();
920     mVold->bindMountSuccess();
921     mDataLoaderManager->bindToDataLoaderFails();
922     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
923     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(0);
924     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(0);
925     EXPECT_CALL(*mDataLoader, start(_)).Times(0);
926     EXPECT_CALL(*mDataLoader, destroy(_)).Times(0);
927     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
928     TemporaryDir tempDir;
929     int storageId =
930             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
931                                                IncrementalService::CreateOptions::CreateNew);
932     ASSERT_GE(storageId, 0);
933     mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {}, {}, {});
934 }
935 
TEST_F(IncrementalServiceTest,testDeleteStorageSuccess)936 TEST_F(IncrementalServiceTest, testDeleteStorageSuccess) {
937     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
938     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
939     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
940     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
941     TemporaryDir tempDir;
942     int storageId =
943             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
944                                                IncrementalService::CreateOptions::CreateNew);
945     ASSERT_GE(storageId, 0);
946     mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {}, {}, {});
947     mIncrementalService->deleteStorage(storageId);
948 }
949 
TEST_F(IncrementalServiceTest,testDataLoaderDestroyedAndDelayed)950 TEST_F(IncrementalServiceTest, testDataLoaderDestroyedAndDelayed) {
951     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(7);
952     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
953     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(7);
954     EXPECT_CALL(*mDataLoader, start(_)).Times(7);
955     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
956     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
957     TemporaryDir tempDir;
958     int storageId =
959             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
960                                                IncrementalService::CreateOptions::CreateNew);
961     ASSERT_GE(storageId, 0);
962     mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {}, {}, {});
963 
964     // Simulated crash/other connection breakage.
965 
966     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
967             .WillByDefault(Invoke(mDataLoaderManager,
968                                   &MockDataLoaderManager::bindToDataLoaderOkWith1sDelay));
969     checkBindingMetrics(storageId, 0, 0);
970     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
971     checkBindingMetrics(storageId, 0, 0);
972     mDataLoaderManager->setDataLoaderStatusDestroyed();
973 
974     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
975             .WillByDefault(Invoke(mDataLoaderManager,
976                                   &MockDataLoaderManager::bindToDataLoaderOkWith10sDelay));
977     checkBindingMetrics(storageId, 0, mDataLoaderManager->bindDelayMs());
978     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
979     checkBindingMetrics(storageId, mDataLoaderManager->bindDelayMs(),
980                         mDataLoaderManager->bindDelayMs());
981     mDataLoaderManager->setDataLoaderStatusDestroyed();
982 
983     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
984             .WillByDefault(Invoke(mDataLoaderManager,
985                                   &MockDataLoaderManager::bindToDataLoaderOkWith100sDelay));
986     checkBindingMetrics(storageId, 0, mDataLoaderManager->bindDelayMs());
987     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
988     checkBindingMetrics(storageId, mDataLoaderManager->bindDelayMs(),
989                         mDataLoaderManager->bindDelayMs());
990     mDataLoaderManager->setDataLoaderStatusDestroyed();
991 
992     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
993             .WillByDefault(Invoke(mDataLoaderManager,
994                                   &MockDataLoaderManager::bindToDataLoaderOkWith1000sDelay));
995     checkBindingMetrics(storageId, 0, mDataLoaderManager->bindDelayMs());
996     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
997     checkBindingMetrics(storageId, mDataLoaderManager->bindDelayMs(),
998                         mDataLoaderManager->bindDelayMs());
999     mDataLoaderManager->setDataLoaderStatusDestroyed();
1000 
1001     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1002             .WillByDefault(Invoke(mDataLoaderManager,
1003                                   &MockDataLoaderManager::bindToDataLoaderOkWith10000sDelay));
1004     checkBindingMetrics(storageId, 0, mDataLoaderManager->bindDelayMs());
1005     // Try the reduced delay, just in case.
1006     mClock->advanceMs(mDataLoaderManager->bindDelayMs() / 2);
1007     checkBindingMetrics(storageId, mDataLoaderManager->bindDelayMs() / 2,
1008                         mDataLoaderManager->bindDelayMs());
1009     mDataLoaderManager->setDataLoaderStatusDestroyed();
1010 
1011     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1012             .WillByDefault(Invoke(mDataLoaderManager,
1013                                   &MockDataLoaderManager::bindToDataLoaderOkWith10000sDelay));
1014     checkBindingMetrics(storageId, 0, mDataLoaderManager->bindDelayMs());
1015     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
1016     checkBindingMetrics(storageId, mDataLoaderManager->bindDelayMs(),
1017                         mDataLoaderManager->bindDelayMs());
1018     mDataLoaderManager->setDataLoaderStatusDestroyed();
1019 }
1020 
TEST_F(IncrementalServiceTest,testDataLoaderOnRestart)1021 TEST_F(IncrementalServiceTest, testDataLoaderOnRestart) {
1022     mIncFs->waitForPendingReadsSuccess();
1023     mIncFs->openMountSuccess();
1024 
1025     constexpr auto bindRetryInterval = 5s;
1026 
1027     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(11);
1028     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
1029     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(7);
1030     EXPECT_CALL(*mDataLoader, start(_)).Times(7);
1031     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1032     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1033     EXPECT_CALL(*mTimedQueue, addJob(_, _, _)).Times(4);
1034     TemporaryDir tempDir;
1035     int storageId =
1036             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1037                                                IncrementalService::CreateOptions::CreateNew);
1038     ASSERT_GE(storageId, 0);
1039 
1040     // First binds to DataLoader fails... because it's restart.
1041     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1042             .WillByDefault(Invoke(mDataLoaderManager,
1043                                   &MockDataLoaderManager::bindToDataLoaderNotOkWithNoDelay));
1044 
1045     // Request DL start.
1046     mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {}, {}, {});
1047 
1048     // Retry callback present.
1049     ASSERT_EQ(storageId, mTimedQueue->mId);
1050     ASSERT_EQ(mTimedQueue->mAfter, bindRetryInterval);
1051     auto retryCallback = mTimedQueue->mWhat;
1052     mTimedQueue->clearJob(storageId);
1053 
1054     // Expecting the same bindToDataLoaderNotOkWithNoDelay call.
1055     mClock->advance(5s);
1056 
1057     retryCallback();
1058     // Retry callback present.
1059     ASSERT_EQ(storageId, mTimedQueue->mId);
1060     ASSERT_EQ(mTimedQueue->mAfter, bindRetryInterval);
1061     retryCallback = mTimedQueue->mWhat;
1062     mTimedQueue->clearJob(storageId);
1063 
1064     // Returning "binding" so that we can retry.
1065     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1066             .WillByDefault(Invoke(mDataLoaderManager,
1067                                   &MockDataLoaderManager::bindToDataLoaderBindingWithNoDelay));
1068 
1069     // Expecting bindToDataLoaderBindingWithNoDelay call.
1070     mClock->advance(5s);
1071 
1072     retryCallback();
1073     // No retry callback.
1074     ASSERT_EQ(mTimedQueue->mAfter, 0ms);
1075     ASSERT_EQ(mTimedQueue->mWhat, nullptr);
1076 
1077     // Should not change the bindToDataLoader call count
1078     ASSERT_NE(nullptr, mLooper->mCallback);
1079     ASSERT_NE(nullptr, mLooper->mCallbackData);
1080     auto looperCb = mLooper->mCallback;
1081     auto looperCbData = mLooper->mCallbackData;
1082     looperCb(-1, -1, looperCbData);
1083 
1084     // Expecting the same bindToDataLoaderBindingWithNoDelay call.
1085     mClock->advance(5s);
1086 
1087     // Use pending reads callback to trigger binding.
1088     looperCb(-1, -1, looperCbData);
1089 
1090     // No retry callback.
1091     ASSERT_EQ(mTimedQueue->mAfter, 0ms);
1092     ASSERT_EQ(mTimedQueue->mWhat, nullptr);
1093 
1094     // Now we are out of 10m "retry" budget, let's finally bind.
1095     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1096             .WillByDefault(Invoke(mDataLoaderManager, &MockDataLoaderManager::bindToDataLoaderOk));
1097     mClock->advance(11min);
1098 
1099     // Use pending reads callback to trigger binding.
1100     looperCb(-1, -1, looperCbData);
1101 
1102     // No retry callback.
1103     ASSERT_EQ(mTimedQueue->mAfter, 0ms);
1104     ASSERT_EQ(mTimedQueue->mWhat, nullptr);
1105 
1106     // And test the rest of the backoff.
1107     // Simulated crash/other connection breakage.
1108     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1109             .WillByDefault(Invoke(mDataLoaderManager,
1110                                   &MockDataLoaderManager::bindToDataLoaderOkWith1sDelay));
1111     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
1112     mDataLoaderManager->setDataLoaderStatusDestroyed();
1113 
1114     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1115             .WillByDefault(Invoke(mDataLoaderManager,
1116                                   &MockDataLoaderManager::bindToDataLoaderOkWith10sDelay));
1117     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
1118     mDataLoaderManager->setDataLoaderStatusDestroyed();
1119 
1120     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1121             .WillByDefault(Invoke(mDataLoaderManager,
1122                                   &MockDataLoaderManager::bindToDataLoaderOkWith100sDelay));
1123     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
1124     mDataLoaderManager->setDataLoaderStatusDestroyed();
1125 
1126     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1127             .WillByDefault(Invoke(mDataLoaderManager,
1128                                   &MockDataLoaderManager::bindToDataLoaderOkWith1000sDelay));
1129     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
1130     mDataLoaderManager->setDataLoaderStatusDestroyed();
1131 
1132     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1133             .WillByDefault(Invoke(mDataLoaderManager,
1134                                   &MockDataLoaderManager::bindToDataLoaderOkWith10000sDelay));
1135     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
1136     mDataLoaderManager->setDataLoaderStatusDestroyed();
1137 
1138     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1139             .WillByDefault(Invoke(mDataLoaderManager,
1140                                   &MockDataLoaderManager::bindToDataLoaderOkWith10000sDelay));
1141     mClock->advanceMs(mDataLoaderManager->bindDelayMs());
1142     mDataLoaderManager->setDataLoaderStatusDestroyed();
1143 }
1144 
TEST_F(IncrementalServiceTest,testStartDataLoaderCreate)1145 TEST_F(IncrementalServiceTest, testStartDataLoaderCreate) {
1146     mDataLoader->initializeCreateOkNoStatus();
1147     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
1148     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
1149     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
1150     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
1151     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1152     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1153     TemporaryDir tempDir;
1154     int storageId =
1155             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1156                                                IncrementalService::CreateOptions::CreateNew);
1157     ASSERT_GE(storageId, 0);
1158     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1159                                                   {}, {}));
1160     mDataLoaderManager->setDataLoaderStatusCreated();
1161     mDataLoaderManager->setDataLoaderStatusStarted();
1162 }
1163 
TEST_F(IncrementalServiceTest,testStartDataLoaderPendingStart)1164 TEST_F(IncrementalServiceTest, testStartDataLoaderPendingStart) {
1165     mDataLoader->initializeCreateOkNoStatus();
1166     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
1167     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
1168     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
1169     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
1170     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1171     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1172     TemporaryDir tempDir;
1173     int storageId =
1174             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1175                                                IncrementalService::CreateOptions::CreateNew);
1176     ASSERT_GE(storageId, 0);
1177     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1178                                                   {}, {}));
1179     mDataLoaderManager->setDataLoaderStatusCreated();
1180 }
1181 
TEST_F(IncrementalServiceTest,testStartDataLoaderCreateUnavailable)1182 TEST_F(IncrementalServiceTest, testStartDataLoaderCreateUnavailable) {
1183     mDataLoader->initializeCreateOkNoStatus();
1184     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
1185     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
1186     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
1187     EXPECT_CALL(*mDataLoader, start(_)).Times(0);
1188     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1189     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1190     TemporaryDir tempDir;
1191     int storageId =
1192             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1193                                                IncrementalService::CreateOptions::CreateNew);
1194     ASSERT_GE(storageId, 0);
1195     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1196                                                   {}, {}));
1197     mDataLoaderManager->setDataLoaderStatusUnavailable();
1198 }
1199 
TEST_F(IncrementalServiceTest,testStartDataLoaderCreateUnrecoverable)1200 TEST_F(IncrementalServiceTest, testStartDataLoaderCreateUnrecoverable) {
1201     mDataLoader->initializeCreateOkNoStatus();
1202     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
1203     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
1204     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
1205     EXPECT_CALL(*mDataLoader, start(_)).Times(0);
1206     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1207     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1208     TemporaryDir tempDir;
1209     int storageId =
1210             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1211                                                IncrementalService::CreateOptions::CreateNew);
1212     ASSERT_GE(storageId, 0);
1213     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1214                                                   {}, {}));
1215     mDataLoaderManager->setDataLoaderStatusUnrecoverable();
1216 }
1217 
TEST_F(IncrementalServiceTest,testStartDataLoaderRecreateOnPendingReads)1218 TEST_F(IncrementalServiceTest, testStartDataLoaderRecreateOnPendingReads) {
1219     mIncFs->waitForPendingReadsSuccess();
1220     mIncFs->openMountSuccess();
1221     mDataLoader->initializeCreateOkNoStatus();
1222 
1223     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(2);
1224     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(2);
1225     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(2);
1226     EXPECT_CALL(*mDataLoader, start(_)).Times(0);
1227     EXPECT_CALL(*mDataLoader, destroy(_)).Times(2);
1228     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1229     EXPECT_CALL(*mLooper, addFd(MockIncFs::kPendingReadsFd, _, _, _, _)).Times(1);
1230     EXPECT_CALL(*mLooper, removeFd(MockIncFs::kPendingReadsFd)).Times(1);
1231     TemporaryDir tempDir;
1232     int storageId =
1233             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1234                                                IncrementalService::CreateOptions::CreateNew);
1235     ASSERT_GE(storageId, 0);
1236     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1237                                                   {}, {}));
1238     mDataLoaderManager->setDataLoaderStatusUnrecoverable();
1239 
1240     // Timed callback present.
1241     ASSERT_EQ(storageId, mTimedQueue->mId);
1242     ASSERT_GE(mTimedQueue->mAfter, 10ms);
1243     auto timedCallback = mTimedQueue->mWhat;
1244     mTimedQueue->clearJob(storageId);
1245 
1246     // First callback call to propagate unrecoverable.
1247     timedCallback();
1248 
1249     // And second call to trigger recreation.
1250     ASSERT_NE(nullptr, mLooper->mCallback);
1251     ASSERT_NE(nullptr, mLooper->mCallbackData);
1252     mLooper->mCallback(-1, -1, mLooper->mCallbackData);
1253 }
1254 
TEST_F(IncrementalServiceTest,testStartDataLoaderUnavailable)1255 TEST_F(IncrementalServiceTest, testStartDataLoaderUnavailable) {
1256     mIncFs->openMountSuccess();
1257     mDataLoader->initializeCreateOkNoStatus();
1258 
1259     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(3);
1260     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(3);
1261     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(3);
1262     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
1263     EXPECT_CALL(*mDataLoader, destroy(_)).Times(2);
1264     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1265     EXPECT_CALL(*mLooper, addFd(MockIncFs::kPendingReadsFd, _, _, _, _)).Times(1);
1266     EXPECT_CALL(*mLooper, removeFd(MockIncFs::kPendingReadsFd)).Times(1);
1267     TemporaryDir tempDir;
1268     int storageId =
1269             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1270                                                IncrementalService::CreateOptions::CreateNew);
1271     ASSERT_GE(storageId, 0);
1272 
1273     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1274             .WillByDefault(Invoke(mDataLoaderManager,
1275                                   &MockDataLoaderManager::bindToDataLoaderOkWithNoDelay));
1276 
1277     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1278                                                   {}, {}));
1279 
1280     // Unavailable.
1281     mDataLoaderManager->setDataLoaderStatusUnavailable();
1282 
1283     // Timed callback present.
1284     ASSERT_EQ(storageId, mTimedQueue->mId);
1285     ASSERT_GE(mTimedQueue->mAfter, 10ms);
1286     auto timedCallback = mTimedQueue->mWhat;
1287     mTimedQueue->clearJob(storageId);
1288 
1289     // Propagating unavailable and expecting it to trigger rebind with 1s retry delay.
1290     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1291             .WillByDefault(Invoke(mDataLoaderManager,
1292                                   &MockDataLoaderManager::bindToDataLoaderOkWith1sDelay));
1293     timedCallback();
1294 
1295     // Unavailable #2.
1296     mDataLoaderManager->setDataLoaderStatusUnavailable();
1297 
1298     // Timed callback present.
1299     ASSERT_EQ(storageId, mTimedQueue->mId);
1300     ASSERT_GE(mTimedQueue->mAfter, 10ms);
1301     timedCallback = mTimedQueue->mWhat;
1302     mTimedQueue->clearJob(storageId);
1303 
1304     // Propagating unavailable and expecting it to trigger rebind with 10s retry delay.
1305     // This time succeed.
1306     mDataLoader->initializeCreateOk();
1307     ON_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _))
1308             .WillByDefault(Invoke(mDataLoaderManager,
1309                                   &MockDataLoaderManager::bindToDataLoaderOkWith10sDelay));
1310     timedCallback();
1311 }
1312 
TEST_F(IncrementalServiceTest,testStartDataLoaderUnhealthyStorage)1313 TEST_F(IncrementalServiceTest, testStartDataLoaderUnhealthyStorage) {
1314     mIncFs->openMountSuccess();
1315 
1316     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
1317     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
1318     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
1319     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
1320     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1321     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1322     EXPECT_CALL(*mLooper, addFd(MockIncFs::kPendingReadsFd, _, _, _, _)).Times(2);
1323     EXPECT_CALL(*mLooper, removeFd(MockIncFs::kPendingReadsFd)).Times(2);
1324     EXPECT_CALL(*mTimedQueue, addJob(_, _, _)).Times(6);
1325 
1326     sp<NiceMock<MockStorageHealthListener>> listener{new NiceMock<MockStorageHealthListener>};
1327     NiceMock<MockStorageHealthListener>* listenerMock = listener.get();
1328     EXPECT_CALL(*listenerMock, onHealthStatus(_, IStorageHealthListener::HEALTH_STATUS_OK))
1329             .Times(2);
1330     EXPECT_CALL(*listenerMock,
1331                 onHealthStatus(_, IStorageHealthListener::HEALTH_STATUS_READS_PENDING))
1332             .Times(1);
1333     EXPECT_CALL(*listenerMock, onHealthStatus(_, IStorageHealthListener::HEALTH_STATUS_BLOCKED))
1334             .Times(1);
1335     EXPECT_CALL(*listenerMock, onHealthStatus(_, IStorageHealthListener::HEALTH_STATUS_UNHEALTHY))
1336             .Times(2);
1337 
1338     StorageHealthCheckParams params;
1339     params.blockedTimeoutMs = 10000;
1340     params.unhealthyTimeoutMs = 20000;
1341     params.unhealthyMonitoringMs = 30000;
1342 
1343     using MS = std::chrono::milliseconds;
1344     using MCS = std::chrono::microseconds;
1345 
1346     const auto blockedTimeout = MS(params.blockedTimeoutMs);
1347     const auto unhealthyTimeout = MS(params.unhealthyTimeoutMs);
1348     const auto unhealthyMonitoring = MS(params.unhealthyMonitoringMs);
1349 
1350     const uint64_t kFirstTimestampUs = 1000000000ll;
1351     const uint64_t kBlockedTimestampUs =
1352             kFirstTimestampUs - std::chrono::duration_cast<MCS>(blockedTimeout).count();
1353     const uint64_t kUnhealthyTimestampUs =
1354             kFirstTimestampUs - std::chrono::duration_cast<MCS>(unhealthyTimeout).count();
1355 
1356     TemporaryDir tempDir;
1357     int storageId =
1358             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1359                                                IncrementalService::CreateOptions::CreateNew);
1360     ASSERT_GE(storageId, 0);
1361     mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {},
1362                                       std::move(params), listener, {});
1363 
1364     // Healthy state, registered for pending reads.
1365     ASSERT_NE(nullptr, mLooper->mCallback);
1366     ASSERT_NE(nullptr, mLooper->mCallbackData);
1367     ASSERT_EQ(storageId, listener->mStorageId);
1368     ASSERT_EQ(IStorageHealthListener::HEALTH_STATUS_OK, listener->mStatus);
1369     checkHealthMetrics(storageId, 0, listener->mStatus);
1370 
1371     // Looper/epoll callback.
1372     mIncFs->waitForPendingReadsSuccess(kFirstTimestampUs);
1373     mLooper->mCallback(-1, -1, mLooper->mCallbackData);
1374 
1375     // Unregister from pending reads and wait.
1376     ASSERT_EQ(nullptr, mLooper->mCallback);
1377     ASSERT_EQ(nullptr, mLooper->mCallbackData);
1378     ASSERT_EQ(storageId, listener->mStorageId);
1379     ASSERT_EQ(IStorageHealthListener::HEALTH_STATUS_READS_PENDING, listener->mStatus);
1380     // Timed callback present.
1381     ASSERT_EQ(storageId, mTimedQueue->mId);
1382     ASSERT_GE(mTimedQueue->mAfter, blockedTimeout);
1383     auto timedCallback = mTimedQueue->mWhat;
1384     mTimedQueue->clearJob(storageId);
1385 
1386     // Timed job callback for blocked.
1387     mIncFs->waitForPendingReadsSuccess(kBlockedTimestampUs);
1388     timedCallback();
1389 
1390     // Still not registered, and blocked.
1391     ASSERT_EQ(nullptr, mLooper->mCallback);
1392     ASSERT_EQ(nullptr, mLooper->mCallbackData);
1393     ASSERT_EQ(storageId, listener->mStorageId);
1394     ASSERT_EQ(IStorageHealthListener::HEALTH_STATUS_BLOCKED, listener->mStatus);
1395     checkHealthMetrics(storageId, params.blockedTimeoutMs, listener->mStatus);
1396 
1397     // Timed callback present.
1398     ASSERT_EQ(storageId, mTimedQueue->mId);
1399     ASSERT_GE(mTimedQueue->mAfter, 1000ms);
1400     timedCallback = mTimedQueue->mWhat;
1401     mTimedQueue->clearJob(storageId);
1402 
1403     // Timed job callback for unhealthy.
1404     mIncFs->waitForPendingReadsSuccess(kUnhealthyTimestampUs);
1405     timedCallback();
1406 
1407     // Still not registered, and blocked.
1408     ASSERT_EQ(nullptr, mLooper->mCallback);
1409     ASSERT_EQ(nullptr, mLooper->mCallbackData);
1410     ASSERT_EQ(storageId, listener->mStorageId);
1411     ASSERT_EQ(IStorageHealthListener::HEALTH_STATUS_UNHEALTHY, listener->mStatus);
1412     checkHealthMetrics(storageId, params.unhealthyTimeoutMs, listener->mStatus);
1413 
1414     // Timed callback present.
1415     ASSERT_EQ(storageId, mTimedQueue->mId);
1416     ASSERT_GE(mTimedQueue->mAfter, unhealthyMonitoring);
1417     timedCallback = mTimedQueue->mWhat;
1418     mTimedQueue->clearJob(storageId);
1419 
1420     // One more unhealthy.
1421     mIncFs->waitForPendingReadsSuccess(kUnhealthyTimestampUs);
1422     timedCallback();
1423 
1424     // Still not registered, and blocked.
1425     ASSERT_EQ(nullptr, mLooper->mCallback);
1426     ASSERT_EQ(nullptr, mLooper->mCallbackData);
1427     ASSERT_EQ(storageId, listener->mStorageId);
1428     ASSERT_EQ(IStorageHealthListener::HEALTH_STATUS_UNHEALTHY, listener->mStatus);
1429     checkHealthMetrics(storageId, params.unhealthyTimeoutMs, listener->mStatus);
1430 
1431     // Timed callback present.
1432     ASSERT_EQ(storageId, mTimedQueue->mId);
1433     ASSERT_GE(mTimedQueue->mAfter, unhealthyMonitoring);
1434     timedCallback = mTimedQueue->mWhat;
1435     mTimedQueue->clearJob(storageId);
1436 
1437     // And now healthy.
1438     mIncFs->waitForPendingReadsTimeout();
1439     timedCallback();
1440 
1441     // Healthy state, registered for pending reads.
1442     ASSERT_NE(nullptr, mLooper->mCallback);
1443     ASSERT_NE(nullptr, mLooper->mCallbackData);
1444     ASSERT_EQ(storageId, listener->mStorageId);
1445     ASSERT_EQ(IStorageHealthListener::HEALTH_STATUS_OK, listener->mStatus);
1446     checkHealthMetrics(storageId, 0, listener->mStatus);
1447 }
1448 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsSuccess)1449 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsSuccess) {
1450     mVold->setIncFsMountOptionsSuccess();
1451     mAppOpsManager->checkPermissionSuccess();
1452 
1453     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1454     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1455     // on startLoading
1456     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(1);
1457     // We are calling setIncFsMountOptions(true).
1458     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(1);
1459     // After setIncFsMountOptions succeeded expecting to start watching.
1460     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(1);
1461     // Not expecting callback removal.
1462     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1463     TemporaryDir tempDir;
1464     int storageId =
1465             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1466                                                IncrementalService::CreateOptions::CreateNew);
1467     ASSERT_GE(storageId, 0);
1468     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1469                                                   {}, {}));
1470     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1471 }
1472 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsSuccessAndDisabled)1473 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsSuccessAndDisabled) {
1474     mVold->setIncFsMountOptionsSuccess();
1475     mAppOpsManager->checkPermissionSuccess();
1476 
1477     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1478     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1479     // Enabling and then disabling readlogs.
1480     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(1);
1481     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(2);
1482     // After setIncFsMountOptions succeeded expecting to start watching.
1483     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(1);
1484     // Not expecting callback removal.
1485     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1486     TemporaryDir tempDir;
1487     int storageId =
1488             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1489                                                IncrementalService::CreateOptions::CreateNew);
1490     ASSERT_GE(storageId, 0);
1491     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1492                                                   {}, {}));
1493     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1494     // Now disable.
1495     mIncrementalService->disallowReadLogs(storageId);
1496     ASSERT_EQ(mDataLoader->setStorageParams(true), -EPERM);
1497 }
1498 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsSuccessAndTimedOut)1499 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsSuccessAndTimedOut) {
1500     mVold->setIncFsMountOptionsSuccess();
1501     mAppOpsManager->checkPermissionSuccess();
1502 
1503     const auto readLogsMaxInterval = 2h;
1504 
1505     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1506     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1507     // Enabling and then disabling readlogs.
1508     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(2);
1509     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(2);
1510     // After setIncFsMountOptions succeeded expecting to start watching.
1511     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(1);
1512     // Not expecting callback removal.
1513     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1514     EXPECT_CALL(*mTimedQueue, addJob(_, _, _)).Times(2);
1515     TemporaryDir tempDir;
1516     int storageId =
1517             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1518                                                IncrementalService::CreateOptions::CreateNew);
1519     ASSERT_GE(storageId, 0);
1520     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1521                                                   {}, {}));
1522 
1523     // Disable readlogs callback present.
1524     ASSERT_EQ(storageId, mTimedQueue->mId);
1525     ASSERT_EQ(mTimedQueue->mAfter, readLogsMaxInterval);
1526     auto callback = mTimedQueue->mWhat;
1527     mTimedQueue->clearJob(storageId);
1528 
1529     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1530     // Now advance clock for 1hr.
1531     mClock->advance(1h);
1532     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1533     // Now call the timed callback, it should turn off the readlogs.
1534     callback();
1535     // Now advance clock for 2hrs.
1536     mClock->advance(readLogsMaxInterval);
1537     ASSERT_EQ(mDataLoader->setStorageParams(true), -EPERM);
1538 }
1539 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsSuccessAndNoTimedOutForSystem)1540 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsSuccessAndNoTimedOutForSystem) {
1541     mVold->setIncFsMountOptionsSuccess();
1542     mAppOpsManager->checkPermissionSuccess();
1543 
1544     const auto readLogsMaxInterval = 2h;
1545 
1546     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1547     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1548     // Enabling and then disabling readlogs.
1549     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(3);
1550     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(1);
1551     // After setIncFsMountOptions succeeded expecting to start watching.
1552     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(1);
1553     // Not expecting callback removal.
1554     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1555     EXPECT_CALL(*mTimedQueue, addJob(_, _, _)).Times(2);
1556     // System data loader.
1557     mDataLoaderParcel.packageName = "android";
1558     TemporaryDir tempDir;
1559     int storageId =
1560             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1561                                                IncrementalService::CreateOptions::CreateNew);
1562     ASSERT_GE(storageId, 0);
1563     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1564                                                   {}, {}));
1565 
1566     // IfsState callback.
1567     auto callback = mTimedQueue->mWhat;
1568     mTimedQueue->clearJob(storageId);
1569 
1570     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1571     // Now advance clock for 1hr.
1572     mClock->advance(1h);
1573     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1574     // Now advance clock for 2hrs.
1575     mClock->advance(readLogsMaxInterval);
1576     // IfsStorage callback should not affect anything.
1577     callback();
1578     ASSERT_EQ(mDataLoader->setStorageParams(true), 0);
1579 }
1580 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsSuccessAndNewInstall)1581 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsSuccessAndNewInstall) {
1582     mVold->setIncFsMountOptionsSuccess();
1583     mAppOpsManager->checkPermissionSuccess();
1584 
1585     const auto readLogsMaxInterval = 2h;
1586 
1587     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(2);
1588     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1589     // Enabling and then disabling readlogs.
1590     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(5);
1591     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(3);
1592     // After setIncFsMountOptions succeeded expecting to start watching.
1593     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(1);
1594     // Not expecting callback removal.
1595     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1596     EXPECT_CALL(*mTimedQueue, addJob(_, _, _)).Times(4);
1597     TemporaryDir tempDir;
1598     int storageId =
1599             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1600                                                IncrementalService::CreateOptions::CreateNew);
1601     ASSERT_GE(storageId, 0);
1602 
1603     // Before install - long timeouts.
1604     ASSERT_TRUE(mVold->readTimeoutsEnabled());
1605 
1606     auto dataLoaderParcel = mDataLoaderParcel;
1607     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(dataLoaderParcel), {}, {},
1608                                                   {}, {}));
1609     // During install - short timeouts.
1610     ASSERT_FALSE(mVold->readTimeoutsEnabled());
1611 
1612     // Disable readlogs callback present.
1613     ASSERT_EQ(storageId, mTimedQueue->mId);
1614     ASSERT_EQ(mTimedQueue->mAfter, readLogsMaxInterval);
1615     auto callback = mTimedQueue->mWhat;
1616     mTimedQueue->clearJob(storageId);
1617 
1618     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1619     // Now advance clock for 1.5hrs.
1620     mClock->advance(90min);
1621     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1622 
1623     mIncrementalService->onInstallationComplete(storageId);
1624     // After install - long timeouts.
1625     ASSERT_TRUE(mVold->readTimeoutsEnabled());
1626 
1627     // New installation.
1628     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1629                                                   {}, {}));
1630     // New installation - short timeouts.
1631     ASSERT_FALSE(mVold->readTimeoutsEnabled());
1632 
1633     // New callback present.
1634     ASSERT_EQ(storageId, mTimedQueue->mId);
1635     ASSERT_EQ(mTimedQueue->mAfter, readLogsMaxInterval);
1636     auto callback2 = mTimedQueue->mWhat;
1637     mTimedQueue->clearJob(storageId);
1638 
1639     // Old callback should not disable readlogs (setIncFsMountOptions should be called only once).
1640     callback();
1641     // Advance clock for another 1.5hrs.
1642     mClock->advance(90min);
1643     // Still success even it's 3hrs past first install.
1644     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1645 
1646     // New one should disable.
1647     callback2();
1648     // And timeout.
1649     mClock->advance(90min);
1650     ASSERT_EQ(mDataLoader->setStorageParams(true), -EPERM);
1651 
1652     mIncrementalService->onInstallationComplete(storageId);
1653     // After install - long timeouts.
1654     ASSERT_TRUE(mVold->readTimeoutsEnabled());
1655 }
1656 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsSuccessAndPermissionChanged)1657 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsSuccessAndPermissionChanged) {
1658     mVold->setIncFsMountOptionsSuccess();
1659     mAppOpsManager->checkPermissionSuccess();
1660     mAppOpsManager->initializeStartWatchingMode();
1661 
1662     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1663     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1664     // We are calling setIncFsMountOptions(true).
1665     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(1);
1666     // setIncFsMountOptions(false) is called on the callback.
1667     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(2);
1668     // After setIncFsMountOptions succeeded expecting to start watching.
1669     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(1);
1670     // After callback is called, disable read logs and remove callback.
1671     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(1);
1672     TemporaryDir tempDir;
1673     int storageId =
1674             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1675                                                IncrementalService::CreateOptions::CreateNew);
1676     ASSERT_GE(storageId, 0);
1677     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1678                                                   {}, {}));
1679     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
1680     ASSERT_NE(nullptr, mAppOpsManager->mStoredCallback.get());
1681     mAppOpsManager->mStoredCallback->opChanged(0, {});
1682 }
1683 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsCheckPermissionFails)1684 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsCheckPermissionFails) {
1685     mAppOpsManager->checkPermissionFails();
1686 
1687     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1688     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1689     // checkPermission fails, no calls to set opitions,  start or stop WatchingMode.
1690     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(0);
1691     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(1);
1692     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(0);
1693     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1694     TemporaryDir tempDir;
1695     int storageId =
1696             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1697                                                IncrementalService::CreateOptions::CreateNew);
1698     ASSERT_GE(storageId, 0);
1699     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1700                                                   {}, {}));
1701     ASSERT_LT(mDataLoader->setStorageParams(true), 0);
1702 }
1703 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsCheckPermissionNoCrossUsers)1704 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsCheckPermissionNoCrossUsers) {
1705     mAppOpsManager->checkPermissionNoCrossUsers();
1706 
1707     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1708     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1709     // checkPermission fails, no calls to set opitions,  start or stop WatchingMode.
1710     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(0);
1711     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(1);
1712     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(0);
1713     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1714     TemporaryDir tempDir;
1715     int storageId =
1716             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1717                                                IncrementalService::CreateOptions::CreateNew);
1718     ASSERT_GE(storageId, 0);
1719     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1720                                                   {}, {}));
1721     ASSERT_LT(mDataLoader->setStorageParams(true), 0);
1722 }
1723 
TEST_F(IncrementalServiceTest,testSetIncFsMountOptionsFails)1724 TEST_F(IncrementalServiceTest, testSetIncFsMountOptionsFails) {
1725     mVold->setIncFsMountOptionsFails();
1726     mAppOpsManager->checkPermissionSuccess();
1727 
1728     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_));
1729     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1730     // We are calling setIncFsMountOptions.
1731     EXPECT_CALL(*mVold, setIncFsMountOptions(_, true, _, _)).Times(1);
1732     EXPECT_CALL(*mVold, setIncFsMountOptions(_, false, _, _)).Times(1);
1733     // setIncFsMountOptions fails, no calls to start or stop WatchingMode.
1734     EXPECT_CALL(*mAppOpsManager, startWatchingMode(_, _, _)).Times(0);
1735     EXPECT_CALL(*mAppOpsManager, stopWatchingMode(_)).Times(0);
1736     TemporaryDir tempDir;
1737     int storageId =
1738             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1739                                                IncrementalService::CreateOptions::CreateNew);
1740     ASSERT_GE(storageId, 0);
1741     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1742                                                   {}, {}));
1743     ASSERT_LT(mDataLoader->setStorageParams(true), 0);
1744 }
1745 
TEST_F(IncrementalServiceTest,testMakeDirectory)1746 TEST_F(IncrementalServiceTest, testMakeDirectory) {
1747     TemporaryDir tempDir;
1748     int storageId =
1749             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1750                                                IncrementalService::CreateOptions::CreateNew);
1751     std::string dir_path("test");
1752 
1753     // Expecting incfs to call makeDir on a path like:
1754     // <root>/*/mount/<storage>/test
1755     EXPECT_CALL(*mIncFs,
1756                 makeDir(_, Truly([&](std::string_view arg) {
1757                             return arg.starts_with(mRootDir.path) &&
1758                                     arg.ends_with("/mount/st_1_0/" + dir_path);
1759                         }),
1760                         _));
1761     auto res = mIncrementalService->makeDir(storageId, dir_path, 0555);
1762     ASSERT_EQ(res, 0);
1763 }
1764 
TEST_F(IncrementalServiceTest,testMakeDirectories)1765 TEST_F(IncrementalServiceTest, testMakeDirectories) {
1766     TemporaryDir tempDir;
1767     int storageId =
1768             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1769                                                IncrementalService::CreateOptions::CreateNew);
1770     auto first = "first"sv;
1771     auto second = "second"sv;
1772     auto third = "third"sv;
1773     auto dir_path = std::string(first) + "/" + std::string(second) + "/" + std::string(third);
1774 
1775     EXPECT_CALL(*mIncFs,
1776                 makeDirs(_, Truly([&](std::string_view arg) {
1777                              return arg.starts_with(mRootDir.path) &&
1778                                      arg.ends_with("/mount/st_1_0/" + dir_path);
1779                          }),
1780                          _));
1781     auto res = mIncrementalService->makeDirs(storageId, dir_path, 0555);
1782     ASSERT_EQ(res, 0);
1783 }
1784 
TEST_F(IncrementalServiceTest,testIsFileFullyLoadedNoData)1785 TEST_F(IncrementalServiceTest, testIsFileFullyLoadedNoData) {
1786     TemporaryDir tempDir;
1787     int storageId =
1788             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1789                                                IncrementalService::CreateOptions::CreateNew);
1790     EXPECT_CALL(*mIncFs, isFileFullyLoaded(_, An<std::string_view>()))
1791             .Times(1)
1792             .WillOnce(Return(incfs::LoadingState::MissingBlocks));
1793     ASSERT_GT((int)mIncrementalService->isFileFullyLoaded(storageId, "base.apk"), 0);
1794 }
1795 
TEST_F(IncrementalServiceTest,testIsFileFullyLoadedError)1796 TEST_F(IncrementalServiceTest, testIsFileFullyLoadedError) {
1797     TemporaryDir tempDir;
1798     int storageId =
1799             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1800                                                IncrementalService::CreateOptions::CreateNew);
1801     EXPECT_CALL(*mIncFs, isFileFullyLoaded(_, An<std::string_view>()))
1802             .Times(1)
1803             .WillOnce(Return(incfs::LoadingState(-1)));
1804     ASSERT_LT((int)mIncrementalService->isFileFullyLoaded(storageId, "base.apk"), 0);
1805 }
1806 
TEST_F(IncrementalServiceTest,testIsFileFullyLoadedSuccess)1807 TEST_F(IncrementalServiceTest, testIsFileFullyLoadedSuccess) {
1808     TemporaryDir tempDir;
1809     int storageId =
1810             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1811                                                IncrementalService::CreateOptions::CreateNew);
1812     EXPECT_CALL(*mIncFs, isFileFullyLoaded(_, An<std::string_view>()))
1813             .Times(1)
1814             .WillOnce(Return(incfs::LoadingState::Full));
1815     ASSERT_EQ(0, (int)mIncrementalService->isFileFullyLoaded(storageId, "base.apk"));
1816 }
1817 
TEST_F(IncrementalServiceTest,testGetLoadingProgressSuccessWithNoFile)1818 TEST_F(IncrementalServiceTest, testGetLoadingProgressSuccessWithNoFile) {
1819     mIncFs->countFilledBlocksSuccess();
1820     mFs->hasNoFile();
1821 
1822     TemporaryDir tempDir;
1823     int storageId =
1824             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1825                                                IncrementalService::CreateOptions::CreateNew);
1826     ASSERT_EQ(1, mIncrementalService->getLoadingProgress(storageId).getProgress());
1827 }
1828 
TEST_F(IncrementalServiceTest,testGetLoadingProgressFailsWithFailedRanges)1829 TEST_F(IncrementalServiceTest, testGetLoadingProgressFailsWithFailedRanges) {
1830     mIncFs->countFilledBlocksFails();
1831     mFs->hasFiles();
1832 
1833     TemporaryDir tempDir;
1834     int storageId =
1835             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1836                                                IncrementalService::CreateOptions::CreateNew);
1837     EXPECT_CALL(*mIncFs, countFilledBlocks(_, _)).Times(1);
1838     ASSERT_EQ(-1, mIncrementalService->getLoadingProgress(storageId).getProgress());
1839 }
1840 
TEST_F(IncrementalServiceTest,testGetLoadingProgressSuccessWithEmptyRanges)1841 TEST_F(IncrementalServiceTest, testGetLoadingProgressSuccessWithEmptyRanges) {
1842     mIncFs->countFilledBlocksEmpty();
1843     mFs->hasFiles();
1844 
1845     TemporaryDir tempDir;
1846     int storageId =
1847             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1848                                                IncrementalService::CreateOptions::CreateNew);
1849     EXPECT_CALL(*mIncFs, countFilledBlocks(_, _)).Times(3);
1850     ASSERT_EQ(1, mIncrementalService->getLoadingProgress(storageId).getProgress());
1851 }
1852 
TEST_F(IncrementalServiceTest,testGetLoadingProgressSuccess)1853 TEST_F(IncrementalServiceTest, testGetLoadingProgressSuccess) {
1854     mIncFs->countFilledBlocksSuccess();
1855     mFs->hasFiles();
1856 
1857     TemporaryDir tempDir;
1858     int storageId =
1859             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1860                                                IncrementalService::CreateOptions::CreateNew);
1861     EXPECT_CALL(*mIncFs, countFilledBlocks(_, _)).Times(3);
1862     ASSERT_EQ(0.5, mIncrementalService->getLoadingProgress(storageId).getProgress());
1863 }
1864 
TEST_F(IncrementalServiceTest,testRegisterLoadingProgressListenerSuccess)1865 TEST_F(IncrementalServiceTest, testRegisterLoadingProgressListenerSuccess) {
1866     mIncFs->countFilledBlocksSuccess();
1867     mFs->hasFiles();
1868 
1869     TemporaryDir tempDir;
1870     int storageId =
1871             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1872                                                IncrementalService::CreateOptions::CreateNew);
1873     sp<NiceMock<MockStorageLoadingProgressListener>> listener{
1874             new NiceMock<MockStorageLoadingProgressListener>};
1875     NiceMock<MockStorageLoadingProgressListener>* listenerMock = listener.get();
1876     EXPECT_CALL(*listenerMock, onStorageLoadingProgressChanged(_, _)).Times(2);
1877     EXPECT_CALL(*mProgressUpdateJobQueue, addJob(_, _, _)).Times(2);
1878     mIncrementalService->registerLoadingProgressListener(storageId, listener);
1879     // Timed callback present.
1880     ASSERT_EQ(storageId, mProgressUpdateJobQueue->mId);
1881     ASSERT_EQ(mProgressUpdateJobQueue->mAfter, 1000ms);
1882     auto timedCallback = mProgressUpdateJobQueue->mWhat;
1883     timedCallback();
1884     ASSERT_EQ(storageId, mProgressUpdateJobQueue->mId);
1885     ASSERT_EQ(mProgressUpdateJobQueue->mAfter, 1000ms);
1886     mIncrementalService->unregisterLoadingProgressListener(storageId);
1887     ASSERT_EQ(mProgressUpdateJobQueue->mAfter, Milliseconds{});
1888 }
1889 
TEST_F(IncrementalServiceTest,testRegisterLoadingProgressListenerFailsToGetProgress)1890 TEST_F(IncrementalServiceTest, testRegisterLoadingProgressListenerFailsToGetProgress) {
1891     mIncFs->countFilledBlocksFails();
1892     mFs->hasFiles();
1893 
1894     TemporaryDir tempDir;
1895     int storageId =
1896             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1897                                                IncrementalService::CreateOptions::CreateNew);
1898     sp<NiceMock<MockStorageLoadingProgressListener>> listener{
1899             new NiceMock<MockStorageLoadingProgressListener>};
1900     NiceMock<MockStorageLoadingProgressListener>* listenerMock = listener.get();
1901     EXPECT_CALL(*listenerMock, onStorageLoadingProgressChanged(_, _)).Times(0);
1902     mIncrementalService->registerLoadingProgressListener(storageId, listener);
1903 }
1904 
TEST_F(IncrementalServiceTest,testStartDataLoaderUnbindOnAllDone)1905 TEST_F(IncrementalServiceTest, testStartDataLoaderUnbindOnAllDone) {
1906     mFs->hasFiles();
1907 
1908     const auto stateUpdateInterval = 1s;
1909 
1910     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
1911     // No unbinding just yet.
1912     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(0);
1913     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
1914     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
1915     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1916     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1917     // System data loader to get rid of readlog timeout callback.
1918     mDataLoaderParcel.packageName = "android";
1919     TemporaryDir tempDir;
1920     int storageId =
1921             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1922                                                IncrementalService::CreateOptions::CreateNew);
1923     ASSERT_GE(storageId, 0);
1924     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1925                                                   {}, {}));
1926 
1927     // Started.
1928     ASSERT_EQ(mDataLoader->status(), IDataLoaderStatusListener::DATA_LOADER_STARTED);
1929 
1930     // IfsState callback present.
1931     ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
1932     ASSERT_EQ(mTimedQueue->mAfter, stateUpdateInterval);
1933     auto callback = mTimedQueue->mWhat;
1934     mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
1935 
1936     // Not loaded yet.
1937     EXPECT_CALL(*mIncFs, isEverythingFullyLoaded(_))
1938             .WillOnce(Return(incfs::LoadingState::MissingBlocks));
1939 
1940     // Send the callback, should not do anything.
1941     callback();
1942 
1943     // Still started.
1944     ASSERT_EQ(mDataLoader->status(), IDataLoaderStatusListener::DATA_LOADER_STARTED);
1945 
1946     // Still present.
1947     ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
1948     ASSERT_EQ(mTimedQueue->mAfter, stateUpdateInterval);
1949     callback = mTimedQueue->mWhat;
1950     mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
1951 
1952     // Fully loaded.
1953     EXPECT_CALL(*mIncFs, isEverythingFullyLoaded(_)).WillOnce(Return(incfs::LoadingState::Full));
1954     // Expect the unbind.
1955     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
1956 
1957     callback();
1958 
1959     // Destroyed.
1960     ASSERT_EQ(mDataLoader->status(), IDataLoaderStatusListener::DATA_LOADER_DESTROYED);
1961 }
1962 
TEST_F(IncrementalServiceTest,testStartDataLoaderUnbindOnAllDoneWithReadlogs)1963 TEST_F(IncrementalServiceTest, testStartDataLoaderUnbindOnAllDoneWithReadlogs) {
1964     mFs->hasFiles();
1965 
1966     // Readlogs.
1967     mVold->setIncFsMountOptionsSuccess();
1968     mAppOpsManager->checkPermissionSuccess();
1969 
1970     const auto stateUpdateInterval = 1s;
1971 
1972     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
1973     // No unbinding just yet.
1974     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(0);
1975     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
1976     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
1977     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
1978     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
1979     // System data loader to get rid of readlog timeout callback.
1980     mDataLoaderParcel.packageName = "android";
1981     TemporaryDir tempDir;
1982     int storageId =
1983             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
1984                                                IncrementalService::CreateOptions::CreateNew);
1985     ASSERT_GE(storageId, 0);
1986     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
1987                                                   {}, {}));
1988 
1989     // Started.
1990     ASSERT_EQ(mDataLoader->status(), IDataLoaderStatusListener::DATA_LOADER_STARTED);
1991 
1992     // IfsState callback present.
1993     ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
1994     ASSERT_EQ(mTimedQueue->mAfter, stateUpdateInterval);
1995     auto callback = mTimedQueue->mWhat;
1996     mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
1997 
1998     // Not loaded yet.
1999     EXPECT_CALL(*mIncFs, isEverythingFullyLoaded(_))
2000             .WillOnce(Return(incfs::LoadingState::MissingBlocks));
2001 
2002     // Send the callback, should not do anything.
2003     callback();
2004 
2005     // Still started.
2006     ASSERT_EQ(mDataLoader->status(), IDataLoaderStatusListener::DATA_LOADER_STARTED);
2007 
2008     // Still present.
2009     ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
2010     ASSERT_EQ(mTimedQueue->mAfter, stateUpdateInterval);
2011     callback = mTimedQueue->mWhat;
2012     mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
2013 
2014     // Fully loaded.
2015     EXPECT_CALL(*mIncFs, isEverythingFullyLoaded(_))
2016             .WillOnce(Return(incfs::LoadingState::Full))
2017             .WillOnce(Return(incfs::LoadingState::Full));
2018     // But with readlogs.
2019     ASSERT_GE(mDataLoader->setStorageParams(true), 0);
2020 
2021     // Send the callback, still nothing.
2022     callback();
2023 
2024     // Still started.
2025     ASSERT_EQ(mDataLoader->status(), IDataLoaderStatusListener::DATA_LOADER_STARTED);
2026 
2027     // Still present.
2028     ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
2029     ASSERT_EQ(mTimedQueue->mAfter, stateUpdateInterval);
2030     callback = mTimedQueue->mWhat;
2031     mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
2032 
2033     // Disable readlogs and expect the unbind.
2034     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
2035     ASSERT_GE(mDataLoader->setStorageParams(false), 0);
2036 
2037     callback();
2038 
2039     // Destroyed.
2040     ASSERT_EQ(mDataLoader->status(), IDataLoaderStatusListener::DATA_LOADER_DESTROYED);
2041 }
2042 
createPerUidTimeouts(std::initializer_list<std::tuple<int,int,int,int>> tuples)2043 static std::vector<PerUidReadTimeouts> createPerUidTimeouts(
2044         std::initializer_list<std::tuple<int, int, int, int>> tuples) {
2045     std::vector<PerUidReadTimeouts> result;
2046     for (auto&& tuple : tuples) {
2047         result.emplace_back();
2048         auto& timeouts = result.back();
2049         timeouts.uid = std::get<0>(tuple);
2050         timeouts.minTimeUs = std::get<1>(tuple);
2051         timeouts.minPendingTimeUs = std::get<2>(tuple);
2052         timeouts.maxPendingTimeUs = std::get<3>(tuple);
2053     }
2054     return result;
2055 }
2056 
checkPerUidTimeouts(const Control & control,const std::vector<PerUidReadTimeouts> & perUidReadTimeouts)2057 static ErrorCode checkPerUidTimeouts(const Control& control,
2058                                      const std::vector<PerUidReadTimeouts>& perUidReadTimeouts) {
2059     std::vector<PerUidReadTimeouts> expected =
2060             createPerUidTimeouts({{0, 1, 2, 3}, {1, 2, 3, 4}, {2, 3, 4, 100000000}});
2061     EXPECT_EQ(expected, perUidReadTimeouts);
2062     return 0;
2063 }
2064 
checkPerUidTimeoutsEmpty(const Control & control,const std::vector<PerUidReadTimeouts> & perUidReadTimeouts)2065 static ErrorCode checkPerUidTimeoutsEmpty(
2066         const Control& control, const std::vector<PerUidReadTimeouts>& perUidReadTimeouts) {
2067     EXPECT_EQ(0u, perUidReadTimeouts.size());
2068     return 0;
2069 }
2070 
TEST_F(IncrementalServiceTest,testPerUidTimeoutsTooShort)2071 TEST_F(IncrementalServiceTest, testPerUidTimeoutsTooShort) {
2072     EXPECT_CALL(*mDataLoaderManager, bindToDataLoader(_, _, _, _, _)).Times(1);
2073     EXPECT_CALL(*mDataLoaderManager, unbindFromDataLoader(_)).Times(1);
2074     EXPECT_CALL(*mDataLoader, create(_, _, _, _)).Times(1);
2075     EXPECT_CALL(*mDataLoader, start(_)).Times(1);
2076     EXPECT_CALL(*mDataLoader, destroy(_)).Times(1);
2077     EXPECT_CALL(*mIncFs, setUidReadTimeouts(_, _)).Times(0);
2078     EXPECT_CALL(*mTimedQueue, addJob(_, _, _)).Times(2);
2079     EXPECT_CALL(*mVold, unmountIncFs(_)).Times(2);
2080     TemporaryDir tempDir;
2081     int storageId =
2082             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
2083                                                IncrementalService::CreateOptions::CreateNew);
2084     ASSERT_GE(storageId, 0);
2085     mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {}, {},
2086                                       createPerUidTimeouts(
2087                                               {{0, 1, 2, 3}, {1, 2, 3, 4}, {2, 3, 4, 5}}));
2088 }
2089 
TEST_F(IncrementalServiceTest,testPerUidTimeoutsSuccess)2090 TEST_F(IncrementalServiceTest, testPerUidTimeoutsSuccess) {
2091     mVold->setIncFsMountOptionsSuccess();
2092     mAppOpsManager->checkPermissionSuccess();
2093     mFs->hasFiles();
2094 
2095     EXPECT_CALL(*mIncFs, setUidReadTimeouts(_, _))
2096             // First call.
2097             .WillOnce(Invoke(&checkPerUidTimeouts))
2098             // Fully loaded and no readlogs.
2099             .WillOnce(Invoke(&checkPerUidTimeoutsEmpty));
2100     EXPECT_CALL(*mTimedQueue, addJob(_, _, _)).Times(3);
2101 
2102     // Loading storage.
2103     EXPECT_CALL(*mIncFs, isEverythingFullyLoaded(_))
2104             .WillOnce(Return(incfs::LoadingState::MissingBlocks))
2105             .WillOnce(Return(incfs::LoadingState::MissingBlocks))
2106             .WillOnce(Return(incfs::LoadingState::Full));
2107 
2108     // Mark DataLoader as 'system' so that readlogs don't pollute the timed queue.
2109     mDataLoaderParcel.packageName = "android";
2110 
2111     TemporaryDir tempDir;
2112     int storageId =
2113             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
2114                                                IncrementalService::CreateOptions::CreateNew);
2115     ASSERT_GE(storageId, 0);
2116     mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {}, {},
2117                                       createPerUidTimeouts(
2118                                               {{0, 1, 2, 3}, {1, 2, 3, 4}, {2, 3, 4, 100000000}}));
2119 
2120     {
2121         // Timed callback present -> 0 progress.
2122         ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
2123         ASSERT_GE(mTimedQueue->mAfter, std::chrono::seconds(1));
2124         const auto timedCallback = mTimedQueue->mWhat;
2125         mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
2126 
2127         // Call it again.
2128         timedCallback();
2129     }
2130 
2131     {
2132         // Still present -> some progress.
2133         ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
2134         ASSERT_GE(mTimedQueue->mAfter, std::chrono::seconds(1));
2135         const auto timedCallback = mTimedQueue->mWhat;
2136         mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
2137 
2138         // Fully loaded but readlogs collection enabled.
2139         ASSERT_GE(mDataLoader->setStorageParams(true), 0);
2140 
2141         // Call it again.
2142         timedCallback();
2143     }
2144 
2145     {
2146         // Still present -> fully loaded + readlogs.
2147         ASSERT_EQ(IncrementalService::kAllStoragesId, mTimedQueue->mId);
2148         ASSERT_GE(mTimedQueue->mAfter, std::chrono::seconds(1));
2149         const auto timedCallback = mTimedQueue->mWhat;
2150         mTimedQueue->clearJob(IncrementalService::kAllStoragesId);
2151 
2152         // Now disable readlogs.
2153         ASSERT_GE(mDataLoader->setStorageParams(false), 0);
2154 
2155         // Call it again.
2156         timedCallback();
2157     }
2158 
2159     // No callbacks anymore -> fully loaded and no readlogs.
2160     ASSERT_EQ(mTimedQueue->mAfter, Milliseconds());
2161 }
2162 
TEST_F(IncrementalServiceTest,testInvalidMetricsQuery)2163 TEST_F(IncrementalServiceTest, testInvalidMetricsQuery) {
2164     const auto invalidStorageId = 100;
2165     android::os::PersistableBundle result{};
2166     mIncrementalService->getMetrics(invalidStorageId, &result);
2167     int64_t expected = -1, value = -1;
2168     ASSERT_FALSE(
2169             result.getLong(String16(BnIncrementalService::METRICS_MILLIS_SINCE_OLDEST_PENDING_READ()
2170                                             .c_str()),
2171                            &value));
2172     ASSERT_EQ(expected, value);
2173     ASSERT_TRUE(result.empty());
2174 }
2175 
TEST_F(IncrementalServiceTest,testNoDataLoaderMetrics)2176 TEST_F(IncrementalServiceTest, testNoDataLoaderMetrics) {
2177     mVold->setIncFsMountOptionsSuccess();
2178     TemporaryDir tempDir;
2179     int storageId =
2180             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
2181                                                IncrementalService::CreateOptions::CreateNew);
2182     ASSERT_GE(storageId, 0);
2183     android::os::PersistableBundle result{};
2184     mIncrementalService->getMetrics(storageId, &result);
2185     int64_t expected = -1, value = -1;
2186     ASSERT_FALSE(
2187             result.getLong(String16(BnIncrementalService::METRICS_MILLIS_SINCE_OLDEST_PENDING_READ()
2188                                             .c_str()),
2189                            &value));
2190     ASSERT_EQ(expected, value);
2191     ASSERT_EQ(1, (int)result.size());
2192     bool expectedReadLogsEnabled = false;
2193     ASSERT_TRUE(
2194             result.getBoolean(String16(BnIncrementalService::METRICS_READ_LOGS_ENABLED().c_str()),
2195                               &expectedReadLogsEnabled));
2196     ASSERT_EQ(mVold->readLogsEnabled(), expectedReadLogsEnabled);
2197 }
2198 
TEST_F(IncrementalServiceTest,testInvalidMetricsKeys)2199 TEST_F(IncrementalServiceTest, testInvalidMetricsKeys) {
2200     mVold->setIncFsMountOptionsSuccess();
2201     TemporaryDir tempDir;
2202     int storageId =
2203             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
2204                                                IncrementalService::CreateOptions::CreateNew);
2205     ASSERT_GE(storageId, 0);
2206     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
2207                                                   {}, {}));
2208     android::os::PersistableBundle result{};
2209     mIncrementalService->getMetrics(storageId, &result);
2210     int64_t expected = -1, value = -1;
2211     ASSERT_FALSE(result.getLong(String16("invalid"), &value));
2212     ASSERT_EQ(expected, value);
2213     ASSERT_EQ(6, (int)result.size());
2214 }
2215 
TEST_F(IncrementalServiceTest,testMetricsWithNoLastReadError)2216 TEST_F(IncrementalServiceTest, testMetricsWithNoLastReadError) {
2217     mVold->setIncFsMountOptionsSuccess();
2218     ON_CALL(*mIncFs, getMetrics(_))
2219             .WillByDefault(Return(Metrics{
2220                     .readsDelayedMin = 10,
2221                     .readsDelayedMinUs = 5000,
2222                     .readsDelayedPending = 10,
2223                     .readsDelayedPendingUs = 5000,
2224                     .readsFailedHashVerification = 10,
2225                     .readsFailedOther = 10,
2226                     .readsFailedTimedOut = 10,
2227             }));
2228     ON_CALL(*mIncFs, getLastReadError(_)).WillByDefault(Return(LastReadError{}));
2229     TemporaryDir tempDir;
2230     int storageId =
2231             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
2232                                                IncrementalService::CreateOptions::CreateNew);
2233     ASSERT_GE(storageId, 0);
2234     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
2235                                                   {}, {}));
2236     android::os::PersistableBundle result{};
2237     mIncrementalService->getMetrics(storageId, &result);
2238     ASSERT_EQ(9, (int)result.size());
2239 
2240     int expectedtotalDelayedReads = 20, totalDelayedReads = -1;
2241     ASSERT_TRUE(result.getInt(String16(BnIncrementalService::METRICS_TOTAL_DELAYED_READS().c_str()),
2242                               &totalDelayedReads));
2243     ASSERT_EQ(expectedtotalDelayedReads, totalDelayedReads);
2244     int expectedtotalFailedReads = 30, totalFailedReads = -1;
2245     ASSERT_TRUE(result.getInt(String16(BnIncrementalService::METRICS_TOTAL_FAILED_READS().c_str()),
2246                               &totalFailedReads));
2247     ASSERT_EQ(expectedtotalFailedReads, totalFailedReads);
2248     int64_t expectedtotalDelayedReadsMillis = 10, totalDelayedReadsMillis = -1;
2249     ASSERT_TRUE(result.getLong(String16(BnIncrementalService::METRICS_TOTAL_DELAYED_READS_MILLIS()
2250                                                 .c_str()),
2251                                &totalDelayedReadsMillis));
2252     ASSERT_EQ(expectedtotalDelayedReadsMillis, totalDelayedReadsMillis);
2253 
2254     int64_t expectedMillisSinceLastReadError = -1, millisSinceLastReadError = -1;
2255     ASSERT_FALSE(
2256             result.getLong(String16(BnIncrementalService::METRICS_MILLIS_SINCE_LAST_READ_ERROR()
2257                                             .c_str()),
2258                            &millisSinceLastReadError));
2259     ASSERT_EQ(expectedMillisSinceLastReadError, millisSinceLastReadError);
2260     int expectedLastReadErrorNumber = -1, lastReadErrorNumber = -1;
2261     ASSERT_FALSE(
2262             result.getInt(String16(BnIncrementalService::METRICS_LAST_READ_ERROR_NUMBER().c_str()),
2263                           &lastReadErrorNumber));
2264     ASSERT_EQ(expectedLastReadErrorNumber, lastReadErrorNumber);
2265     int expectedLastReadUid = -1, lastReadErrorUid = -1;
2266     ASSERT_FALSE(
2267             result.getInt(String16(BnIncrementalService::METRICS_LAST_READ_ERROR_UID().c_str()),
2268                           &lastReadErrorUid));
2269     ASSERT_EQ(expectedLastReadUid, lastReadErrorUid);
2270 }
2271 
TEST_F(IncrementalServiceTest,testMetricsWithLastReadError)2272 TEST_F(IncrementalServiceTest, testMetricsWithLastReadError) {
2273     mVold->setIncFsMountOptionsSuccess();
2274     ON_CALL(*mIncFs, getMetrics(_)).WillByDefault(Return(Metrics{}));
2275     mClock->advanceMs(5);
2276     const auto now = mClock->getClock();
2277     ON_CALL(*mIncFs, getLastReadError(_))
2278             .WillByDefault(Return(LastReadError{.timestampUs = static_cast<uint64_t>(
2279                                                         duration_cast<std::chrono::microseconds>(
2280                                                                 now.time_since_epoch())
2281                                                                 .count()),
2282                                                 .errorNo = static_cast<uint32_t>(-ETIME),
2283                                                 .uid = 20000}));
2284     TemporaryDir tempDir;
2285     int storageId =
2286             mIncrementalService->createStorage(tempDir.path, mDataLoaderParcel,
2287                                                IncrementalService::CreateOptions::CreateNew);
2288     ASSERT_GE(storageId, 0);
2289     ASSERT_TRUE(mIncrementalService->startLoading(storageId, std::move(mDataLoaderParcel), {}, {},
2290                                                   {}, {}));
2291     mClock->advanceMs(10);
2292     android::os::PersistableBundle result{};
2293     mIncrementalService->getMetrics(storageId, &result);
2294     ASSERT_EQ(12, (int)result.size());
2295     int64_t expectedMillisSinceLastReadError = 10, millisSinceLastReadError = -1;
2296     ASSERT_TRUE(result.getLong(String16(BnIncrementalService::METRICS_MILLIS_SINCE_LAST_READ_ERROR()
2297                                                 .c_str()),
2298                                &millisSinceLastReadError));
2299     ASSERT_EQ(expectedMillisSinceLastReadError, millisSinceLastReadError);
2300     int expectedLastReadErrorNumber = -ETIME, lastReadErrorNumber = -1;
2301     ASSERT_TRUE(
2302             result.getInt(String16(BnIncrementalService::METRICS_LAST_READ_ERROR_NUMBER().c_str()),
2303                           &lastReadErrorNumber));
2304     ASSERT_EQ(expectedLastReadErrorNumber, lastReadErrorNumber);
2305     int expectedLastReadUid = 20000, lastReadErrorUid = -1;
2306     ASSERT_TRUE(result.getInt(String16(BnIncrementalService::METRICS_LAST_READ_ERROR_UID().c_str()),
2307                               &lastReadErrorUid));
2308     ASSERT_EQ(expectedLastReadUid, lastReadErrorUid);
2309 }
2310 
2311 } // namespace android::os::incremental
2312