/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "NStorage" #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace android; struct ObbActionListener : public BnObbActionListener { private: sp mStorageManager; public: explicit ObbActionListener(AStorageManager* mgr) : mStorageManager(mgr) {} virtual void onObbResult(const android::String16& filename, const int32_t nonce, const int32_t state); }; class ObbCallback { public: ObbCallback(int32_t _nonce, AStorageManager_obbCallbackFunc _cb, void* _data) : nonce(_nonce) , cb(_cb) , data(_data) {} int32_t nonce; AStorageManager_obbCallbackFunc cb; void* data; }; struct AStorageManager : public RefBase { protected: Mutex mCallbackLock; Vector mCallbacks; volatile int32_t mNextNonce; sp mObbActionListener; sp mMountService; int32_t getNextNonce() { return android_atomic_inc(&mNextNonce); } ObbCallback* registerObbCallback(AStorageManager_obbCallbackFunc func, void* data) { ObbCallback* cb = new ObbCallback(getNextNonce(), func, data); { AutoMutex _l(mCallbackLock); mCallbacks.push(cb); } return cb; } ObbInfo* getObbInfo(char* canonicalPath) { sp obbFile = new ObbFile(); if (!obbFile->readFrom(canonicalPath)) { return nullptr; } String16 fileName(canonicalPath); String16 packageName(obbFile->getPackageName()); size_t length; const unsigned char* salt = obbFile->getSalt(&length); return new ObbInfo(fileName, packageName, obbFile->getVersion(), obbFile->getFlags(), length, salt); } public: AStorageManager() { } bool initialize() { sp sm = defaultServiceManager(); if (sm == NULL) { ALOGE("Couldn't get default ServiceManager\n"); return false; } mMountService = interface_cast(sm->getService(String16("mount"))); if (mMountService == NULL) { ALOGE("Couldn't get connection to MountService\n"); return false; } mObbActionListener = new ObbActionListener(this); return true; } void fireCallback(const char* filename, const int32_t nonce, const int32_t state) { ObbCallback* target = NULL; { AutoMutex _l(mCallbackLock); int N = mCallbacks.size(); for (int i = 0; i < N; i++) { ObbCallback* cb = mCallbacks.editItemAt(i); if (cb->nonce == nonce) { target = cb; mCallbacks.removeAt(i); break; } } } if (target != NULL) { target->cb(filename, state, target->data); delete target; } else { ALOGI("Didn't find the callback handler for: %s\n", filename); } } void mountObb(const char* rawPath, AStorageManager_obbCallbackFunc func, void* data) { // Resolve path before sending to MountService char canonicalPath[PATH_MAX]; if (realpath(rawPath, canonicalPath) == NULL) { ALOGE("mountObb failed to resolve path %s: %s", rawPath, strerror(errno)); return; } sp obbInfo = getObbInfo(canonicalPath); if (obbInfo == nullptr) { ALOGE("Couldn't get obb info for %s: %s", canonicalPath, strerror(errno)); return; } ObbCallback* cb = registerObbCallback(func, data); String16 rawPath16(rawPath); String16 canonicalPath16(canonicalPath); mMountService->mountObb(rawPath16, canonicalPath16, mObbActionListener, cb->nonce, obbInfo); } void unmountObb(const char* filename, const bool force, AStorageManager_obbCallbackFunc func, void* data) { ObbCallback* cb = registerObbCallback(func, data); String16 filename16(filename); mMountService->unmountObb(filename16, force, mObbActionListener, cb->nonce); } int isObbMounted(const char* filename) { String16 filename16(filename); return mMountService->isObbMounted(filename16); } const char* getMountedObbPath(const char* filename) { String16 filename16(filename); String16 path16; if (mMountService->getMountedObbPath(filename16, path16)) { return String8(path16).c_str(); } else { return NULL; } } }; void ObbActionListener::onObbResult(const android::String16& filename, const int32_t nonce, const int32_t state) { mStorageManager->fireCallback(String8(filename).c_str(), nonce, state); } AStorageManager* AStorageManager_new() { sp mgr = new AStorageManager(); if (mgr == NULL || !mgr->initialize()) { return NULL; } mgr->incStrong((void*)AStorageManager_new); return static_cast(mgr.get()); } void AStorageManager_delete(AStorageManager* mgr) { if (mgr) { mgr->decStrong((void*)AStorageManager_new); } } void AStorageManager_mountObb(AStorageManager* mgr, const char* filename, const char* key, AStorageManager_obbCallbackFunc cb, void* data) { if (key != nullptr && key[0] != '\0') { ALOGE("mounting encrypted OBBs is no longer supported"); return; } mgr->mountObb(filename, cb, data); } void AStorageManager_unmountObb(AStorageManager* mgr, const char* filename, const int force, AStorageManager_obbCallbackFunc cb, void* data) { mgr->unmountObb(filename, force != 0, cb, data); } int AStorageManager_isObbMounted(AStorageManager* mgr, const char* filename) { return mgr->isObbMounted(filename) != 0; } const char* AStorageManager_getMountedObbPath(AStorageManager* mgr, const char* filename) { return mgr->getMountedObbPath(filename); }