Home | History | Annotate | Download | only in drmserver
      1 /*
      2  * Copyright (C) 2010 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 //#define LOG_NDEBUG 0
     18 #define LOG_TAG "DrmManager(Native)"
     19 #include "utils/Log.h"
     20 
     21 #include <utils/String8.h>
     22 #include <drm/DrmInfo.h>
     23 #include <drm/DrmInfoEvent.h>
     24 #include <drm/DrmRights.h>
     25 #include <drm/DrmConstraints.h>
     26 #include <drm/DrmMetadata.h>
     27 #include <drm/DrmInfoStatus.h>
     28 #include <drm/DrmInfoRequest.h>
     29 #include <drm/DrmSupportInfo.h>
     30 #include <drm/DrmConvertedStatus.h>
     31 #include <IDrmEngine.h>
     32 
     33 #include "DrmManager.h"
     34 #include "ReadWriteUtils.h"
     35 
     36 #define DECRYPT_FILE_ERROR -1
     37 
     38 using namespace android;
     39 
     40 const String8 DrmManager::EMPTY_STRING("");
     41 
     42 DrmManager::DrmManager() :
     43     mDecryptSessionId(0),
     44     mConvertId(0) {
     45     srand(time(NULL));
     46     memset(mUniqueIdArray, 0, sizeof(bool) * kMaxNumUniqueIds);
     47 }
     48 
     49 DrmManager::~DrmManager() {
     50 
     51 }
     52 
     53 int DrmManager::addUniqueId(bool isNative) {
     54     Mutex::Autolock _l(mLock);
     55 
     56     int uniqueId = -1;
     57     int random = rand();
     58 
     59     for (size_t index = 0; index < kMaxNumUniqueIds; ++index) {
     60         int temp = (random + index) % kMaxNumUniqueIds;
     61         if (!mUniqueIdArray[temp]) {
     62             uniqueId = temp;
     63             mUniqueIdArray[uniqueId] = true;
     64 
     65             if (isNative) {
     66                 // set a flag to differentiate DrmManagerClient
     67                 // created from native side and java side
     68                 uniqueId |= 0x1000;
     69             }
     70             break;
     71         }
     72     }
     73 
     74     // -1 indicates that no unique id can be allocated.
     75     return uniqueId;
     76 }
     77 
     78 void DrmManager::removeUniqueId(int uniqueId) {
     79     Mutex::Autolock _l(mLock);
     80     if (uniqueId & 0x1000) {
     81         // clear the flag for the native side.
     82         uniqueId &= ~(0x1000);
     83     }
     84 
     85     if (uniqueId >= 0 && uniqueId < kMaxNumUniqueIds) {
     86         mUniqueIdArray[uniqueId] = false;
     87     }
     88 }
     89 
     90 status_t DrmManager::loadPlugIns() {
     91 
     92     String8 vendorPluginDirPath("/vendor/lib/drm");
     93     loadPlugIns(vendorPluginDirPath);
     94 
     95     String8 pluginDirPath("/system/lib/drm");
     96     loadPlugIns(pluginDirPath);
     97     return DRM_NO_ERROR;
     98 
     99 }
    100 
    101 status_t DrmManager::loadPlugIns(const String8& plugInDirPath) {
    102     mPlugInManager.loadPlugIns(plugInDirPath);
    103     Vector<String8> plugInPathList = mPlugInManager.getPlugInIdList();
    104     for (unsigned int i = 0; i < plugInPathList.size(); ++i) {
    105         String8 plugInPath = plugInPathList[i];
    106         DrmSupportInfo* info = mPlugInManager.getPlugIn(plugInPath).getSupportInfo(0);
    107         if (NULL != info) {
    108             if (mSupportInfoToPlugInIdMap.indexOfKey(*info) < 0) {
    109                 mSupportInfoToPlugInIdMap.add(*info, plugInPath);
    110             }
    111             delete info;
    112         }
    113     }
    114     return DRM_NO_ERROR;
    115 }
    116 
    117 status_t DrmManager::unloadPlugIns() {
    118     Mutex::Autolock _l(mLock);
    119     mConvertSessionMap.clear();
    120     mDecryptSessionMap.clear();
    121     mPlugInManager.unloadPlugIns();
    122     mSupportInfoToPlugInIdMap.clear();
    123     return DRM_NO_ERROR;
    124 }
    125 
    126 status_t DrmManager::setDrmServiceListener(
    127             int uniqueId, const sp<IDrmServiceListener>& drmServiceListener) {
    128     Mutex::Autolock _l(mListenerLock);
    129     if (NULL != drmServiceListener.get()) {
    130         mServiceListeners.add(uniqueId, drmServiceListener);
    131     } else {
    132         mServiceListeners.removeItem(uniqueId);
    133     }
    134     return DRM_NO_ERROR;
    135 }
    136 
    137 void DrmManager::addClient(int uniqueId) {
    138     Mutex::Autolock _l(mLock);
    139     if (!mSupportInfoToPlugInIdMap.isEmpty()) {
    140         Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
    141         for (unsigned int index = 0; index < plugInIdList.size(); index++) {
    142             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInIdList.itemAt(index));
    143             rDrmEngine.initialize(uniqueId);
    144             rDrmEngine.setOnInfoListener(uniqueId, this);
    145         }
    146     }
    147 }
    148 
    149 void DrmManager::removeClient(int uniqueId) {
    150     Mutex::Autolock _l(mLock);
    151     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
    152     for (unsigned int index = 0; index < plugInIdList.size(); index++) {
    153         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInIdList.itemAt(index));
    154         rDrmEngine.terminate(uniqueId);
    155     }
    156 }
    157 
    158 DrmConstraints* DrmManager::getConstraints(int uniqueId, const String8* path, const int action) {
    159     Mutex::Autolock _l(mLock);
    160     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, *path);
    161     if (EMPTY_STRING != plugInId) {
    162         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    163         return rDrmEngine.getConstraints(uniqueId, path, action);
    164     }
    165     return NULL;
    166 }
    167 
    168 DrmMetadata* DrmManager::getMetadata(int uniqueId, const String8* path) {
    169     Mutex::Autolock _l(mLock);
    170     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, *path);
    171     if (EMPTY_STRING != plugInId) {
    172         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    173         return rDrmEngine.getMetadata(uniqueId, path);
    174     }
    175     return NULL;
    176 }
    177 
    178 bool DrmManager::canHandle(int uniqueId, const String8& path, const String8& mimeType) {
    179     Mutex::Autolock _l(mLock);
    180     const String8 plugInId = getSupportedPlugInId(mimeType);
    181     bool result = (EMPTY_STRING != plugInId) ? true : false;
    182 
    183     if (0 < path.length()) {
    184         if (result) {
    185             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    186             result = rDrmEngine.canHandle(uniqueId, path);
    187         } else {
    188             String8 extension = path.getPathExtension();
    189             if (String8("") != extension) {
    190                 result = canHandle(uniqueId, path);
    191             }
    192         }
    193     }
    194     return result;
    195 }
    196 
    197 DrmInfoStatus* DrmManager::processDrmInfo(int uniqueId, const DrmInfo* drmInfo) {
    198     Mutex::Autolock _l(mLock);
    199     const String8 plugInId = getSupportedPlugInId(drmInfo->getMimeType());
    200     if (EMPTY_STRING != plugInId) {
    201         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    202         return rDrmEngine.processDrmInfo(uniqueId, drmInfo);
    203     }
    204     return NULL;
    205 }
    206 
    207 bool DrmManager::canHandle(int uniqueId, const String8& path) {
    208     bool result = false;
    209     Vector<String8> plugInPathList = mPlugInManager.getPlugInIdList();
    210 
    211     for (unsigned int i = 0; i < plugInPathList.size(); ++i) {
    212         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInPathList[i]);
    213         result = rDrmEngine.canHandle(uniqueId, path);
    214 
    215         if (result) {
    216             break;
    217         }
    218     }
    219     return result;
    220 }
    221 
    222 DrmInfo* DrmManager::acquireDrmInfo(int uniqueId, const DrmInfoRequest* drmInfoRequest) {
    223     Mutex::Autolock _l(mLock);
    224     const String8 plugInId = getSupportedPlugInId(drmInfoRequest->getMimeType());
    225     if (EMPTY_STRING != plugInId) {
    226         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    227         return rDrmEngine.acquireDrmInfo(uniqueId, drmInfoRequest);
    228     }
    229     return NULL;
    230 }
    231 
    232 status_t DrmManager::saveRights(int uniqueId, const DrmRights& drmRights,
    233             const String8& rightsPath, const String8& contentPath) {
    234     Mutex::Autolock _l(mLock);
    235     const String8 plugInId = getSupportedPlugInId(drmRights.getMimeType());
    236     status_t result = DRM_ERROR_UNKNOWN;
    237     if (EMPTY_STRING != plugInId) {
    238         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    239         result = rDrmEngine.saveRights(uniqueId, drmRights, rightsPath, contentPath);
    240     }
    241     return result;
    242 }
    243 
    244 String8 DrmManager::getOriginalMimeType(int uniqueId, const String8& path, int fd) {
    245     Mutex::Autolock _l(mLock);
    246     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
    247     if (EMPTY_STRING != plugInId) {
    248         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    249         return rDrmEngine.getOriginalMimeType(uniqueId, path, fd);
    250     }
    251     return EMPTY_STRING;
    252 }
    253 
    254 int DrmManager::getDrmObjectType(int uniqueId, const String8& path, const String8& mimeType) {
    255     Mutex::Autolock _l(mLock);
    256     const String8 plugInId = getSupportedPlugInId(uniqueId, path, mimeType);
    257     if (EMPTY_STRING != plugInId) {
    258         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    259         return rDrmEngine.getDrmObjectType(uniqueId, path, mimeType);
    260     }
    261     return DrmObjectType::UNKNOWN;
    262 }
    263 
    264 int DrmManager::checkRightsStatus(int uniqueId, const String8& path, int action) {
    265     Mutex::Autolock _l(mLock);
    266     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
    267     if (EMPTY_STRING != plugInId) {
    268         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    269         return rDrmEngine.checkRightsStatus(uniqueId, path, action);
    270     }
    271     return RightsStatus::RIGHTS_INVALID;
    272 }
    273 
    274 status_t DrmManager::consumeRights(
    275     int uniqueId, DecryptHandle* decryptHandle, int action, bool reserve) {
    276     status_t result = DRM_ERROR_UNKNOWN;
    277     Mutex::Autolock _l(mDecryptLock);
    278     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
    279         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
    280         result = drmEngine->consumeRights(uniqueId, decryptHandle, action, reserve);
    281     }
    282     return result;
    283 }
    284 
    285 status_t DrmManager::setPlaybackStatus(
    286     int uniqueId, DecryptHandle* decryptHandle, int playbackStatus, int64_t position) {
    287     status_t result = DRM_ERROR_UNKNOWN;
    288     Mutex::Autolock _l(mDecryptLock);
    289     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
    290         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
    291         result = drmEngine->setPlaybackStatus(uniqueId, decryptHandle, playbackStatus, position);
    292     }
    293     return result;
    294 }
    295 
    296 bool DrmManager::validateAction(
    297     int uniqueId, const String8& path, int action, const ActionDescription& description) {
    298     Mutex::Autolock _l(mLock);
    299     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
    300     if (EMPTY_STRING != plugInId) {
    301         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    302         return rDrmEngine.validateAction(uniqueId, path, action, description);
    303     }
    304     return false;
    305 }
    306 
    307 status_t DrmManager::removeRights(int uniqueId, const String8& path) {
    308     Mutex::Autolock _l(mLock);
    309     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
    310     status_t result = DRM_ERROR_UNKNOWN;
    311     if (EMPTY_STRING != plugInId) {
    312         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    313         result = rDrmEngine.removeRights(uniqueId, path);
    314     }
    315     return result;
    316 }
    317 
    318 status_t DrmManager::removeAllRights(int uniqueId) {
    319     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
    320     status_t result = DRM_ERROR_UNKNOWN;
    321     for (unsigned int index = 0; index < plugInIdList.size(); index++) {
    322         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInIdList.itemAt(index));
    323         result = rDrmEngine.removeAllRights(uniqueId);
    324         if (DRM_NO_ERROR != result) {
    325             break;
    326         }
    327     }
    328     return result;
    329 }
    330 
    331 int DrmManager::openConvertSession(int uniqueId, const String8& mimeType) {
    332     Mutex::Autolock _l(mConvertLock);
    333     int convertId = -1;
    334 
    335     const String8 plugInId = getSupportedPlugInId(mimeType);
    336     if (EMPTY_STRING != plugInId) {
    337         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    338 
    339         if (DRM_NO_ERROR == rDrmEngine.openConvertSession(uniqueId, mConvertId + 1)) {
    340             ++mConvertId;
    341             convertId = mConvertId;
    342             mConvertSessionMap.add(convertId, &rDrmEngine);
    343         }
    344     }
    345     return convertId;
    346 }
    347 
    348 DrmConvertedStatus* DrmManager::convertData(
    349             int uniqueId, int convertId, const DrmBuffer* inputData) {
    350     DrmConvertedStatus *drmConvertedStatus = NULL;
    351 
    352     Mutex::Autolock _l(mConvertLock);
    353     if (mConvertSessionMap.indexOfKey(convertId) != NAME_NOT_FOUND) {
    354         IDrmEngine* drmEngine = mConvertSessionMap.valueFor(convertId);
    355         drmConvertedStatus = drmEngine->convertData(uniqueId, convertId, inputData);
    356     }
    357     return drmConvertedStatus;
    358 }
    359 
    360 DrmConvertedStatus* DrmManager::closeConvertSession(int uniqueId, int convertId) {
    361     Mutex::Autolock _l(mConvertLock);
    362     DrmConvertedStatus *drmConvertedStatus = NULL;
    363 
    364     if (mConvertSessionMap.indexOfKey(convertId) != NAME_NOT_FOUND) {
    365         IDrmEngine* drmEngine = mConvertSessionMap.valueFor(convertId);
    366         drmConvertedStatus = drmEngine->closeConvertSession(uniqueId, convertId);
    367         mConvertSessionMap.removeItem(convertId);
    368     }
    369     return drmConvertedStatus;
    370 }
    371 
    372 status_t DrmManager::getAllSupportInfo(
    373                     int uniqueId, int* length, DrmSupportInfo** drmSupportInfoArray) {
    374     Mutex::Autolock _l(mLock);
    375     Vector<String8> plugInPathList = mPlugInManager.getPlugInIdList();
    376     int size = plugInPathList.size();
    377     int validPlugins = 0;
    378 
    379     if (0 < size) {
    380         Vector<DrmSupportInfo> drmSupportInfoList;
    381 
    382         for (int i = 0; i < size; ++i) {
    383             String8 plugInPath = plugInPathList[i];
    384             DrmSupportInfo* drmSupportInfo
    385                 = mPlugInManager.getPlugIn(plugInPath).getSupportInfo(0);
    386             if (NULL != drmSupportInfo) {
    387                 drmSupportInfoList.add(*drmSupportInfo);
    388                 delete drmSupportInfo; drmSupportInfo = NULL;
    389             }
    390         }
    391 
    392         validPlugins = drmSupportInfoList.size();
    393         if (0 < validPlugins) {
    394             *drmSupportInfoArray = new DrmSupportInfo[validPlugins];
    395             for (int i = 0; i < validPlugins; ++i) {
    396                 (*drmSupportInfoArray)[i] = drmSupportInfoList[i];
    397             }
    398         }
    399     }
    400     *length = validPlugins;
    401     return DRM_NO_ERROR;
    402 }
    403 
    404 DecryptHandle* DrmManager::openDecryptSession(
    405         int uniqueId, int fd, off64_t offset, off64_t length, const char* mime) {
    406 
    407     Mutex::Autolock _l(mDecryptLock);
    408     status_t result = DRM_ERROR_CANNOT_HANDLE;
    409     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
    410 
    411     DecryptHandle* handle = new DecryptHandle();
    412     if (NULL != handle) {
    413         handle->decryptId = mDecryptSessionId + 1;
    414 
    415         for (unsigned int index = 0; index < plugInIdList.size(); index++) {
    416             String8 plugInId = plugInIdList.itemAt(index);
    417             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    418             result = rDrmEngine.openDecryptSession(uniqueId, handle, fd, offset, length, mime);
    419 
    420             if (DRM_NO_ERROR == result) {
    421                 ++mDecryptSessionId;
    422                 mDecryptSessionMap.add(mDecryptSessionId, &rDrmEngine);
    423                 break;
    424             }
    425         }
    426     }
    427     if (DRM_NO_ERROR != result) {
    428         delete handle; handle = NULL;
    429     }
    430     return handle;
    431 }
    432 
    433 DecryptHandle* DrmManager::openDecryptSession(
    434         int uniqueId, const char* uri, const char* mime) {
    435     Mutex::Autolock _l(mDecryptLock);
    436     status_t result = DRM_ERROR_CANNOT_HANDLE;
    437     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
    438 
    439     DecryptHandle* handle = new DecryptHandle();
    440     if (NULL != handle) {
    441         handle->decryptId = mDecryptSessionId + 1;
    442 
    443         for (unsigned int index = 0; index < plugInIdList.size(); index++) {
    444             String8 plugInId = plugInIdList.itemAt(index);
    445             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    446             result = rDrmEngine.openDecryptSession(uniqueId, handle, uri, mime);
    447 
    448             if (DRM_NO_ERROR == result) {
    449                 ++mDecryptSessionId;
    450                 mDecryptSessionMap.add(mDecryptSessionId, &rDrmEngine);
    451                 break;
    452             }
    453         }
    454     }
    455     if (DRM_NO_ERROR != result) {
    456         delete handle; handle = NULL;
    457         ALOGV("DrmManager::openDecryptSession: no capable plug-in found");
    458     }
    459     return handle;
    460 }
    461 
    462 DecryptHandle* DrmManager::openDecryptSession(
    463         int uniqueId, const DrmBuffer& buf, const String8& mimeType) {
    464     Mutex::Autolock _l(mDecryptLock);
    465     status_t result = DRM_ERROR_CANNOT_HANDLE;
    466     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
    467 
    468     DecryptHandle* handle = new DecryptHandle();
    469     if (NULL != handle) {
    470         handle->decryptId = mDecryptSessionId + 1;
    471 
    472         for (size_t index = 0; index < plugInIdList.size(); index++) {
    473             String8 plugInId = plugInIdList.itemAt(index);
    474             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
    475             result = rDrmEngine.openDecryptSession(uniqueId, handle, buf, mimeType);
    476 
    477             if (DRM_NO_ERROR == result) {
    478                 ++mDecryptSessionId;
    479                 mDecryptSessionMap.add(mDecryptSessionId, &rDrmEngine);
    480                 break;
    481             }
    482         }
    483     }
    484     if (DRM_NO_ERROR != result) {
    485         delete handle;
    486         handle = NULL;
    487         ALOGV("DrmManager::openDecryptSession: no capable plug-in found");
    488     }
    489     return handle;
    490 }
    491 
    492 status_t DrmManager::closeDecryptSession(int uniqueId, DecryptHandle* decryptHandle) {
    493     Mutex::Autolock _l(mDecryptLock);
    494     status_t result = DRM_ERROR_UNKNOWN;
    495     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
    496         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
    497         result = drmEngine->closeDecryptSession(uniqueId, decryptHandle);
    498         if (DRM_NO_ERROR == result) {
    499             mDecryptSessionMap.removeItem(decryptHandle->decryptId);
    500         }
    501     }
    502     return result;
    503 }
    504 
    505 status_t DrmManager::initializeDecryptUnit(
    506     int uniqueId, DecryptHandle* decryptHandle, int decryptUnitId, const DrmBuffer* headerInfo) {
    507     status_t result = DRM_ERROR_UNKNOWN;
    508     Mutex::Autolock _l(mDecryptLock);
    509     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
    510         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
    511         result = drmEngine->initializeDecryptUnit(uniqueId, decryptHandle, decryptUnitId, headerInfo);
    512     }
    513     return result;
    514 }
    515 
    516 status_t DrmManager::decrypt(int uniqueId, DecryptHandle* decryptHandle, int decryptUnitId,
    517             const DrmBuffer* encBuffer, DrmBuffer** decBuffer, DrmBuffer* IV) {
    518     status_t result = DRM_ERROR_UNKNOWN;
    519 
    520     Mutex::Autolock _l(mDecryptLock);
    521     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
    522         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
    523         result = drmEngine->decrypt(
    524                 uniqueId, decryptHandle, decryptUnitId, encBuffer, decBuffer, IV);
    525     }
    526     return result;
    527 }
    528 
    529 status_t DrmManager::finalizeDecryptUnit(
    530             int uniqueId, DecryptHandle* decryptHandle, int decryptUnitId) {
    531     status_t result = DRM_ERROR_UNKNOWN;
    532     Mutex::Autolock _l(mDecryptLock);
    533     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
    534         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
    535         result = drmEngine->finalizeDecryptUnit(uniqueId, decryptHandle, decryptUnitId);
    536     }
    537     return result;
    538 }
    539 
    540 ssize_t DrmManager::pread(int uniqueId, DecryptHandle* decryptHandle,
    541             void* buffer, ssize_t numBytes, off64_t offset) {
    542     ssize_t result = DECRYPT_FILE_ERROR;
    543 
    544     Mutex::Autolock _l(mDecryptLock);
    545     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
    546         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
    547         result = drmEngine->pread(uniqueId, decryptHandle, buffer, numBytes, offset);
    548     }
    549     return result;
    550 }
    551 
    552 String8 DrmManager::getSupportedPlugInId(
    553             int uniqueId, const String8& path, const String8& mimeType) {
    554     String8 plugInId("");
    555 
    556     if (EMPTY_STRING != mimeType) {
    557         plugInId = getSupportedPlugInId(mimeType);
    558     } else {
    559         plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
    560     }
    561     return plugInId;
    562 }
    563 
    564 String8 DrmManager::getSupportedPlugInId(const String8& mimeType) {
    565     String8 plugInId("");
    566 
    567     if (EMPTY_STRING != mimeType) {
    568         for (unsigned int index = 0; index < mSupportInfoToPlugInIdMap.size(); index++) {
    569             const DrmSupportInfo& drmSupportInfo = mSupportInfoToPlugInIdMap.keyAt(index);
    570 
    571             if (drmSupportInfo.isSupportedMimeType(mimeType)) {
    572                 plugInId = mSupportInfoToPlugInIdMap.valueFor(drmSupportInfo);
    573                 break;
    574             }
    575         }
    576     }
    577     return plugInId;
    578 }
    579 
    580 String8 DrmManager::getSupportedPlugInIdFromPath(int uniqueId, const String8& path) {
    581     String8 plugInId("");
    582     const String8 fileSuffix = path.getPathExtension();
    583 
    584     for (unsigned int index = 0; index < mSupportInfoToPlugInIdMap.size(); index++) {
    585         const DrmSupportInfo& drmSupportInfo = mSupportInfoToPlugInIdMap.keyAt(index);
    586 
    587         if (drmSupportInfo.isSupportedFileSuffix(fileSuffix)) {
    588             String8 key = mSupportInfoToPlugInIdMap.valueFor(drmSupportInfo);
    589             IDrmEngine& drmEngine = mPlugInManager.getPlugIn(key);
    590 
    591             if (drmEngine.canHandle(uniqueId, path)) {
    592                 plugInId = key;
    593                 break;
    594             }
    595         }
    596     }
    597     return plugInId;
    598 }
    599 
    600 void DrmManager::onInfo(const DrmInfoEvent& event) {
    601     Mutex::Autolock _l(mListenerLock);
    602     for (unsigned int index = 0; index < mServiceListeners.size(); index++) {
    603         int uniqueId = mServiceListeners.keyAt(index);
    604 
    605         if (uniqueId == event.getUniqueId()) {
    606             sp<IDrmServiceListener> serviceListener = mServiceListeners.valueFor(uniqueId);
    607             serviceListener->notify(event);
    608         }
    609     }
    610 }
    611 
    612