/* * Copyright (C) 2012 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_NDEBUG 0 #define LOG_TAG "Crypto" #include <utils/Log.h> #include <dirent.h> #include <dlfcn.h> #include <binder/IMemory.h> #include <media/Crypto.h> #include <media/DrmPluginPath.h> #include <media/hardware/CryptoAPI.h> #include <media/stagefright/foundation/ADebug.h> #include <media/stagefright/foundation/AString.h> #include <media/stagefright/foundation/hexdump.h> #include <media/stagefright/MediaErrors.h> namespace android { KeyedVector<Vector<uint8_t>, String8> Crypto::mUUIDToLibraryPathMap; KeyedVector<String8, wp<SharedLibrary> > Crypto::mLibraryPathToOpenLibraryMap; Mutex Crypto::mMapLock; static bool operator<(const Vector<uint8_t> &lhs, const Vector<uint8_t> &rhs) { if (lhs.size() < rhs.size()) { return true; } else if (lhs.size() > rhs.size()) { return false; } return memcmp((void *)lhs.array(), (void *)rhs.array(), rhs.size()) < 0; } Crypto::Crypto() : mInitCheck(NO_INIT), mFactory(NULL), mPlugin(NULL) { } Crypto::~Crypto() { delete mPlugin; mPlugin = NULL; closeFactory(); } void Crypto::closeFactory() { delete mFactory; mFactory = NULL; mLibrary.clear(); } status_t Crypto::initCheck() const { return mInitCheck; } /* * Search the plugins directory for a plugin that supports the scheme * specified by uuid * * If found: * mLibrary holds a strong pointer to the dlopen'd library * mFactory is set to the library's factory method * mInitCheck is set to OK * * If not found: * mLibrary is cleared and mFactory are set to NULL * mInitCheck is set to an error (!OK) */ void Crypto::findFactoryForScheme(const uint8_t uuid[16]) { closeFactory(); // lock static maps Mutex::Autolock autoLock(mMapLock); // first check cache Vector<uint8_t> uuidVector; uuidVector.appendArray(uuid, sizeof(uuid[0]) * 16); ssize_t index = mUUIDToLibraryPathMap.indexOfKey(uuidVector); if (index >= 0) { if (loadLibraryForScheme(mUUIDToLibraryPathMap[index], uuid)) { mInitCheck = OK; return; } else { ALOGE("Failed to load from cached library path!"); mInitCheck = ERROR_UNSUPPORTED; return; } } // no luck, have to search String8 dirPath(getDrmPluginPath()); String8 pluginPath; DIR* pDir = opendir(dirPath.string()); if (pDir) { struct dirent* pEntry; while ((pEntry = readdir(pDir))) { pluginPath = dirPath + "/" + pEntry->d_name; if (pluginPath.getPathExtension() == ".so") { if (loadLibraryForScheme(pluginPath, uuid)) { mUUIDToLibraryPathMap.add(uuidVector, pluginPath); mInitCheck = OK; closedir(pDir); return; } } } closedir(pDir); } // try the legacy libdrmdecrypt.so pluginPath = "libdrmdecrypt.so"; if (loadLibraryForScheme(pluginPath, uuid)) { mUUIDToLibraryPathMap.add(uuidVector, pluginPath); mInitCheck = OK; return; } mInitCheck = ERROR_UNSUPPORTED; } bool Crypto::loadLibraryForScheme(const String8 &path, const uint8_t uuid[16]) { // get strong pointer to open shared library ssize_t index = mLibraryPathToOpenLibraryMap.indexOfKey(path); if (index >= 0) { mLibrary = mLibraryPathToOpenLibraryMap[index].promote(); } else { index = mLibraryPathToOpenLibraryMap.add(path, NULL); } if (!mLibrary.get()) { mLibrary = new SharedLibrary(path); if (!*mLibrary) { ALOGE("loadLibraryForScheme failed:%s", mLibrary->lastError()); return false; } mLibraryPathToOpenLibraryMap.replaceValueAt(index, mLibrary); } typedef CryptoFactory *(*CreateCryptoFactoryFunc)(); CreateCryptoFactoryFunc createCryptoFactory = (CreateCryptoFactoryFunc)mLibrary->lookup("createCryptoFactory"); if (createCryptoFactory == NULL || (mFactory = createCryptoFactory()) == NULL || !mFactory->isCryptoSchemeSupported(uuid)) { ALOGE("createCryptoFactory failed:%s", mLibrary->lastError()); closeFactory(); return false; } return true; } bool Crypto::isCryptoSchemeSupported(const uint8_t uuid[16]) { Mutex::Autolock autoLock(mLock); if (mFactory && mFactory->isCryptoSchemeSupported(uuid)) { return true; } findFactoryForScheme(uuid); return (mInitCheck == OK); } status_t Crypto::createPlugin( const uint8_t uuid[16], const void *data, size_t size) { Mutex::Autolock autoLock(mLock); if (mPlugin != NULL) { return -EINVAL; } if (!mFactory || !mFactory->isCryptoSchemeSupported(uuid)) { findFactoryForScheme(uuid); } if (mInitCheck != OK) { return mInitCheck; } return mFactory->createPlugin(uuid, data, size, &mPlugin); } status_t Crypto::destroyPlugin() { Mutex::Autolock autoLock(mLock); if (mInitCheck != OK) { return mInitCheck; } if (mPlugin == NULL) { return -EINVAL; } delete mPlugin; mPlugin = NULL; return OK; } bool Crypto::requiresSecureDecoderComponent(const char *mime) const { Mutex::Autolock autoLock(mLock); if (mInitCheck != OK) { return mInitCheck; } if (mPlugin == NULL) { return -EINVAL; } return mPlugin->requiresSecureDecoderComponent(mime); } ssize_t Crypto::decrypt(const uint8_t key[16], const uint8_t iv[16], CryptoPlugin::Mode mode, const CryptoPlugin::Pattern &pattern, const sp<IMemory> &source, size_t offset, const CryptoPlugin::SubSample *subSamples, size_t numSubSamples, const ICrypto::DestinationBuffer &destination, AString *errorDetailMsg) { Mutex::Autolock autoLock(mLock); if (mInitCheck != OK) { return mInitCheck; } if (mPlugin == NULL) { return -EINVAL; } const void *srcPtr = static_cast<uint8_t *>(source->pointer()) + offset; void *destPtr; bool secure = false; if (destination.mType == kDestinationTypeNativeHandle) { destPtr = static_cast<void *>(destination.mHandle); secure = true; } else { destPtr = destination.mSharedMemory->pointer(); } ssize_t result = mPlugin->decrypt(secure, key, iv, mode, pattern, srcPtr, subSamples, numSubSamples, destPtr, errorDetailMsg); return result; } void Crypto::notifyResolution(uint32_t width, uint32_t height) { Mutex::Autolock autoLock(mLock); if (mInitCheck == OK && mPlugin != NULL) { mPlugin->notifyResolution(width, height); } } status_t Crypto::setMediaDrmSession(const Vector<uint8_t> &sessionId) { Mutex::Autolock autoLock(mLock); status_t result = NO_INIT; if (mInitCheck == OK && mPlugin != NULL) { result = mPlugin->setMediaDrmSession(sessionId); } return result; } } // namespace android