1 /*
2  * Copyright (C) 2017 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 "NuPlayerDrm"
19 
20 #include "NuPlayerDrm.h"
21 
22 #include <binder/IServiceManager.h>
23 #include <mediadrm/IMediaDrmService.h>
24 #include <utils/Log.h>
25 
26 
27 namespace android {
28 
29 // static helpers - internal
30 
CreateDrm(status_t * pstatus)31 sp<IDrm> NuPlayerDrm::CreateDrm(status_t *pstatus)
32 {
33     status_t &status = *pstatus;
34     sp<IServiceManager> sm = defaultServiceManager();
35     sp<IBinder> binder = sm->getService(String16("media.drm"));
36     ALOGV("CreateDrm binder %p", (binder != NULL ? binder.get() : 0));
37 
38     sp<IMediaDrmService> service = interface_cast<IMediaDrmService>(binder);
39     if (service == NULL) {
40         ALOGE("CreateDrm failed at IMediaDrmService");
41         return NULL;
42     }
43 
44     sp<IDrm> drm = service->makeDrm();
45     if (drm == NULL) {
46         ALOGE("CreateDrm failed at makeDrm");
47         return NULL;
48     }
49 
50     // this is before plugin creation so NO_INIT is fine
51     status = drm->initCheck();
52     if (status != OK && status != NO_INIT) {
53         ALOGE("CreateDrm failed drm->initCheck(): %d", status);
54         return NULL;
55     }
56     return drm;
57 }
58 
createCrypto(status_t * pstatus)59 sp<ICrypto> NuPlayerDrm::createCrypto(status_t *pstatus)
60 {
61     status_t &status = *pstatus;
62     sp<IServiceManager> sm = defaultServiceManager();
63     sp<IBinder> binder = sm->getService(String16("media.drm"));
64 
65     sp<IMediaDrmService> service = interface_cast<IMediaDrmService>(binder);
66     if (service == NULL) {
67         status = UNKNOWN_ERROR;
68         ALOGE("CreateCrypto failed at IMediaDrmService");
69         return NULL;
70     }
71 
72     sp<ICrypto> crypto = service->makeCrypto();
73     if (crypto == NULL) {
74         status = UNKNOWN_ERROR;
75         ALOGE("createCrypto failed");
76         return NULL;
77     }
78 
79     // this is before plugin creation so NO_INIT is fine
80     status = crypto->initCheck();
81     if (status != OK && status != NO_INIT) {
82         ALOGE("createCrypto failed crypto->initCheck(): %d", status);
83         return NULL;
84     }
85 
86     return crypto;
87 }
88 
parsePSSH(const void * pssh,size_t psshsize)89 Vector<DrmUUID> NuPlayerDrm::parsePSSH(const void *pssh, size_t psshsize)
90 {
91     Vector<DrmUUID> drmSchemes, empty;
92     const int DATALEN_SIZE = 4;
93 
94     // the format of the buffer is 1 or more of:
95     //    {
96     //        16 byte uuid
97     //        4 byte data length N
98     //        N bytes of data
99     //    }
100     // Determine the number of entries in the source data.
101     // Since we got the data from stagefright, we trust it is valid and properly formatted.
102 
103     const uint8_t *data = (const uint8_t*)pssh;
104     size_t len = psshsize;
105     size_t numentries = 0;
106     while (len > 0) {
107         if (len < DrmUUID::UUID_SIZE) {
108             ALOGE("ParsePSSH: invalid PSSH data");
109             return empty;
110         }
111 
112         const uint8_t *uuidPtr = data;
113 
114         // skip uuid
115         data += DrmUUID::UUID_SIZE;
116         len -= DrmUUID::UUID_SIZE;
117 
118         // get data length
119         if (len < DATALEN_SIZE) {
120             ALOGE("ParsePSSH: invalid PSSH data");
121             return empty;
122         }
123 
124         uint32_t datalen = *((uint32_t*)data);
125         data += DATALEN_SIZE;
126         len -= DATALEN_SIZE;
127 
128         if (len < datalen) {
129             ALOGE("ParsePSSH: invalid PSSH data");
130             return empty;
131         }
132 
133         // skip the data
134         data += datalen;
135         len -= datalen;
136 
137         DrmUUID _uuid(uuidPtr);
138         drmSchemes.add(_uuid);
139 
140         ALOGV("ParsePSSH[%zu]: %s: %s", numentries,
141                 _uuid.toHexString().string(),
142                 DrmUUID::arrayToHex(data, datalen).string()
143              );
144 
145         numentries++;
146     }
147 
148     return drmSchemes;
149 }
150 
getSupportedDrmSchemes(const void * pssh,size_t psshsize)151 Vector<DrmUUID> NuPlayerDrm::getSupportedDrmSchemes(const void *pssh, size_t psshsize)
152 {
153     Vector<DrmUUID> psshDRMs = parsePSSH(pssh, psshsize);
154 
155     Vector<DrmUUID> supportedDRMs;
156      // temporary DRM object for crypto Scheme enquiry (without creating a plugin)
157     status_t status = OK;
158     sp<IDrm> drm = CreateDrm(&status);
159     if (drm != NULL) {
160         for (size_t i = 0; i < psshDRMs.size(); i++) {
161             DrmUUID uuid = psshDRMs[i];
162             if (drm->isCryptoSchemeSupported(uuid.ptr(), String8()))
163                 supportedDRMs.add(uuid);
164         }
165 
166         drm.clear();
167     } else {
168         ALOGE("getSupportedDrmSchemes: Can't create Drm obj: %d", status);
169     }
170 
171     ALOGV("getSupportedDrmSchemes: psshDRMs: %zu supportedDRMs: %zu",
172             psshDRMs.size(), supportedDRMs.size());
173 
174     return supportedDRMs;
175 }
176 
177 // static helpers - public
178 
createCryptoAndPlugin(const uint8_t uuid[16],const Vector<uint8_t> & drmSessionId,status_t & status)179 sp<ICrypto> NuPlayerDrm::createCryptoAndPlugin(const uint8_t uuid[16],
180         const Vector<uint8_t> &drmSessionId, status_t &status)
181 {
182     // Extra check
183     if (drmSessionId.isEmpty()) {
184         status = INVALID_OPERATION;
185         ALOGE("createCryptoAndPlugin: Failed. Empty drmSessionId. status: %d", status);
186         return NULL;
187     }
188 
189     status = OK;
190     sp<ICrypto> crypto = createCrypto(&status);
191     if (crypto == NULL) {
192         ALOGE("createCryptoAndPlugin: createCrypto failed. status: %d", status);
193         return NULL;
194     }
195     ALOGV("createCryptoAndPlugin: createCrypto succeeded");
196 
197     status = crypto->createPlugin(uuid, drmSessionId.array(), drmSessionId.size());
198     if (status != OK) {
199         ALOGE("createCryptoAndPlugin: createCryptoPlugin failed. status: %d", status);
200         // crypto will clean itself when leaving the current scope
201         return NULL;
202     }
203 
204     return crypto;
205 }
206 
207 // Parcel has only private copy constructor so passing it in rather than returning
retrieveDrmInfo(const void * pssh,size_t psshsize,Parcel * parcel)208 void NuPlayerDrm::retrieveDrmInfo(const void *pssh, size_t psshsize, Parcel *parcel)
209 {
210     // 1) PSSH bytes
211     parcel->writeUint32(psshsize);
212     parcel->writeByteArray(psshsize, (const uint8_t*)pssh);
213 
214     ALOGV("retrieveDrmInfo: MEDIA_DRM_INFO  PSSH: size: %zu %s", psshsize,
215             DrmUUID::arrayToHex((uint8_t*)pssh, psshsize).string());
216 
217     // 2) supportedDRMs
218     Vector<DrmUUID> supportedDRMs = getSupportedDrmSchemes(pssh, psshsize);
219     parcel->writeUint32(supportedDRMs.size());
220     for (size_t i = 0; i < supportedDRMs.size(); i++) {
221         DrmUUID uuid = supportedDRMs[i];
222         parcel->writeByteArray(DrmUUID::UUID_SIZE, uuid.ptr());
223 
224         ALOGV("retrieveDrmInfo: MEDIA_DRM_INFO  supportedScheme[%zu] %s", i,
225                 uuid.toHexString().string());
226     }
227 }
228 
229 ////////////////////////////////////////////////////////////////////////////////////////////
230 /// Helpers for NuPlayerDecoder
231 ////////////////////////////////////////////////////////////////////////////////////////////
232 
makeCryptoInfo(int numSubSamples,uint8_t key[kBlockSize],uint8_t iv[kBlockSize],CryptoPlugin::Mode mode,size_t * clearbytes,size_t * encryptedbytes)233 NuPlayerDrm::CryptoInfo *NuPlayerDrm::makeCryptoInfo(
234         int numSubSamples,
235         uint8_t key[kBlockSize],
236         uint8_t iv[kBlockSize],
237         CryptoPlugin::Mode mode,
238         size_t *clearbytes,
239         size_t *encryptedbytes)
240 {
241     // size needed to store all the crypto data
242     size_t cryptosize = sizeof(CryptoInfo) +
243                         sizeof(CryptoPlugin::SubSample) * numSubSamples;
244     CryptoInfo *ret = (CryptoInfo*) malloc(cryptosize);
245     if (ret == NULL) {
246         ALOGE("couldn't allocate %zu bytes", cryptosize);
247         return NULL;
248     }
249     ret->numSubSamples = numSubSamples;
250     memcpy(ret->key, key, kBlockSize);
251     memcpy(ret->iv, iv, kBlockSize);
252     ret->mode = mode;
253     ret->pattern.mEncryptBlocks = 0;
254     ret->pattern.mSkipBlocks = 0;
255     ret->subSamples = (CryptoPlugin::SubSample*)(ret + 1);
256     CryptoPlugin::SubSample *subSamples = ret->subSamples;
257 
258     for (int i = 0; i < numSubSamples; i++) {
259         subSamples[i].mNumBytesOfClearData = (clearbytes == NULL) ? 0 : clearbytes[i];
260         subSamples[i].mNumBytesOfEncryptedData = (encryptedbytes == NULL) ?
261                                                   0 :
262                                                   encryptedbytes[i];
263     }
264 
265     return ret;
266 }
267 
getSampleCryptoInfo(MetaDataBase & meta)268 NuPlayerDrm::CryptoInfo *NuPlayerDrm::getSampleCryptoInfo(MetaDataBase &meta)
269 {
270     uint32_t type;
271     const void *crypteddata;
272     size_t cryptedsize;
273 
274     if (!meta.findData(kKeyEncryptedSizes, &type, &crypteddata, &cryptedsize)) {
275         return NULL;
276     }
277     size_t numSubSamples = cryptedsize / sizeof(size_t);
278 
279     if (numSubSamples <= 0) {
280         ALOGE("getSampleCryptoInfo INVALID numSubSamples: %zu", numSubSamples);
281         return NULL;
282     }
283 
284     const void *cleardata;
285     size_t clearsize;
286     if (meta.findData(kKeyPlainSizes, &type, &cleardata, &clearsize)) {
287         if (clearsize != cryptedsize) {
288             // The two must be of the same length.
289             ALOGE("getSampleCryptoInfo mismatch cryptedsize: %zu != clearsize: %zu",
290                     cryptedsize, clearsize);
291             return NULL;
292         }
293     }
294 
295     const void *key;
296     size_t keysize;
297     if (meta.findData(kKeyCryptoKey, &type, &key, &keysize)) {
298         if (keysize != kBlockSize) {
299             ALOGE("getSampleCryptoInfo Keys must be %d bytes in length: %zu",
300                     kBlockSize, keysize);
301             // Keys must be 16 bytes in length.
302             return NULL;
303         }
304     }
305 
306     const void *iv;
307     size_t ivsize;
308     if (meta.findData(kKeyCryptoIV, &type, &iv, &ivsize)) {
309         if (ivsize != kBlockSize) {
310             ALOGE("getSampleCryptoInfo IV must be %d bytes in length: %zu",
311                     kBlockSize, ivsize);
312             // IVs must be 16 bytes in length.
313             return NULL;
314         }
315     }
316 
317     int32_t mode;
318     if (!meta.findInt32(kKeyCryptoMode, &mode)) {
319         mode = CryptoPlugin::kMode_AES_CTR;
320     }
321 
322     return makeCryptoInfo(numSubSamples,
323             (uint8_t*) key,
324             (uint8_t*) iv,
325             (CryptoPlugin::Mode)mode,
326             (size_t*) cleardata,
327             (size_t*) crypteddata);
328 }
329 
330 }   // namespace android
331 
332