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 
20 #include <cutils/properties.h>
21 #include <utils/String8.h>
22 #include <utils/Log.h>
23 
24 #include <binder/IPCThreadState.h>
25 #include <drm/DrmInfo.h>
26 
27 #include <drm/DrmInfoEvent.h>
28 #include <drm/DrmRights.h>
29 #include <drm/DrmConstraints.h>
30 #include <drm/DrmMetadata.h>
31 #include <drm/DrmInfoStatus.h>
32 #include <drm/DrmInfoRequest.h>
33 #include <drm/DrmSupportInfo.h>
34 #include <drm/DrmConvertedStatus.h>
35 #include <media/MediaMetricsItem.h>
36 #include <IDrmEngine.h>
37 
38 #include "DrmManager.h"
39 #include "ReadWriteUtils.h"
40 
41 #include <algorithm>
42 
43 #define DECRYPT_FILE_ERROR (-1)
44 
45 using namespace android;
46 
47 const String8 DrmManager::EMPTY_STRING("");
48 
49 const std::map<const char*, size_t> DrmManager::kMethodIdMap {
50     {"getConstraints"     , DrmManagerMethodId::GET_CONSTRAINTS       },
51     {"getMetadata"        , DrmManagerMethodId::GET_METADATA          },
52     {"canHandle"          , DrmManagerMethodId::CAN_HANDLE            },
53     {"processDrmInfo"     , DrmManagerMethodId::PROCESS_DRM_INFO      },
54     {"acquireDrmInfo"     , DrmManagerMethodId::ACQUIRE_DRM_INFO      },
55     {"saveRights"         , DrmManagerMethodId::SAVE_RIGHTS           },
56     {"getOriginalMimeType", DrmManagerMethodId::GET_ORIGINAL_MIME_TYPE},
57     {"getDrmObjectType"   , DrmManagerMethodId::GET_DRM_OBJECT_TYPE   },
58     {"checkRightsStatus"  , DrmManagerMethodId::CHECK_RIGHTS_STATUS   },
59     {"removeRights"       , DrmManagerMethodId::REMOVE_RIGHTS         },
60     {"removeAllRights"    , DrmManagerMethodId::REMOVE_ALL_RIGHTS     },
61     {"openConvertSession" , DrmManagerMethodId::OPEN_CONVERT_SESSION  },
62     {"openDecryptSession" , DrmManagerMethodId::OPEN_DECRYPT_SESSION  }
63 };
64 
DrmManager()65 DrmManager::DrmManager() :
66     mDecryptSessionId(0),
67     mConvertId(0) {
68     srand(time(NULL));
69     memset(mUniqueIdArray, 0, sizeof(bool) * kMaxNumUniqueIds);
70 }
71 
~DrmManager()72 DrmManager::~DrmManager() {
73     if (mMetricsLooper != NULL) {
74         mMetricsLooper->stop();
75     }
76     flushEngineMetrics();
77 }
78 
initMetricsLooper()79 void DrmManager::initMetricsLooper() {
80     if (mMetricsLooper != NULL) {
81         return;
82     }
83     mMetricsLooper = new ALooper;
84     mMetricsLooper->setName("DrmManagerMetricsLooper");
85     mMetricsLooper->start();
86     mMetricsLooper->registerHandler(this);
87 
88     sp<AMessage> msg = new AMessage(kWhatFlushMetrics, this);
89     msg->post(getMetricsFlushPeriodUs());
90 }
91 
onMessageReceived(const sp<AMessage> & msg)92 void DrmManager::onMessageReceived(const sp<AMessage> &msg) {
93     switch (msg->what()) {
94         case kWhatFlushMetrics:
95         {
96             flushEngineMetrics();
97             msg->post(getMetricsFlushPeriodUs());
98             break;
99         }
100         default:
101         {
102             ALOGW("Unrecognized message type: %u", msg->what());
103         }
104     }
105 }
106 
getMetricsFlushPeriodUs()107 int64_t DrmManager::getMetricsFlushPeriodUs() {
108     return 1000 * 1000 * std::max(1ll, (long long)property_get_int64("drmmanager.metrics.period", 86400));
109 }
110 
recordEngineMetrics(const char func[],const String8 & plugInId8,const String8 & mimeType)111 void DrmManager::recordEngineMetrics(
112         const char func[], const String8& plugInId8, const String8& mimeType) {
113     IDrmEngine& engine = mPlugInManager.getPlugIn(plugInId8);
114     std::unique_ptr<DrmSupportInfo> info(engine.getSupportInfo(0));
115 
116     uid_t callingUid = IPCThreadState::self()->getCallingUid();
117     std::string plugInId(plugInId8.getPathLeaf().getBasePath().c_str());
118     ALOGV("%d calling %s %s", callingUid, plugInId.c_str(), func);
119 
120     Mutex::Autolock _l(mMetricsLock);
121     auto& metrics = mPluginMetrics[std::make_pair(callingUid, plugInId)];
122     if (metrics.mPluginId.empty()) {
123         metrics.mPluginId = plugInId;
124         metrics.mCallingUid = callingUid;
125         if (NULL != info) {
126             metrics.mDescription = info->getDescription().c_str();
127         }
128     }
129 
130     if (!mimeType.isEmpty()) {
131         metrics.mMimeTypes.insert(mimeType.c_str());
132     } else if (NULL != info) {
133         DrmSupportInfo::MimeTypeIterator mimeIter = info->getMimeTypeIterator();
134         while (mimeIter.hasNext()) {
135             metrics.mMimeTypes.insert(mimeIter.next().c_str());
136         }
137     }
138 
139     size_t methodId = kMethodIdMap.at(func);
140     if (methodId < metrics.mMethodCounts.size()) {
141         metrics.mMethodCounts[methodId]++;
142     }
143 }
144 
flushEngineMetrics()145 void DrmManager::flushEngineMetrics() {
146     using namespace std::string_literals;
147     Mutex::Autolock _l(mMetricsLock);
148     for (auto kv : mPluginMetrics) {
149         DrmManagerMetrics& metrics = kv.second;
150         std::unique_ptr<mediametrics::Item> item(mediametrics::Item::create("drmmanager"));
151         item->setUid(metrics.mCallingUid);
152         item->setCString("plugin_id", metrics.mPluginId.c_str());
153         item->setCString("description", metrics.mDescription.c_str());
154 
155         std::vector<std::string> mimeTypes(metrics.mMimeTypes.begin(), metrics.mMimeTypes.end());
156         std::string mimeTypesStr(mimeTypes.empty() ? "" : mimeTypes[0]);
157         for (size_t i = 1; i < mimeTypes.size() ; i++) {
158             mimeTypesStr.append(",").append(mimeTypes[i]);
159         }
160         item->setCString("mime_types", mimeTypesStr.c_str());
161 
162         for (size_t i = 0; i < metrics.mMethodCounts.size() ; i++) {
163             item->setInt64(("method"s + std::to_string(i)).c_str(), metrics.mMethodCounts[i]);
164         }
165 
166         if (!item->selfrecord()) {
167             ALOGE("Failed to record metrics");
168         }
169     }
170     mPluginMetrics.clear();
171 }
172 
addUniqueId(bool isNative)173 int DrmManager::addUniqueId(bool isNative) {
174     Mutex::Autolock _l(mLock);
175 
176     int uniqueId = -1;
177     int random = rand();
178 
179     for (size_t index = 0; index < kMaxNumUniqueIds; ++index) {
180         int temp = (random + index) % kMaxNumUniqueIds;
181         if (!mUniqueIdArray[temp]) {
182             uniqueId = temp;
183             mUniqueIdArray[uniqueId] = true;
184 
185             if (isNative) {
186                 // set a flag to differentiate DrmManagerClient
187                 // created from native side and java side
188                 uniqueId |= 0x1000;
189             }
190             break;
191         }
192     }
193 
194     // -1 indicates that no unique id can be allocated.
195     return uniqueId;
196 }
197 
removeUniqueId(int uniqueId)198 void DrmManager::removeUniqueId(int uniqueId) {
199     Mutex::Autolock _l(mLock);
200     if (uniqueId & 0x1000) {
201         // clear the flag for the native side.
202         uniqueId &= ~(0x1000);
203     }
204 
205     if (uniqueId >= 0 && uniqueId < kMaxNumUniqueIds) {
206         mUniqueIdArray[uniqueId] = false;
207     }
208 }
209 
loadPlugIns()210 status_t DrmManager::loadPlugIns() {
211 #if __LP64__
212     String8 pluginDirPath("/system/lib64/drm");
213 #else
214     String8 pluginDirPath("/system/lib/drm");
215 #endif
216     loadPlugIns(pluginDirPath);
217     return DRM_NO_ERROR;
218 }
219 
loadPlugIns(const String8 & plugInDirPath)220 status_t DrmManager::loadPlugIns(const String8& plugInDirPath) {
221     mPlugInManager.loadPlugIns(plugInDirPath);
222     Vector<String8> plugInPathList = mPlugInManager.getPlugInIdList();
223     for (size_t i = 0; i < plugInPathList.size(); ++i) {
224         String8 plugInPath = plugInPathList[i];
225         DrmSupportInfo* info = mPlugInManager.getPlugIn(plugInPath).getSupportInfo(0);
226         if (NULL != info) {
227             if (mSupportInfoToPlugInIdMap.indexOfKey(*info) < 0) {
228                 mSupportInfoToPlugInIdMap.add(*info, plugInPath);
229             }
230             delete info;
231         }
232     }
233     return DRM_NO_ERROR;
234 }
235 
unloadPlugIns()236 status_t DrmManager::unloadPlugIns() {
237     Mutex::Autolock _l(mLock);
238     mConvertSessionMap.clear();
239     mDecryptSessionMap.clear();
240     mPlugInManager.unloadPlugIns();
241     mSupportInfoToPlugInIdMap.clear();
242     return DRM_NO_ERROR;
243 }
244 
setDrmServiceListener(int uniqueId,const sp<IDrmServiceListener> & drmServiceListener)245 status_t DrmManager::setDrmServiceListener(
246             int uniqueId, const sp<IDrmServiceListener>& drmServiceListener) {
247     Mutex::Autolock _l(mListenerLock);
248     if (NULL != drmServiceListener.get()) {
249         mServiceListeners.add(uniqueId, drmServiceListener);
250     } else {
251         mServiceListeners.removeItem(uniqueId);
252     }
253     return DRM_NO_ERROR;
254 }
255 
addClient(int uniqueId)256 void DrmManager::addClient(int uniqueId) {
257     Mutex::Autolock _l(mLock);
258     if (!mSupportInfoToPlugInIdMap.isEmpty()) {
259         Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
260         for (size_t index = 0; index < plugInIdList.size(); index++) {
261             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInIdList.itemAt(index));
262             rDrmEngine.initialize(uniqueId);
263             rDrmEngine.setOnInfoListener(uniqueId, this);
264         }
265     }
266 }
267 
removeClient(int uniqueId)268 void DrmManager::removeClient(int uniqueId) {
269     Mutex::Autolock _l(mLock);
270     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
271     for (size_t index = 0; index < plugInIdList.size(); index++) {
272         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInIdList.itemAt(index));
273         rDrmEngine.terminate(uniqueId);
274     }
275 }
276 
getConstraints(int uniqueId,const String8 * path,const int action)277 DrmConstraints* DrmManager::getConstraints(int uniqueId, const String8* path, const int action) {
278     Mutex::Autolock _l(mLock);
279     DrmConstraints *constraints = NULL;
280     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, *path);
281     if (EMPTY_STRING != plugInId) {
282         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
283         constraints = rDrmEngine.getConstraints(uniqueId, path, action);
284     }
285     if (NULL != constraints) {
286         recordEngineMetrics(__func__, plugInId);
287     }
288     return constraints;
289 }
290 
getMetadata(int uniqueId,const String8 * path)291 DrmMetadata* DrmManager::getMetadata(int uniqueId, const String8* path) {
292     Mutex::Autolock _l(mLock);
293     DrmMetadata *meta = NULL;
294     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, *path);
295     if (EMPTY_STRING != plugInId) {
296         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
297         meta = rDrmEngine.getMetadata(uniqueId, path);
298     }
299     if (NULL != meta) {
300         recordEngineMetrics(__func__, plugInId);
301     }
302     return meta;
303 }
304 
canHandle(int uniqueId,const String8 & path,const String8 & mimeType)305 bool DrmManager::canHandle(int uniqueId, const String8& path, const String8& mimeType) {
306     Mutex::Autolock _l(mLock);
307     const String8 plugInId = getSupportedPlugInId(mimeType);
308     bool result = (EMPTY_STRING != plugInId) ? true : false;
309 
310     if (result) {
311         recordEngineMetrics(__func__, plugInId, mimeType);
312     }
313 
314     if (0 < path.length()) {
315         if (result) {
316             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
317             result = rDrmEngine.canHandle(uniqueId, path);
318         } else {
319             String8 extension = path.getPathExtension();
320             if (String8("") != extension) {
321                 result = canHandle(uniqueId, path);
322             }
323         }
324     }
325     return result;
326 }
327 
processDrmInfo(int uniqueId,const DrmInfo * drmInfo)328 DrmInfoStatus* DrmManager::processDrmInfo(int uniqueId, const DrmInfo* drmInfo) {
329     Mutex::Autolock _l(mLock);
330     DrmInfoStatus *infoStatus = NULL;
331     const String8 mimeType = drmInfo->getMimeType();
332     const String8 plugInId = getSupportedPlugInId(mimeType);
333     if (EMPTY_STRING != plugInId) {
334         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
335         infoStatus = rDrmEngine.processDrmInfo(uniqueId, drmInfo);
336     }
337     if (NULL != infoStatus) {
338         recordEngineMetrics(__func__, plugInId, mimeType);
339     }
340     return infoStatus;
341 }
342 
canHandle(int uniqueId,const String8 & path)343 bool DrmManager::canHandle(int uniqueId, const String8& path) {
344     bool result = false;
345     Vector<String8> plugInPathList = mPlugInManager.getPlugInIdList();
346 
347     for (size_t i = 0; i < plugInPathList.size(); ++i) {
348         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInPathList[i]);
349         result = rDrmEngine.canHandle(uniqueId, path);
350 
351         if (result) {
352             recordEngineMetrics(__func__, plugInPathList[i]);
353             break;
354         }
355     }
356     return result;
357 }
358 
acquireDrmInfo(int uniqueId,const DrmInfoRequest * drmInfoRequest)359 DrmInfo* DrmManager::acquireDrmInfo(int uniqueId, const DrmInfoRequest* drmInfoRequest) {
360     Mutex::Autolock _l(mLock);
361     DrmInfo *info = NULL;
362     const String8 mimeType = drmInfoRequest->getMimeType();
363     const String8 plugInId = getSupportedPlugInId(mimeType);
364     if (EMPTY_STRING != plugInId) {
365         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
366         info = rDrmEngine.acquireDrmInfo(uniqueId, drmInfoRequest);
367     }
368     if (NULL != info) {
369         recordEngineMetrics(__func__, plugInId, mimeType);
370     }
371     return info;
372 }
373 
saveRights(int uniqueId,const DrmRights & drmRights,const String8 & rightsPath,const String8 & contentPath)374 status_t DrmManager::saveRights(int uniqueId, const DrmRights& drmRights,
375             const String8& rightsPath, const String8& contentPath) {
376     Mutex::Autolock _l(mLock);
377     const String8 mimeType = drmRights.getMimeType();
378     const String8 plugInId = getSupportedPlugInId(mimeType);
379     status_t result = DRM_ERROR_UNKNOWN;
380     if (EMPTY_STRING != plugInId) {
381         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
382         result = rDrmEngine.saveRights(uniqueId, drmRights, rightsPath, contentPath);
383     }
384     if (DRM_NO_ERROR == result) {
385         recordEngineMetrics(__func__, plugInId, mimeType);
386     }
387     return result;
388 }
389 
getOriginalMimeType(int uniqueId,const String8 & path,int fd)390 String8 DrmManager::getOriginalMimeType(int uniqueId, const String8& path, int fd) {
391     Mutex::Autolock _l(mLock);
392     String8 mimeType(EMPTY_STRING);
393     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
394     if (EMPTY_STRING != plugInId) {
395         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
396         mimeType = rDrmEngine.getOriginalMimeType(uniqueId, path, fd);
397     }
398     if (!mimeType.isEmpty()) {
399         recordEngineMetrics(__func__, plugInId, mimeType);
400     }
401     return mimeType;
402 }
403 
getDrmObjectType(int uniqueId,const String8 & path,const String8 & mimeType)404 int DrmManager::getDrmObjectType(int uniqueId, const String8& path, const String8& mimeType) {
405     Mutex::Autolock _l(mLock);
406     int type = DrmObjectType::UNKNOWN;
407     const String8 plugInId = getSupportedPlugInId(uniqueId, path, mimeType);
408     if (EMPTY_STRING != plugInId) {
409         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
410         type = rDrmEngine.getDrmObjectType(uniqueId, path, mimeType);
411     }
412     if (DrmObjectType::UNKNOWN != type) {
413         recordEngineMetrics(__func__, plugInId, mimeType);
414     }
415     return type;
416 }
417 
checkRightsStatus(int uniqueId,const String8 & path,int action)418 int DrmManager::checkRightsStatus(int uniqueId, const String8& path, int action) {
419     Mutex::Autolock _l(mLock);
420     int rightsStatus = RightsStatus::RIGHTS_INVALID;
421     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
422     if (EMPTY_STRING != plugInId) {
423         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
424         rightsStatus = rDrmEngine.checkRightsStatus(uniqueId, path, action);
425     }
426     if (RightsStatus::RIGHTS_INVALID != rightsStatus) {
427         recordEngineMetrics(__func__, plugInId);
428     }
429     return rightsStatus;
430 }
431 
consumeRights(int uniqueId,sp<DecryptHandle> & decryptHandle,int action,bool reserve)432 status_t DrmManager::consumeRights(
433     int uniqueId, sp<DecryptHandle>& decryptHandle, int action, bool reserve) {
434     status_t result = DRM_ERROR_UNKNOWN;
435     Mutex::Autolock _l(mDecryptLock);
436     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
437         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
438         result = drmEngine->consumeRights(uniqueId, decryptHandle, action, reserve);
439     }
440     return result;
441 }
442 
setPlaybackStatus(int uniqueId,sp<DecryptHandle> & decryptHandle,int playbackStatus,int64_t position)443 status_t DrmManager::setPlaybackStatus(
444     int uniqueId, sp<DecryptHandle>& decryptHandle, int playbackStatus, int64_t position) {
445     status_t result = DRM_ERROR_UNKNOWN;
446     Mutex::Autolock _l(mDecryptLock);
447     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
448         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
449         result = drmEngine->setPlaybackStatus(uniqueId, decryptHandle, playbackStatus, position);
450     }
451     return result;
452 }
453 
validateAction(int uniqueId,const String8 & path,int action,const ActionDescription & description)454 bool DrmManager::validateAction(
455     int uniqueId, const String8& path, int action, const ActionDescription& description) {
456     Mutex::Autolock _l(mLock);
457     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
458     if (EMPTY_STRING != plugInId) {
459         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
460         return rDrmEngine.validateAction(uniqueId, path, action, description);
461     }
462     return false;
463 }
464 
removeRights(int uniqueId,const String8 & path)465 status_t DrmManager::removeRights(int uniqueId, const String8& path) {
466     Mutex::Autolock _l(mLock);
467     const String8 plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
468     status_t result = DRM_ERROR_UNKNOWN;
469     if (EMPTY_STRING != plugInId) {
470         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
471         result = rDrmEngine.removeRights(uniqueId, path);
472     }
473     if (DRM_NO_ERROR == result) {
474         recordEngineMetrics(__func__, plugInId);
475     }
476     return result;
477 }
478 
removeAllRights(int uniqueId)479 status_t DrmManager::removeAllRights(int uniqueId) {
480     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
481     status_t result = DRM_ERROR_UNKNOWN;
482     for (size_t index = 0; index < plugInIdList.size(); index++) {
483         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInIdList.itemAt(index));
484         result = rDrmEngine.removeAllRights(uniqueId);
485         if (DRM_NO_ERROR != result) {
486             break;
487         }
488         recordEngineMetrics(__func__, plugInIdList[index]);
489     }
490     return result;
491 }
492 
openConvertSession(int uniqueId,const String8 & mimeType)493 int DrmManager::openConvertSession(int uniqueId, const String8& mimeType) {
494     Mutex::Autolock _l(mConvertLock);
495     int convertId = -1;
496 
497     const String8 plugInId = getSupportedPlugInId(mimeType);
498     if (EMPTY_STRING != plugInId) {
499         IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
500 
501         if (DRM_NO_ERROR == rDrmEngine.openConvertSession(uniqueId, mConvertId + 1)) {
502             ++mConvertId;
503             convertId = mConvertId;
504             mConvertSessionMap.add(convertId, &rDrmEngine);
505             recordEngineMetrics(__func__, plugInId, mimeType);
506         }
507     }
508     return convertId;
509 }
510 
convertData(int uniqueId,int convertId,const DrmBuffer * inputData)511 DrmConvertedStatus* DrmManager::convertData(
512             int uniqueId, int convertId, const DrmBuffer* inputData) {
513     DrmConvertedStatus *drmConvertedStatus = NULL;
514 
515     Mutex::Autolock _l(mConvertLock);
516     if (mConvertSessionMap.indexOfKey(convertId) != NAME_NOT_FOUND) {
517         IDrmEngine* drmEngine = mConvertSessionMap.valueFor(convertId);
518         drmConvertedStatus = drmEngine->convertData(uniqueId, convertId, inputData);
519     }
520     return drmConvertedStatus;
521 }
522 
closeConvertSession(int uniqueId,int convertId)523 DrmConvertedStatus* DrmManager::closeConvertSession(int uniqueId, int convertId) {
524     Mutex::Autolock _l(mConvertLock);
525     DrmConvertedStatus *drmConvertedStatus = NULL;
526 
527     if (mConvertSessionMap.indexOfKey(convertId) != NAME_NOT_FOUND) {
528         IDrmEngine* drmEngine = mConvertSessionMap.valueFor(convertId);
529         drmConvertedStatus = drmEngine->closeConvertSession(uniqueId, convertId);
530         mConvertSessionMap.removeItem(convertId);
531     }
532     return drmConvertedStatus;
533 }
534 
getAllSupportInfo(int,int * length,DrmSupportInfo ** drmSupportInfoArray)535 status_t DrmManager::getAllSupportInfo(
536                     int /* uniqueId */, int* length, DrmSupportInfo** drmSupportInfoArray) {
537     Mutex::Autolock _l(mLock);
538     Vector<String8> plugInPathList = mPlugInManager.getPlugInIdList();
539     int size = plugInPathList.size();
540     int validPlugins = 0;
541 
542     if (0 < size) {
543         Vector<DrmSupportInfo> drmSupportInfoList;
544 
545         for (int i = 0; i < size; ++i) {
546             String8 plugInPath = plugInPathList[i];
547             DrmSupportInfo* drmSupportInfo
548                 = mPlugInManager.getPlugIn(plugInPath).getSupportInfo(0);
549             if (NULL != drmSupportInfo) {
550                 drmSupportInfoList.add(*drmSupportInfo);
551                 delete drmSupportInfo; drmSupportInfo = NULL;
552             }
553         }
554 
555         validPlugins = drmSupportInfoList.size();
556         if (0 < validPlugins) {
557             *drmSupportInfoArray = new DrmSupportInfo[validPlugins];
558             for (int i = 0; i < validPlugins; ++i) {
559                 (*drmSupportInfoArray)[i] = drmSupportInfoList[i];
560             }
561         }
562     }
563     *length = validPlugins;
564     return DRM_NO_ERROR;
565 }
566 
openDecryptSession(int uniqueId,int fd,off64_t offset,off64_t length,const char * mime)567 sp<DecryptHandle> DrmManager::openDecryptSession(
568         int uniqueId, int fd, off64_t offset, off64_t length, const char* mime) {
569 
570     Mutex::Autolock _l(mDecryptLock);
571     status_t result = DRM_ERROR_CANNOT_HANDLE;
572     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
573 
574     sp<DecryptHandle> handle = new DecryptHandle();
575     if (NULL != handle.get()) {
576         handle->decryptId = mDecryptSessionId + 1;
577 
578         for (size_t index = 0; index < plugInIdList.size(); index++) {
579             const String8& plugInId = plugInIdList.itemAt(index);
580             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
581             result = rDrmEngine.openDecryptSession(uniqueId, handle, fd, offset, length, mime);
582 
583             if (DRM_NO_ERROR == result) {
584                 ++mDecryptSessionId;
585                 mDecryptSessionMap.add(mDecryptSessionId, &rDrmEngine);
586                 recordEngineMetrics(__func__, plugInId, String8(mime));
587                 break;
588             }
589         }
590     }
591     if (DRM_NO_ERROR != result) {
592         handle.clear();
593     }
594     return handle;
595 }
596 
openDecryptSession(int uniqueId,const char * uri,const char * mime)597 sp<DecryptHandle> DrmManager::openDecryptSession(
598         int uniqueId, const char* uri, const char* mime) {
599     Mutex::Autolock _l(mDecryptLock);
600     status_t result = DRM_ERROR_CANNOT_HANDLE;
601     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
602 
603     sp<DecryptHandle> handle = new DecryptHandle();
604     if (NULL != handle.get()) {
605         handle->decryptId = mDecryptSessionId + 1;
606 
607         for (size_t index = 0; index < plugInIdList.size(); index++) {
608             const String8& plugInId = plugInIdList.itemAt(index);
609             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
610             result = rDrmEngine.openDecryptSession(uniqueId, handle, uri, mime);
611 
612             if (DRM_NO_ERROR == result) {
613                 ++mDecryptSessionId;
614                 mDecryptSessionMap.add(mDecryptSessionId, &rDrmEngine);
615                 recordEngineMetrics(__func__, plugInId, String8(mime));
616                 break;
617             }
618         }
619     }
620     if (DRM_NO_ERROR != result) {
621         handle.clear();
622         ALOGV("DrmManager::openDecryptSession: no capable plug-in found");
623     }
624     return handle;
625 }
626 
openDecryptSession(int uniqueId,const DrmBuffer & buf,const String8 & mimeType)627 sp<DecryptHandle> DrmManager::openDecryptSession(
628         int uniqueId, const DrmBuffer& buf, const String8& mimeType) {
629     Mutex::Autolock _l(mDecryptLock);
630     status_t result = DRM_ERROR_CANNOT_HANDLE;
631     Vector<String8> plugInIdList = mPlugInManager.getPlugInIdList();
632 
633     sp<DecryptHandle> handle = new DecryptHandle();
634     if (NULL != handle.get()) {
635         handle->decryptId = mDecryptSessionId + 1;
636 
637         for (size_t index = 0; index < plugInIdList.size(); index++) {
638             const String8& plugInId = plugInIdList.itemAt(index);
639             IDrmEngine& rDrmEngine = mPlugInManager.getPlugIn(plugInId);
640             result = rDrmEngine.openDecryptSession(uniqueId, handle, buf, mimeType);
641 
642             if (DRM_NO_ERROR == result) {
643                 ++mDecryptSessionId;
644                 mDecryptSessionMap.add(mDecryptSessionId, &rDrmEngine);
645                 recordEngineMetrics(__func__, plugInId, mimeType);
646                 break;
647             }
648         }
649     }
650     if (DRM_NO_ERROR != result) {
651         handle.clear();
652         ALOGV("DrmManager::openDecryptSession: no capable plug-in found");
653     }
654     return handle;
655 }
656 
closeDecryptSession(int uniqueId,sp<DecryptHandle> & decryptHandle)657 status_t DrmManager::closeDecryptSession(int uniqueId, sp<DecryptHandle>& decryptHandle) {
658     Mutex::Autolock _l(mDecryptLock);
659     status_t result = DRM_ERROR_UNKNOWN;
660     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
661         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
662         result = drmEngine->closeDecryptSession(uniqueId, decryptHandle);
663         if (DRM_NO_ERROR == result && NULL != decryptHandle.get()) {
664             mDecryptSessionMap.removeItem(decryptHandle->decryptId);
665         }
666     }
667     return result;
668 }
669 
initializeDecryptUnit(int uniqueId,sp<DecryptHandle> & decryptHandle,int decryptUnitId,const DrmBuffer * headerInfo)670 status_t DrmManager::initializeDecryptUnit(
671         int uniqueId, sp<DecryptHandle>& decryptHandle, int decryptUnitId,
672         const DrmBuffer* headerInfo) {
673     status_t result = DRM_ERROR_UNKNOWN;
674     Mutex::Autolock _l(mDecryptLock);
675     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
676         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
677         result = drmEngine->initializeDecryptUnit(uniqueId, decryptHandle, decryptUnitId, headerInfo);
678     }
679     return result;
680 }
681 
decrypt(int uniqueId,sp<DecryptHandle> & decryptHandle,int decryptUnitId,const DrmBuffer * encBuffer,DrmBuffer ** decBuffer,DrmBuffer * IV)682 status_t DrmManager::decrypt(int uniqueId, sp<DecryptHandle>& decryptHandle, int decryptUnitId,
683             const DrmBuffer* encBuffer, DrmBuffer** decBuffer, DrmBuffer* IV) {
684     status_t result = DRM_ERROR_UNKNOWN;
685 
686     Mutex::Autolock _l(mDecryptLock);
687     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
688         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
689         result = drmEngine->decrypt(
690                 uniqueId, decryptHandle, decryptUnitId, encBuffer, decBuffer, IV);
691     }
692     return result;
693 }
694 
finalizeDecryptUnit(int uniqueId,sp<DecryptHandle> & decryptHandle,int decryptUnitId)695 status_t DrmManager::finalizeDecryptUnit(
696             int uniqueId, sp<DecryptHandle>& decryptHandle, int decryptUnitId) {
697     status_t result = DRM_ERROR_UNKNOWN;
698     Mutex::Autolock _l(mDecryptLock);
699     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
700         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
701         result = drmEngine->finalizeDecryptUnit(uniqueId, decryptHandle, decryptUnitId);
702     }
703     return result;
704 }
705 
pread(int uniqueId,sp<DecryptHandle> & decryptHandle,void * buffer,ssize_t numBytes,off64_t offset)706 ssize_t DrmManager::pread(int uniqueId, sp<DecryptHandle>& decryptHandle,
707             void* buffer, ssize_t numBytes, off64_t offset) {
708     ssize_t result = DECRYPT_FILE_ERROR;
709 
710     Mutex::Autolock _l(mDecryptLock);
711     if (mDecryptSessionMap.indexOfKey(decryptHandle->decryptId) != NAME_NOT_FOUND) {
712         IDrmEngine* drmEngine = mDecryptSessionMap.valueFor(decryptHandle->decryptId);
713         result = drmEngine->pread(uniqueId, decryptHandle, buffer, numBytes, offset);
714     }
715     return result;
716 }
717 
getSupportedPlugInId(int uniqueId,const String8 & path,const String8 & mimeType)718 String8 DrmManager::getSupportedPlugInId(
719             int uniqueId, const String8& path, const String8& mimeType) {
720     String8 plugInId("");
721 
722     if (EMPTY_STRING != mimeType) {
723         plugInId = getSupportedPlugInId(mimeType);
724     } else {
725         plugInId = getSupportedPlugInIdFromPath(uniqueId, path);
726     }
727     return plugInId;
728 }
729 
getSupportedPlugInId(const String8 & mimeType)730 String8 DrmManager::getSupportedPlugInId(const String8& mimeType) {
731     String8 plugInId("");
732 
733     if (EMPTY_STRING != mimeType) {
734         for (size_t index = 0; index < mSupportInfoToPlugInIdMap.size(); index++) {
735             const DrmSupportInfo& drmSupportInfo = mSupportInfoToPlugInIdMap.keyAt(index);
736 
737             if (drmSupportInfo.isSupportedMimeType(mimeType)) {
738                 plugInId = mSupportInfoToPlugInIdMap.valueFor(drmSupportInfo);
739                 break;
740             }
741         }
742     }
743     return plugInId;
744 }
745 
getSupportedPlugInIdFromPath(int uniqueId,const String8 & path)746 String8 DrmManager::getSupportedPlugInIdFromPath(int uniqueId, const String8& path) {
747     String8 plugInId("");
748     const String8 fileSuffix = path.getPathExtension();
749 
750     for (size_t index = 0; index < mSupportInfoToPlugInIdMap.size(); index++) {
751         const DrmSupportInfo& drmSupportInfo = mSupportInfoToPlugInIdMap.keyAt(index);
752 
753         if (drmSupportInfo.isSupportedFileSuffix(fileSuffix)) {
754             String8 key = mSupportInfoToPlugInIdMap.valueFor(drmSupportInfo);
755             IDrmEngine& drmEngine = mPlugInManager.getPlugIn(key);
756 
757             if (drmEngine.canHandle(uniqueId, path)) {
758                 plugInId = key;
759                 break;
760             }
761         }
762     }
763     return plugInId;
764 }
765 
onInfo(const DrmInfoEvent & event)766 void DrmManager::onInfo(const DrmInfoEvent& event) {
767     Mutex::Autolock _l(mListenerLock);
768     for (size_t index = 0; index < mServiceListeners.size(); index++) {
769         int uniqueId = mServiceListeners.keyAt(index);
770 
771         if (uniqueId == event.getUniqueId()) {
772             sp<IDrmServiceListener> serviceListener = mServiceListeners.valueFor(uniqueId);
773             serviceListener->notify(event);
774         }
775     }
776 }
777 
778