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 #define LOG_TAG "incfs-dataloaderconnector"
17
18 #include <android-base/logging.h>
19 #include <fcntl.h>
20 #include <nativehelper/JNIPlatformHelp.h>
21 #include <nativehelper/scoped_local_ref.h>
22 #include <nativehelper/scoped_utf_chars.h>
23 #include <sys/stat.h>
24 #include <utils/Looper.h>
25
26 #include <thread>
27 #include <unordered_map>
28
29 #include "JNIHelpers.h"
30 #include "ManagedDataLoader.h"
31 #include "dataloader.h"
32 #include "incfs.h"
33 #include "path.h"
34
35 namespace {
36
37 using namespace std::literals;
38
39 using ReadInfo = android::dataloader::ReadInfo;
40 using ReadInfoWithUid = android::dataloader::ReadInfoWithUid;
41
42 using FileId = android::incfs::FileId;
43 using RawMetadata = android::incfs::RawMetadata;
44 using UniqueControl = android::incfs::UniqueControl;
45
46 struct JniIds {
47 struct {
48 jint DATA_LOADER_CREATED;
49 jint DATA_LOADER_DESTROYED;
50 jint DATA_LOADER_STARTED;
51 jint DATA_LOADER_STOPPED;
52 jint DATA_LOADER_IMAGE_READY;
53 jint DATA_LOADER_IMAGE_NOT_READY;
54 jint DATA_LOADER_UNAVAILABLE;
55 jint DATA_LOADER_UNRECOVERABLE;
56
57 jint DATA_LOADER_TYPE_NONE;
58 jint DATA_LOADER_TYPE_STREAMING;
59 jint DATA_LOADER_TYPE_INCREMENTAL;
60
61 jint DATA_LOADER_LOCATION_DATA_APP;
62 jint DATA_LOADER_LOCATION_MEDIA_OBB;
63 jint DATA_LOADER_LOCATION_MEDIA_DATA;
64 } constants;
65
66 jmethodID parcelFileDescriptorGetFileDescriptor;
67
68 jfieldID incremental;
69 jfieldID service;
70 jfieldID callback;
71
72 jfieldID controlCmd;
73 jfieldID controlPendingReads;
74 jfieldID controlLog;
75 jfieldID controlBlocksWritten;
76
77 jfieldID paramsType;
78 jfieldID paramsPackageName;
79 jfieldID paramsClassName;
80 jfieldID paramsArguments;
81
82 jclass listener;
83 jmethodID listenerOnStatusChanged;
84
85 jmethodID callbackControlWriteData;
86
87 jmethodID listGet;
88 jmethodID listSize;
89
90 jfieldID installationFileLocation;
91 jfieldID installationFileName;
92 jfieldID installationFileLengthBytes;
93 jfieldID installationFileMetadata;
94
95 jmethodID incrementalServiceConnectorSetStorageParams;
96
JniIds__anone91230200111::JniIds97 JniIds(JNIEnv* env) {
98 listener = (jclass)env->NewGlobalRef(
99 FindClassOrDie(env, "android/content/pm/IDataLoaderStatusListener"));
100 listenerOnStatusChanged = GetMethodIDOrDie(env, listener, "onStatusChanged", "(II)V");
101
102 constants.DATA_LOADER_CREATED =
103 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_CREATED");
104 constants.DATA_LOADER_DESTROYED =
105 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_DESTROYED");
106 constants.DATA_LOADER_STARTED =
107 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_STARTED");
108 constants.DATA_LOADER_STOPPED =
109 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_STOPPED");
110 constants.DATA_LOADER_IMAGE_READY =
111 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_IMAGE_READY");
112 constants.DATA_LOADER_IMAGE_NOT_READY =
113 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_IMAGE_NOT_READY");
114
115 constants.DATA_LOADER_UNAVAILABLE =
116 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_UNAVAILABLE");
117 constants.DATA_LOADER_UNRECOVERABLE =
118 GetStaticIntFieldValueOrDie(env, listener, "DATA_LOADER_UNRECOVERABLE");
119
120 auto packageInstaller = (jclass)FindClassOrDie(env, "android/content/pm/PackageInstaller");
121
122 constants.DATA_LOADER_TYPE_NONE =
123 GetStaticIntFieldValueOrDie(env, packageInstaller, "DATA_LOADER_TYPE_NONE");
124 constants.DATA_LOADER_TYPE_STREAMING =
125 GetStaticIntFieldValueOrDie(env, packageInstaller, "DATA_LOADER_TYPE_STREAMING");
126 constants.DATA_LOADER_TYPE_INCREMENTAL =
127 GetStaticIntFieldValueOrDie(env, packageInstaller, "DATA_LOADER_TYPE_INCREMENTAL");
128
129 CHECK(constants.DATA_LOADER_TYPE_NONE == DATA_LOADER_TYPE_NONE);
130 CHECK(constants.DATA_LOADER_TYPE_STREAMING == DATA_LOADER_TYPE_STREAMING);
131 CHECK(constants.DATA_LOADER_TYPE_INCREMENTAL == DATA_LOADER_TYPE_INCREMENTAL);
132
133 constants.DATA_LOADER_LOCATION_DATA_APP =
134 GetStaticIntFieldValueOrDie(env, packageInstaller, "LOCATION_DATA_APP");
135 constants.DATA_LOADER_LOCATION_MEDIA_OBB =
136 GetStaticIntFieldValueOrDie(env, packageInstaller, "LOCATION_MEDIA_OBB");
137 constants.DATA_LOADER_LOCATION_MEDIA_DATA =
138 GetStaticIntFieldValueOrDie(env, packageInstaller, "LOCATION_MEDIA_DATA");
139
140 CHECK(constants.DATA_LOADER_LOCATION_DATA_APP == DATA_LOADER_LOCATION_DATA_APP);
141 CHECK(constants.DATA_LOADER_LOCATION_MEDIA_OBB == DATA_LOADER_LOCATION_MEDIA_OBB);
142 CHECK(constants.DATA_LOADER_LOCATION_MEDIA_DATA == DATA_LOADER_LOCATION_MEDIA_DATA);
143
144 auto parcelFileDescriptor = FindClassOrDie(env, "android/os/ParcelFileDescriptor");
145 parcelFileDescriptorGetFileDescriptor =
146 GetMethodIDOrDie(env, parcelFileDescriptor, "getFileDescriptor",
147 "()Ljava/io/FileDescriptor;");
148
149 auto control = FindClassOrDie(env, "android/content/pm/FileSystemControlParcel");
150 incremental =
151 GetFieldIDOrDie(env, control, "incremental",
152 "Landroid/os/incremental/IncrementalFileSystemControlParcel;");
153 service = GetFieldIDOrDie(env, control, "service",
154 "Landroid/os/incremental/IIncrementalServiceConnector;");
155 callback =
156 GetFieldIDOrDie(env, control, "callback",
157 "Landroid/content/pm/IPackageInstallerSessionFileSystemConnector;");
158
159 auto incControl =
160 FindClassOrDie(env, "android/os/incremental/IncrementalFileSystemControlParcel");
161 controlCmd = GetFieldIDOrDie(env, incControl, "cmd", "Landroid/os/ParcelFileDescriptor;");
162 controlPendingReads = GetFieldIDOrDie(env, incControl, "pendingReads",
163 "Landroid/os/ParcelFileDescriptor;");
164 controlLog = GetFieldIDOrDie(env, incControl, "log", "Landroid/os/ParcelFileDescriptor;");
165 controlBlocksWritten = GetFieldIDOrDie(env, incControl, "blocksWritten",
166 "Landroid/os/ParcelFileDescriptor;");
167
168 auto params = FindClassOrDie(env, "android/content/pm/DataLoaderParamsParcel");
169 paramsType = GetFieldIDOrDie(env, params, "type", "I");
170 paramsPackageName = GetFieldIDOrDie(env, params, "packageName", "Ljava/lang/String;");
171 paramsClassName = GetFieldIDOrDie(env, params, "className", "Ljava/lang/String;");
172 paramsArguments = GetFieldIDOrDie(env, params, "arguments", "Ljava/lang/String;");
173
174 auto callbackControl =
175 FindClassOrDie(env,
176 "android/content/pm/IPackageInstallerSessionFileSystemConnector");
177 callbackControlWriteData =
178 GetMethodIDOrDie(env, callbackControl, "writeData",
179 "(Ljava/lang/String;JJLandroid/os/ParcelFileDescriptor;)V");
180
181 auto list = (jclass)FindClassOrDie(env, "java/util/List");
182 listGet = GetMethodIDOrDie(env, list, "get", "(I)Ljava/lang/Object;");
183 listSize = GetMethodIDOrDie(env, list, "size", "()I");
184
185 auto installationFileParcel =
186 (jclass)FindClassOrDie(env, "android/content/pm/InstallationFileParcel");
187 installationFileLocation = GetFieldIDOrDie(env, installationFileParcel, "location", "I");
188 installationFileName =
189 GetFieldIDOrDie(env, installationFileParcel, "name", "Ljava/lang/String;");
190 installationFileLengthBytes = GetFieldIDOrDie(env, installationFileParcel, "size", "J");
191 installationFileMetadata = GetFieldIDOrDie(env, installationFileParcel, "metadata", "[B");
192
193 auto incrementalServiceConnector =
194 FindClassOrDie(env, "android/os/incremental/IIncrementalServiceConnector");
195 incrementalServiceConnectorSetStorageParams =
196 GetMethodIDOrDie(env, incrementalServiceConnector, "setStorageParams", "(Z)I");
197 }
198 };
199
getJavaVM(JNIEnv * env)200 JavaVM* getJavaVM(JNIEnv* env) {
201 CHECK(env);
202 JavaVM* jvm = nullptr;
203 env->GetJavaVM(&jvm);
204 CHECK(jvm);
205 return jvm;
206 }
207
jniIds(JNIEnv * env)208 const JniIds& jniIds(JNIEnv* env) {
209 static const JniIds ids(env);
210 return ids;
211 }
212
checkAndClearJavaException(JNIEnv * env,std::string_view method)213 bool checkAndClearJavaException(JNIEnv* env, std::string_view method) {
214 if (!env->ExceptionCheck()) {
215 return false;
216 }
217
218 LOG(ERROR) << "Java exception during DataLoader::" << method;
219 env->ExceptionDescribe();
220 env->ExceptionClear();
221 return true;
222 }
223
reportStatusViaCallback(JNIEnv * env,jobject listener,jint storageId,jint status)224 bool reportStatusViaCallback(JNIEnv* env, jobject listener, jint storageId, jint status) {
225 if (listener == nullptr) {
226 ALOGE("No listener object to talk to IncrementalService. "
227 "DataLoaderId=%d, "
228 "status=%d",
229 storageId, status);
230 return false;
231 }
232
233 const auto& jni = jniIds(env);
234
235 env->CallVoidMethod(listener, jni.listenerOnStatusChanged, storageId, status);
236 ALOGI("Reported status back to IncrementalService. DataLoaderId=%d, "
237 "status=%d",
238 storageId, status);
239
240 return true;
241 }
242
243 class DataLoaderConnector;
244 using DataLoaderConnectorPtr = std::shared_ptr<DataLoaderConnector>;
245 using DataLoaderConnectorsMap = std::unordered_map<int, DataLoaderConnectorPtr>;
246
247 struct Globals {
Globals__anone91230200111::Globals248 Globals() { managedDataLoaderFactory = new android::dataloader::ManagedDataLoaderFactory(); }
249
250 DataLoaderFactory* managedDataLoaderFactory = nullptr;
251 DataLoaderFactory* legacyDataLoaderFactory = nullptr;
252 DataLoaderFactory* dataLoaderFactory = nullptr;
253
254 std::mutex dataLoaderConnectorsLock;
255 // id->DataLoader map
256 DataLoaderConnectorsMap dataLoaderConnectors GUARDED_BY(dataLoaderConnectorsLock);
257
258 std::atomic_bool stopped;
259 std::thread pendingReadsLooperThread;
260 std::thread logLooperThread;
261 std::vector<ReadInfo> pendingReads;
262 std::vector<ReadInfo> pageReads;
263 std::vector<ReadInfoWithUid> pendingReadsWithUid;
264 std::vector<ReadInfoWithUid> pageReadsWithUid;
265 };
266
globals()267 static Globals& globals() {
268 static Globals globals;
269 return globals;
270 }
271
272 struct IncFsLooper : public android::Looper {
IncFsLooper__anone91230200111::IncFsLooper273 IncFsLooper() : Looper(/*allowNonCallbacks=*/false) {}
~IncFsLooper__anone91230200111::IncFsLooper274 ~IncFsLooper() {}
275 };
276
pendingReadsLooper()277 static android::Looper& pendingReadsLooper() {
278 static IncFsLooper pendingReadsLooper;
279 return pendingReadsLooper;
280 }
281
logLooper()282 static android::Looper& logLooper() {
283 static IncFsLooper logLooper;
284 return logLooper;
285 }
286
287 struct DataLoaderParamsPair {
288 static DataLoaderParamsPair createFromManaged(JNIEnv* env, jobject params);
289
dataLoaderParams__anone91230200111::DataLoaderParamsPair290 const android::dataloader::DataLoaderParams& dataLoaderParams() const {
291 return mDataLoaderParams;
292 }
ndkDataLoaderParams__anone91230200111::DataLoaderParamsPair293 const ::DataLoaderParams& ndkDataLoaderParams() const { return mNDKDataLoaderParams; }
294
295 private:
296 DataLoaderParamsPair(android::dataloader::DataLoaderParams&& dataLoaderParams);
297
298 android::dataloader::DataLoaderParams mDataLoaderParams;
299 ::DataLoaderParams mNDKDataLoaderParams;
300 };
301
302 static constexpr auto kPendingReadsBufferSize = 256;
303
304 class DataLoaderConnector : public android::dataloader::FilesystemConnector,
305 public android::dataloader::StatusListener {
306 public:
DataLoaderConnector(JNIEnv * env,jobject service,jint storageId,UniqueControl control,jobject serviceConnector,jobject callbackControl,jobject listener)307 DataLoaderConnector(JNIEnv* env, jobject service, jint storageId, UniqueControl control,
308 jobject serviceConnector, jobject callbackControl, jobject listener)
309 : mJvm(getJavaVM(env)),
310 mService(env->NewGlobalRef(service)),
311 mServiceConnector(env->NewGlobalRef(serviceConnector)),
312 mCallbackControl(env->NewGlobalRef(callbackControl)),
313 mListener(env->NewGlobalRef(listener)),
314 mStorageId(storageId),
315 mControl(std::move(control)) {
316 CHECK(mJvm != nullptr);
317 }
318 DataLoaderConnector(const DataLoaderConnector&) = delete;
319 DataLoaderConnector(const DataLoaderConnector&&) = delete;
~DataLoaderConnector()320 virtual ~DataLoaderConnector() {
321 if (mDataLoader && mDataLoader->onDestroy) {
322 mDataLoader->onDestroy(mDataLoader);
323 checkAndClearJavaException(__func__);
324 }
325 mDataLoader = nullptr;
326
327 JNIEnv* env = GetOrAttachJNIEnvironment(mJvm);
328
329 const auto& jni = jniIds(env);
330 reportStatusViaCallback(env, mListener, mStorageId, jni.constants.DATA_LOADER_DESTROYED);
331
332 env->DeleteGlobalRef(mService);
333 env->DeleteGlobalRef(mServiceConnector);
334 env->DeleteGlobalRef(mCallbackControl);
335 env->DeleteGlobalRef(mListener);
336 } // to avoid delete-non-virtual-dtor
337
tryFactory(DataLoaderFactory * factory,bool withFeatures,const DataLoaderParamsPair & params,jobject managedParams)338 bool tryFactory(DataLoaderFactory* factory, bool withFeatures,
339 const DataLoaderParamsPair& params, jobject managedParams) {
340 if (!factory) {
341 return true;
342 }
343
344 // Let's try the non-default first.
345 mDataLoader = factory->onCreate(factory, ¶ms.ndkDataLoaderParams(), this, this, mJvm,
346 mService, managedParams);
347 if (checkAndClearJavaException(__func__)) {
348 return false;
349 }
350 if (!mDataLoader) {
351 return true;
352 }
353
354 mDataLoaderFeatures = withFeatures && mDataLoader->getFeatures
355 ? mDataLoader->getFeatures(mDataLoader)
356 : DATA_LOADER_FEATURE_NONE;
357 if (mDataLoaderFeatures & DATA_LOADER_FEATURE_UID) {
358 ALOGE("DataLoader supports UID");
359 CHECK(mDataLoader->onPageReadsWithUid);
360 CHECK(mDataLoader->onPendingReadsWithUid);
361 }
362 return true;
363 }
364
onCreate(const DataLoaderParamsPair & params,jobject managedParams)365 bool onCreate(const DataLoaderParamsPair& params, jobject managedParams) {
366 CHECK(mDataLoader == nullptr);
367
368 if (!mDataLoader &&
369 !tryFactory(globals().dataLoaderFactory, /*withFeatures=*/true, params,
370 managedParams)) {
371 return false;
372 }
373 if (!mDataLoader &&
374 !tryFactory(globals().legacyDataLoaderFactory, /*withFeatures=*/false, params,
375 managedParams)) {
376 return false;
377 }
378 if (!mDataLoader &&
379 !tryFactory(globals().managedDataLoaderFactory, /*withFeatures=*/false, params,
380 managedParams)) {
381 return false;
382 }
383 if (!mDataLoader) {
384 return false;
385 }
386
387 return true;
388 }
onStart()389 bool onStart() {
390 CHECK(mDataLoader);
391 bool result = !mDataLoader->onStart || mDataLoader->onStart(mDataLoader);
392 if (checkAndClearJavaException(__func__)) {
393 result = false;
394 }
395 mRunning = result;
396 return result;
397 }
onStop()398 void onStop() {
399 CHECK(mDataLoader);
400
401 // Stopping both loopers and waiting for them to exit - we should be able to acquire/release
402 // both mutexes.
403 mRunning = false;
404 std::lock_guard{mPendingReadsLooperBusy}; // NOLINT
405 std::lock_guard{mLogLooperBusy}; // NOLINT
406
407 if (mDataLoader->onStop) {
408 mDataLoader->onStop(mDataLoader);
409 }
410 checkAndClearJavaException(__func__);
411 }
412
onPrepareImage(const android::dataloader::DataLoaderInstallationFiles & addedFiles)413 bool onPrepareImage(const android::dataloader::DataLoaderInstallationFiles& addedFiles) {
414 CHECK(mDataLoader);
415 bool result = !mDataLoader->onPrepareImage ||
416 mDataLoader->onPrepareImage(mDataLoader, addedFiles.data(), addedFiles.size());
417 return result;
418 }
419
420 template <class ReadInfoType>
onPendingReadsLooperEvent(std::vector<ReadInfoType> & pendingReads)421 int onPendingReadsLooperEvent(std::vector<ReadInfoType>& pendingReads) {
422 CHECK(mDataLoader);
423 std::lock_guard lock{mPendingReadsLooperBusy};
424 while (mRunning.load(std::memory_order_relaxed)) {
425 pendingReads.resize(kPendingReadsBufferSize);
426 if (android::incfs::waitForPendingReads(mControl, 0ms, &pendingReads) !=
427 android::incfs::WaitResult::HaveData ||
428 pendingReads.empty()) {
429 return 1;
430 }
431 if constexpr (std::is_same_v<ReadInfoType, ReadInfo>) {
432 if (mDataLoader->onPendingReads) {
433 mDataLoader->onPendingReads(mDataLoader, pendingReads.data(),
434 pendingReads.size());
435 }
436 } else {
437 if (mDataLoader->onPendingReadsWithUid) {
438 mDataLoader->onPendingReadsWithUid(mDataLoader, pendingReads.data(),
439 pendingReads.size());
440 }
441 }
442 }
443 return 1;
444 }
445
446 template <class ReadInfoType>
onLogLooperEvent(std::vector<ReadInfoType> & pageReads)447 int onLogLooperEvent(std::vector<ReadInfoType>& pageReads) {
448 CHECK(mDataLoader);
449 std::lock_guard lock{mLogLooperBusy};
450 while (mRunning.load(std::memory_order_relaxed)) {
451 pageReads.clear();
452 if (android::incfs::waitForPageReads(mControl, 0ms, &pageReads) !=
453 android::incfs::WaitResult::HaveData ||
454 pageReads.empty()) {
455 return 1;
456 }
457 if constexpr (std::is_same_v<ReadInfoType, ReadInfo>) {
458 if (mDataLoader->onPageReads) {
459 mDataLoader->onPageReads(mDataLoader, pageReads.data(), pageReads.size());
460 }
461 } else {
462 if (mDataLoader->onPageReadsWithUid) {
463 mDataLoader->onPageReadsWithUid(mDataLoader, pageReads.data(),
464 pageReads.size());
465 }
466 }
467 }
468 return 1;
469 }
470
onPendingReadsLooperEvent(std::vector<ReadInfo> & pendingReads,std::vector<ReadInfoWithUid> & pendingReadsWithUid)471 int onPendingReadsLooperEvent(std::vector<ReadInfo>& pendingReads,
472 std::vector<ReadInfoWithUid>& pendingReadsWithUid) {
473 CHECK(mDataLoader);
474 if (mDataLoaderFeatures & DATA_LOADER_FEATURE_UID) {
475 return this->onPendingReadsLooperEvent(pendingReadsWithUid);
476 } else {
477 return this->onPendingReadsLooperEvent(pendingReads);
478 }
479 }
480
onLogLooperEvent(std::vector<ReadInfo> & pageReads,std::vector<ReadInfoWithUid> & pageReadsWithUid)481 int onLogLooperEvent(std::vector<ReadInfo>& pageReads,
482 std::vector<ReadInfoWithUid>& pageReadsWithUid) {
483 CHECK(mDataLoader);
484 if (mDataLoaderFeatures & DATA_LOADER_FEATURE_UID) {
485 return this->onLogLooperEvent(pageReadsWithUid);
486 } else {
487 return this->onLogLooperEvent(pageReads);
488 }
489 }
490
writeData(jstring name,jlong offsetBytes,jlong lengthBytes,jobject incomingFd) const491 void writeData(jstring name, jlong offsetBytes, jlong lengthBytes, jobject incomingFd) const {
492 CHECK(mCallbackControl);
493 JNIEnv* env = GetOrAttachJNIEnvironment(mJvm);
494 const auto& jni = jniIds(env);
495 env->CallVoidMethod(mCallbackControl, jni.callbackControlWriteData, name, offsetBytes,
496 lengthBytes, incomingFd);
497 }
498
openForSpecialOps(FileId fid) const499 android::incfs::UniqueFd openForSpecialOps(FileId fid) const {
500 return android::incfs::openForSpecialOps(mControl, fid);
501 }
502
writeBlocks(android::dataloader::Span<const IncFsDataBlock> blocks) const503 int writeBlocks(android::dataloader::Span<const IncFsDataBlock> blocks) const {
504 return android::incfs::writeBlocks(blocks);
505 }
506
getRawMetadata(FileId fid,char buffer[],size_t * bufferSize) const507 int getRawMetadata(FileId fid, char buffer[], size_t* bufferSize) const {
508 return IncFs_GetMetadataById(mControl, fid, buffer, bufferSize);
509 }
510
setParams(DataLoaderFilesystemParams params) const511 bool setParams(DataLoaderFilesystemParams params) const {
512 CHECK(mServiceConnector);
513 JNIEnv* env = GetOrAttachJNIEnvironment(mJvm);
514 const auto& jni = jniIds(env);
515 int result = env->CallIntMethod(mServiceConnector,
516 jni.incrementalServiceConnectorSetStorageParams,
517 params.readLogsEnabled);
518 if (result != 0) {
519 LOG(ERROR) << "setStorageParams failed with error: " << result;
520 }
521 if (checkAndClearJavaException(__func__)) {
522 return false;
523 }
524 return (result == 0);
525 }
526
reportStatus(DataLoaderStatus status)527 bool reportStatus(DataLoaderStatus status) {
528 JNIEnv* env = GetOrAttachJNIEnvironment(mJvm);
529 const auto& jni = jniIds(env);
530
531 jint osStatus;
532 switch (status) {
533 case DATA_LOADER_UNAVAILABLE:
534 osStatus = jni.constants.DATA_LOADER_UNAVAILABLE;
535 break;
536 case DATA_LOADER_UNRECOVERABLE:
537 osStatus = jni.constants.DATA_LOADER_UNRECOVERABLE;
538 break;
539 default: {
540 ALOGE("Unable to report invalid status. status=%d", status);
541 return false;
542 }
543 }
544 return reportStatusViaCallback(env, mListener, mStorageId, osStatus);
545 }
546
checkAndClearJavaException(std::string_view method) const547 bool checkAndClearJavaException(std::string_view method) const {
548 JNIEnv* env = GetOrAttachJNIEnvironment(mJvm);
549 return ::checkAndClearJavaException(env, method);
550 }
551
control() const552 const UniqueControl& control() const { return mControl; }
getListenerLocalRef(JNIEnv * env) const553 jobject getListenerLocalRef(JNIEnv* env) const { return env->NewLocalRef(mListener); }
554
555 private:
556 JavaVM* const mJvm;
557 jobject const mService;
558 jobject const mServiceConnector;
559 jobject const mCallbackControl;
560 jobject const mListener;
561
562 jint const mStorageId;
563 UniqueControl const mControl;
564
565 ::DataLoader* mDataLoader = nullptr;
566 DataLoaderFeatures mDataLoaderFeatures = DATA_LOADER_FEATURE_NONE;
567
568 std::mutex mPendingReadsLooperBusy;
569 std::mutex mLogLooperBusy;
570 std::atomic<bool> mRunning{false};
571 };
572
onPendingReadsLooperEvent(int fd,int events,void * data)573 static int onPendingReadsLooperEvent(int fd, int events, void* data) {
574 if (globals().stopped) {
575 // No more listeners.
576 return 0;
577 }
578 auto&& dataLoaderConnector = (DataLoaderConnector*)data;
579 return dataLoaderConnector->onPendingReadsLooperEvent(globals().pendingReads,
580 globals().pendingReadsWithUid);
581 }
582
onLogLooperEvent(int fd,int events,void * data)583 static int onLogLooperEvent(int fd, int events, void* data) {
584 if (globals().stopped) {
585 // No more listeners.
586 return 0;
587 }
588 auto&& dataLoaderConnector = (DataLoaderConnector*)data;
589 return dataLoaderConnector->onLogLooperEvent(globals().pageReads, globals().pageReadsWithUid);
590 }
591
createFdFromManaged(JNIEnv * env,jobject pfd)592 static int createFdFromManaged(JNIEnv* env, jobject pfd) {
593 if (!pfd) {
594 return -1;
595 }
596
597 const auto& jni = jniIds(env);
598 auto managedFd = env->CallObjectMethod(pfd, jni.parcelFileDescriptorGetFileDescriptor);
599 return fcntl(jniGetFDFromFileDescriptor(env, managedFd), F_DUPFD_CLOEXEC, 0);
600 }
601
createServiceConnector(JNIEnv * env,jobject managedControl)602 static jobject createServiceConnector(JNIEnv* env, jobject managedControl) {
603 const auto& jni = jniIds(env);
604 return env->GetObjectField(managedControl, jni.service);
605 }
606
createCallbackControl(JNIEnv * env,jobject managedControl)607 static jobject createCallbackControl(JNIEnv* env, jobject managedControl) {
608 const auto& jni = jniIds(env);
609 return env->GetObjectField(managedControl, jni.callback);
610 }
611
createIncFsControlFromManaged(JNIEnv * env,jobject managedControl)612 static UniqueControl createIncFsControlFromManaged(JNIEnv* env, jobject managedControl) {
613 const auto& jni = jniIds(env);
614 auto managedIncControl = env->GetObjectField(managedControl, jni.incremental);
615 if (!managedIncControl) {
616 return UniqueControl();
617 }
618 auto cmd = createFdFromManaged(env, env->GetObjectField(managedIncControl, jni.controlCmd));
619 auto pr = createFdFromManaged(env,
620 env->GetObjectField(managedIncControl, jni.controlPendingReads));
621 auto log = createFdFromManaged(env, env->GetObjectField(managedIncControl, jni.controlLog));
622 auto blocksWritten =
623 createFdFromManaged(env,
624 env->GetObjectField(managedIncControl, jni.controlBlocksWritten));
625 return android::incfs::createControl(cmd, pr, log, blocksWritten);
626 }
627
DataLoaderParamsPair(android::dataloader::DataLoaderParams && dataLoaderParams)628 DataLoaderParamsPair::DataLoaderParamsPair(android::dataloader::DataLoaderParams&& dataLoaderParams)
629 : mDataLoaderParams(std::move(dataLoaderParams)) {
630 mNDKDataLoaderParams.type = mDataLoaderParams.type();
631 mNDKDataLoaderParams.packageName = mDataLoaderParams.packageName().c_str();
632 mNDKDataLoaderParams.className = mDataLoaderParams.className().c_str();
633 mNDKDataLoaderParams.arguments = mDataLoaderParams.arguments().c_str();
634 }
635
createFromManaged(JNIEnv * env,jobject managedParams)636 DataLoaderParamsPair DataLoaderParamsPair::createFromManaged(JNIEnv* env, jobject managedParams) {
637 const auto& jni = jniIds(env);
638
639 const DataLoaderType type = (DataLoaderType)env->GetIntField(managedParams, jni.paramsType);
640
641 ScopedLocalRef<jstring> paramsPackageName(env,
642 GetStringField(env, managedParams,
643 jni.paramsPackageName));
644 ScopedLocalRef<jstring> paramsClassName(env,
645 GetStringField(env, managedParams,
646 jni.paramsClassName));
647 ScopedLocalRef<jstring> paramsArguments(env,
648 GetStringField(env, managedParams,
649 jni.paramsArguments));
650 ScopedUtfChars package(env, paramsPackageName.get());
651 ScopedUtfChars className(env, paramsClassName.get());
652 ScopedUtfChars arguments(env, paramsArguments.get());
653 return DataLoaderParamsPair(android::dataloader::DataLoaderParams(type, package.c_str(),
654 className.c_str(),
655 arguments.c_str()));
656 }
657
pendingReadsLooperThread()658 static void pendingReadsLooperThread() {
659 constexpr auto kTimeoutMsecs = -1;
660 while (!globals().stopped) {
661 pendingReadsLooper().pollAll(kTimeoutMsecs);
662 }
663 }
664
logLooperThread()665 static void logLooperThread() {
666 constexpr auto kTimeoutMsecs = -1;
667 while (!globals().stopped) {
668 logLooper().pollAll(kTimeoutMsecs);
669 }
670 }
671
pathFromFd(int fd)672 static std::string pathFromFd(int fd) {
673 static constexpr char fdNameFormat[] = "/proc/self/fd/%d";
674 char fdNameBuffer[NELEM(fdNameFormat) + 11 + 1]; // max int length + '\0'
675 snprintf(fdNameBuffer, NELEM(fdNameBuffer), fdNameFormat, fd);
676
677 std::string res;
678 // lstat() is supposed to return us exactly the needed buffer size, but
679 // somehow it may also return a smaller (but still >0) st_size field.
680 // That's why let's only use it for the initial estimate.
681 struct stat st = {};
682 if (::lstat(fdNameBuffer, &st) || st.st_size == 0) {
683 st.st_size = PATH_MAX;
684 }
685 auto bufSize = st.st_size;
686 for (;;) {
687 res.resize(bufSize + 1, '\0');
688 auto size = ::readlink(fdNameBuffer, &res[0], res.size());
689 if (size < 0) {
690 return {};
691 }
692 if (size > bufSize) {
693 // File got renamed in between lstat() and readlink() calls? Retry.
694 bufSize *= 2;
695 continue;
696 }
697 res.resize(size);
698 return res;
699 }
700 }
701
702 } // namespace
703
DataLoader_Initialize(struct::DataLoaderFactory * factory)704 void DataLoader_Initialize(struct ::DataLoaderFactory* factory) {
705 CHECK(factory) << "DataLoader factory is invalid.";
706 globals().legacyDataLoaderFactory = factory;
707 }
708
DataLoader_Initialize_WithFeatures(struct::DataLoaderFactory * factory)709 void DataLoader_Initialize_WithFeatures(struct ::DataLoaderFactory* factory) {
710 CHECK(factory) << "DataLoader factory is invalid.";
711 globals().dataLoaderFactory = factory;
712 }
713
DataLoader_FilesystemConnector_writeData(DataLoaderFilesystemConnectorPtr ifs,jstring name,jlong offsetBytes,jlong lengthBytes,jobject incomingFd)714 void DataLoader_FilesystemConnector_writeData(DataLoaderFilesystemConnectorPtr ifs, jstring name,
715 jlong offsetBytes, jlong lengthBytes,
716 jobject incomingFd) {
717 auto connector = static_cast<DataLoaderConnector*>(ifs);
718 return connector->writeData(name, offsetBytes, lengthBytes, incomingFd);
719 }
720
DataLoader_FilesystemConnector_openForSpecialOps(DataLoaderFilesystemConnectorPtr ifs,IncFsFileId fid)721 int DataLoader_FilesystemConnector_openForSpecialOps(DataLoaderFilesystemConnectorPtr ifs,
722 IncFsFileId fid) {
723 auto connector = static_cast<DataLoaderConnector*>(ifs);
724 return connector->openForSpecialOps(fid).release();
725 }
726
DataLoader_FilesystemConnector_writeBlocks(DataLoaderFilesystemConnectorPtr ifs,const IncFsDataBlock blocks[],int blocksCount)727 int DataLoader_FilesystemConnector_writeBlocks(DataLoaderFilesystemConnectorPtr ifs,
728 const IncFsDataBlock blocks[], int blocksCount) {
729 auto connector = static_cast<DataLoaderConnector*>(ifs);
730 return connector->writeBlocks({blocks, static_cast<size_t>(blocksCount)});
731 }
732
DataLoader_FilesystemConnector_getRawMetadata(DataLoaderFilesystemConnectorPtr ifs,IncFsFileId fid,char buffer[],size_t * bufferSize)733 int DataLoader_FilesystemConnector_getRawMetadata(DataLoaderFilesystemConnectorPtr ifs,
734 IncFsFileId fid, char buffer[],
735 size_t* bufferSize) {
736 auto connector = static_cast<DataLoaderConnector*>(ifs);
737 return connector->getRawMetadata(fid, buffer, bufferSize);
738 }
739
DataLoader_FilesystemConnector_setParams(DataLoaderFilesystemConnectorPtr ifs,DataLoaderFilesystemParams params)740 bool DataLoader_FilesystemConnector_setParams(DataLoaderFilesystemConnectorPtr ifs,
741 DataLoaderFilesystemParams params) {
742 auto connector = static_cast<DataLoaderConnector*>(ifs);
743 return connector->setParams(params);
744 }
745
DataLoader_StatusListener_reportStatus(DataLoaderStatusListenerPtr listener,DataLoaderStatus status)746 int DataLoader_StatusListener_reportStatus(DataLoaderStatusListenerPtr listener,
747 DataLoaderStatus status) {
748 auto connector = static_cast<DataLoaderConnector*>(listener);
749 return connector->reportStatus(status);
750 }
751
DataLoaderService_OnCreate(JNIEnv * env,jobject service,jint storageId,jobject control,jobject params,jobject listener)752 bool DataLoaderService_OnCreate(JNIEnv* env, jobject service, jint storageId, jobject control,
753 jobject params, jobject listener) {
754 {
755 std::lock_guard lock{globals().dataLoaderConnectorsLock};
756 auto dlIt = globals().dataLoaderConnectors.find(storageId);
757 if (dlIt != globals().dataLoaderConnectors.end()) {
758 ALOGI("id(%d): already exist, skipping creation.", storageId);
759 return true;
760 }
761 }
762 auto nativeControl = createIncFsControlFromManaged(env, control);
763 if (nativeControl) {
764 using namespace android::incfs;
765 ALOGI("DataLoader::create incremental fds: %d/%d/%d/%d", nativeControl.cmd(),
766 nativeControl.pendingReads(), nativeControl.logs(), nativeControl.blocksWritten());
767 auto cmdPath = pathFromFd(nativeControl.cmd());
768 auto dir = path::dirName(cmdPath);
769 ALOGI("DataLoader::create incremental dir: %s, files: %s/%s/%s/%s",
770 details::c_str(dir).get(), details::c_str(path::baseName(cmdPath)).get(),
771 details::c_str(path::baseName(pathFromFd(nativeControl.pendingReads()))).get(),
772 details::c_str(path::baseName(pathFromFd(nativeControl.logs()))).get(),
773 details::c_str(path::baseName(pathFromFd(nativeControl.blocksWritten()))).get());
774 } else {
775 ALOGI("DataLoader::create no incremental control");
776 }
777
778 auto nativeParams = DataLoaderParamsPair::createFromManaged(env, params);
779 ALOGI("DataLoader::create params: %d|%s|%s|%s", nativeParams.dataLoaderParams().type(),
780 nativeParams.dataLoaderParams().packageName().c_str(),
781 nativeParams.dataLoaderParams().className().c_str(),
782 nativeParams.dataLoaderParams().arguments().c_str());
783
784 auto serviceConnector = createServiceConnector(env, control);
785 auto callbackControl = createCallbackControl(env, control);
786
787 auto reportUnavailable = [env, storageId](jobject listener) {
788 const auto& jni = jniIds(env);
789 reportStatusViaCallback(env, listener, storageId, jni.constants.DATA_LOADER_UNAVAILABLE);
790 };
791 // By default, it's disabled. Need to assign listener to enable.
792 std::unique_ptr<_jobject, decltype(reportUnavailable)>
793 reportUnavailableOnExit(nullptr, reportUnavailable);
794
795 auto dataLoaderConnector =
796 std::make_unique<DataLoaderConnector>(env, service, storageId, std::move(nativeControl),
797 serviceConnector, callbackControl, listener);
798 bool created = dataLoaderConnector->onCreate(nativeParams, params);
799 {
800 std::lock_guard lock{globals().dataLoaderConnectorsLock};
801 auto [dlIt, dlInserted] =
802 globals().dataLoaderConnectors.try_emplace(storageId,
803 std::move(dataLoaderConnector));
804 if (!dlInserted) {
805 ALOGE("id(%d): already exist, skipping creation.", storageId);
806 return false;
807 }
808
809 if (!created) {
810 globals().dataLoaderConnectors.erase(dlIt);
811 // Enable the reporter.
812 reportUnavailableOnExit.reset(listener);
813 return false;
814 }
815 }
816
817 const auto& jni = jniIds(env);
818 reportStatusViaCallback(env, listener, storageId, jni.constants.DATA_LOADER_CREATED);
819
820 return true;
821 }
822
DataLoaderService_OnStart(JNIEnv * env,jint storageId)823 bool DataLoaderService_OnStart(JNIEnv* env, jint storageId) {
824 auto destroyAndReportUnavailable = [env, storageId](jobject listener) {
825 // Because of the MT the installer can call commit and recreate/restart dataLoader before
826 // system server has a change to destroy it.
827 DataLoaderService_OnDestroy(env, storageId);
828 const auto& jni = jniIds(env);
829 reportStatusViaCallback(env, listener, storageId, jni.constants.DATA_LOADER_UNAVAILABLE);
830 };
831 // By default, it's disabled. Need to assign listener to enable.
832 std::unique_ptr<_jobject, decltype(destroyAndReportUnavailable)>
833 destroyAndReportUnavailableOnExit(nullptr, destroyAndReportUnavailable);
834
835 DataLoaderConnectorPtr dataLoaderConnector;
836 {
837 std::lock_guard lock{globals().dataLoaderConnectorsLock};
838 auto dlIt = globals().dataLoaderConnectors.find(storageId);
839 if (dlIt == globals().dataLoaderConnectors.end()) {
840 ALOGE("Failed to start id(%d): not found", storageId);
841 return false;
842 }
843 dataLoaderConnector = dlIt->second;
844 }
845 const UniqueControl* control = &(dataLoaderConnector->control());
846 jobject listener = dataLoaderConnector->getListenerLocalRef(env);
847
848 if (!dataLoaderConnector->onStart()) {
849 ALOGE("Failed to start id(%d): onStart returned false", storageId);
850 destroyAndReportUnavailableOnExit.reset(listener);
851 return false;
852 }
853
854 if (control->pendingReads() >= 0) {
855 auto&& looper = pendingReadsLooper();
856 if (!globals().pendingReadsLooperThread.joinable()) {
857 std::lock_guard lock{globals().dataLoaderConnectorsLock};
858 if (!globals().pendingReadsLooperThread.joinable()) {
859 globals().pendingReadsLooperThread = std::thread(&pendingReadsLooperThread);
860 }
861 }
862
863 looper.addFd(control->pendingReads(), android::Looper::POLL_CALLBACK,
864 android::Looper::EVENT_INPUT, &onPendingReadsLooperEvent,
865 dataLoaderConnector.get());
866 looper.wake();
867 }
868
869 if (control->logs() >= 0) {
870 auto&& looper = logLooper();
871 if (!globals().logLooperThread.joinable()) {
872 std::lock_guard lock{globals().dataLoaderConnectorsLock};
873 if (!globals().logLooperThread.joinable()) {
874 globals().logLooperThread = std::thread(&logLooperThread);
875 }
876 }
877
878 looper.addFd(control->logs(), android::Looper::POLL_CALLBACK, android::Looper::EVENT_INPUT,
879 &onLogLooperEvent, dataLoaderConnector.get());
880 looper.wake();
881 }
882
883 const auto& jni = jniIds(env);
884 reportStatusViaCallback(env, listener, storageId, jni.constants.DATA_LOADER_STARTED);
885
886 return true;
887 }
888
DataLoaderService_OnStop_NoStatus(const UniqueControl * control,const DataLoaderConnectorPtr & dataLoaderConnector)889 static void DataLoaderService_OnStop_NoStatus(const UniqueControl* control,
890 const DataLoaderConnectorPtr& dataLoaderConnector) {
891 if (control->pendingReads() >= 0) {
892 pendingReadsLooper().removeFd(control->pendingReads());
893 pendingReadsLooper().wake();
894 }
895 if (control->logs() >= 0) {
896 logLooper().removeFd(control->logs());
897 logLooper().wake();
898 }
899 dataLoaderConnector->onStop();
900 }
901
DataLoaderService_OnStop(JNIEnv * env,jint storageId)902 bool DataLoaderService_OnStop(JNIEnv* env, jint storageId) {
903 DataLoaderConnectorPtr dataLoaderConnector;
904 {
905 std::lock_guard lock{globals().dataLoaderConnectorsLock};
906 auto dlIt = globals().dataLoaderConnectors.find(storageId);
907 if (dlIt == globals().dataLoaderConnectors.end()) {
908 ALOGI("Failed to stop id(%d): not found", storageId);
909 return true;
910 }
911 dataLoaderConnector = dlIt->second;
912 }
913 const UniqueControl* control = &(dataLoaderConnector->control());
914 jobject listener = dataLoaderConnector->getListenerLocalRef(env);
915
916 // Just stop.
917 DataLoaderService_OnStop_NoStatus(control, dataLoaderConnector);
918
919 const auto& jni = jniIds(env);
920 reportStatusViaCallback(env, listener, storageId, jni.constants.DATA_LOADER_STOPPED);
921
922 return true;
923 }
924
DataLoaderService_OnDestroy(JNIEnv * env,jint storageId)925 bool DataLoaderService_OnDestroy(JNIEnv* env, jint storageId) {
926 DataLoaderConnectorPtr dataLoaderConnector;
927 {
928 std::lock_guard lock{globals().dataLoaderConnectorsLock};
929 auto dlIt = globals().dataLoaderConnectors.find(storageId);
930 if (dlIt == globals().dataLoaderConnectors.end()) {
931 ALOGI("Failed to destroy id(%d): not found", storageId);
932 return true;
933 }
934 dataLoaderConnector = std::move(dlIt->second);
935 globals().dataLoaderConnectors.erase(dlIt);
936 }
937 const UniqueControl* control = &(dataLoaderConnector->control());
938
939 // Stop/destroy.
940 DataLoaderService_OnStop_NoStatus(control, dataLoaderConnector);
941 // This will destroy the last instance of the DataLoaderConnectorPtr and should trigger the
942 // destruction of the DataLoader. However if there are any hanging instances, the destruction
943 // will be postponed. E.g. OnPrepareImage in progress at the same time we call OnDestroy.
944 dataLoaderConnector = {};
945
946 return true;
947 }
948
949 struct DataLoaderInstallationFilesPair {
950 static DataLoaderInstallationFilesPair createFromManaged(JNIEnv* env, jobjectArray jfiles);
951
952 using Files = std::vector<android::dataloader::DataLoaderInstallationFile>;
filesDataLoaderInstallationFilesPair953 const Files& files() const { return mFiles; }
954
955 using NDKFiles = std::vector<::DataLoaderInstallationFile>;
ndkFilesDataLoaderInstallationFilesPair956 const NDKFiles& ndkFiles() const { return mNDKFiles; }
957
958 private:
959 DataLoaderInstallationFilesPair(Files&& files);
960
961 Files mFiles;
962 NDKFiles mNDKFiles;
963 };
964
createFromManaged(JNIEnv * env,jobjectArray jfiles)965 DataLoaderInstallationFilesPair DataLoaderInstallationFilesPair::createFromManaged(
966 JNIEnv* env, jobjectArray jfiles) {
967 const auto& jni = jniIds(env);
968
969 // jfiles is a Java array of InstallationFileParcel
970 auto size = env->GetArrayLength(jfiles);
971 DataLoaderInstallationFilesPair::Files files;
972 files.reserve(size);
973
974 for (int i = 0; i < size; ++i) {
975 ScopedLocalRef<jobject> jfile(env, env->GetObjectArrayElement(jfiles, i));
976
977 DataLoaderLocation location =
978 (DataLoaderLocation)env->GetIntField(jfile.get(), jni.installationFileLocation);
979 ScopedUtfChars name(env, GetStringField(env, jfile.get(), jni.installationFileName));
980 IncFsSize size = env->GetLongField(jfile.get(), jni.installationFileLengthBytes);
981
982 ScopedLocalRef<jbyteArray> jmetadataBytes(env,
983 GetByteArrayField(env, jfile.get(),
984 jni.installationFileMetadata));
985 auto metadataElements = env->GetByteArrayElements(jmetadataBytes.get(), nullptr);
986 auto metadataLength = env->GetArrayLength(jmetadataBytes.get());
987 RawMetadata metadata(metadataElements, metadataElements + metadataLength);
988 env->ReleaseByteArrayElements(jmetadataBytes.get(), metadataElements, 0);
989
990 files.emplace_back(location, name.c_str(), size, std::move(metadata));
991 }
992
993 return DataLoaderInstallationFilesPair(std::move(files));
994 }
995
DataLoaderInstallationFilesPair(Files && files)996 DataLoaderInstallationFilesPair::DataLoaderInstallationFilesPair(Files&& files)
997 : mFiles(std::move(files)) {
998 const auto size = mFiles.size();
999 mNDKFiles.resize(size);
1000 for (size_t i = 0; i < size; ++i) {
1001 const auto& file = mFiles[i];
1002 auto&& ndkFile = mNDKFiles[i];
1003
1004 ndkFile.location = file.location();
1005 ndkFile.name = file.name().c_str();
1006 ndkFile.size = file.size();
1007 ndkFile.metadata.data = file.metadata().data();
1008 ndkFile.metadata.size = file.metadata().size();
1009 }
1010 }
1011
DataLoaderService_OnPrepareImage(JNIEnv * env,jint storageId,jobjectArray addedFiles,jobjectArray removedFiles)1012 bool DataLoaderService_OnPrepareImage(JNIEnv* env, jint storageId, jobjectArray addedFiles,
1013 jobjectArray removedFiles) {
1014 DataLoaderConnectorPtr dataLoaderConnector;
1015 {
1016 std::lock_guard lock{globals().dataLoaderConnectorsLock};
1017 auto dlIt = globals().dataLoaderConnectors.find(storageId);
1018 if (dlIt == globals().dataLoaderConnectors.end()) {
1019 ALOGE("Failed to handle onPrepareImage for id(%d): not found", storageId);
1020 return false;
1021 }
1022 dataLoaderConnector = dlIt->second;
1023 }
1024 jobject listener = dataLoaderConnector->getListenerLocalRef(env);
1025
1026 auto addedFilesPair = DataLoaderInstallationFilesPair::createFromManaged(env, addedFiles);
1027 bool result = dataLoaderConnector->onPrepareImage(addedFilesPair.ndkFiles());
1028
1029 const auto& jni = jniIds(env);
1030
1031 if (checkAndClearJavaException(env, "onPrepareImage")) {
1032 reportStatusViaCallback(env, listener, storageId, jni.constants.DATA_LOADER_UNAVAILABLE);
1033 return false;
1034 }
1035
1036 reportStatusViaCallback(env, listener, storageId,
1037 result ? jni.constants.DATA_LOADER_IMAGE_READY
1038 : jni.constants.DATA_LOADER_IMAGE_NOT_READY);
1039
1040 return result;
1041 }
1042