/*
* Copyright 2017 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 "NuPlayer2Drm"
#include "NuPlayer2Drm.h"
#include <media/NdkWrapper.h>
#include <utils/Log.h>
#include <sstream>
namespace android {
Vector<DrmUUID> NuPlayer2Drm::parsePSSH(const void *pssh, size_t psshsize)
{
Vector<DrmUUID> drmSchemes, empty;
const int DATALEN_SIZE = 4;
// the format of the buffer is 1 or more of:
// {
// 16 byte uuid
// 4 byte data length N
// N bytes of data
// }
// Determine the number of entries in the source data.
// Since we got the data from stagefright, we trust it is valid and properly formatted.
const uint8_t *data = (const uint8_t*)pssh;
size_t len = psshsize;
size_t numentries = 0;
while (len > 0) {
if (len < DrmUUID::UUID_SIZE) {
ALOGE("ParsePSSH: invalid PSSH data");
return empty;
}
const uint8_t *uuidPtr = data;
// skip uuid
data += DrmUUID::UUID_SIZE;
len -= DrmUUID::UUID_SIZE;
// get data length
if (len < DATALEN_SIZE) {
ALOGE("ParsePSSH: invalid PSSH data");
return empty;
}
uint32_t datalen = *((uint32_t*)data);
data += DATALEN_SIZE;
len -= DATALEN_SIZE;
if (len < datalen) {
ALOGE("ParsePSSH: invalid PSSH data");
return empty;
}
// skip the data
data += datalen;
len -= datalen;
DrmUUID _uuid(uuidPtr);
drmSchemes.add(_uuid);
ALOGV("ParsePSSH[%zu]: %s: %s", numentries,
_uuid.toHexString().string(),
DrmUUID::arrayToHex(data, datalen).string()
);
numentries++;
}
return drmSchemes;
}
Vector<DrmUUID> NuPlayer2Drm::getSupportedDrmSchemes(const void *pssh, size_t psshsize)
{
Vector<DrmUUID> psshDRMs = parsePSSH(pssh, psshsize);
Vector<DrmUUID> supportedDRMs;
for (size_t i = 0; i < psshDRMs.size(); i++) {
DrmUUID uuid = psshDRMs[i];
if (AMediaDrmWrapper::isCryptoSchemeSupported(uuid.ptr(), NULL)) {
supportedDRMs.add(uuid);
}
}
ALOGV("getSupportedDrmSchemes: psshDRMs: %zu supportedDRMs: %zu",
psshDRMs.size(), supportedDRMs.size());
return supportedDRMs;
}
sp<ABuffer> NuPlayer2Drm::retrieveDrmInfo(const void *pssh, uint32_t psshsize)
{
std::ostringstream buf;
// 1) PSSH bytes
buf.write(reinterpret_cast<const char *>(&psshsize), sizeof(psshsize));
buf.write(reinterpret_cast<const char *>(pssh), psshsize);
ALOGV("retrieveDrmInfo: MEDIA2_DRM_INFO PSSH: size: %u %s", psshsize,
DrmUUID::arrayToHex((uint8_t*)pssh, psshsize).string());
// 2) supportedDRMs
Vector<DrmUUID> supportedDRMs = getSupportedDrmSchemes(pssh, psshsize);
uint32_t n = supportedDRMs.size();
buf.write(reinterpret_cast<char *>(&n), sizeof(n));
for (size_t i = 0; i < n; i++) {
DrmUUID uuid = supportedDRMs[i];
buf.write(reinterpret_cast<const char *>(&n), sizeof(n));
buf.write(reinterpret_cast<const char *>(uuid.ptr()), DrmUUID::UUID_SIZE);
ALOGV("retrieveDrmInfo: MEDIA2_DRM_INFO supportedScheme[%zu] %s", i,
uuid.toHexString().string());
}
sp<ABuffer> drmInfoBuffer = ABuffer::CreateAsCopy(buf.str().c_str(), buf.tellp());
return drmInfoBuffer;
}
sp<ABuffer> NuPlayer2Drm::retrieveDrmInfo(PsshInfo *psshInfo)
{
std::ostringstream pssh, drmInfo;
// 0) Generate PSSH bytes
for (size_t i = 0; i < psshInfo->numentries; i++) {
PsshEntry *entry = &psshInfo->entries[i];
uint32_t datalen = entry->datalen;
pssh.write(reinterpret_cast<const char *>(&entry->uuid), sizeof(entry->uuid));
pssh.write(reinterpret_cast<const char *>(&datalen), sizeof(datalen));
pssh.write(reinterpret_cast<const char *>(entry->data), datalen);
}
uint32_t psshSize = pssh.tellp();
const uint8_t* psshPtr = reinterpret_cast<const uint8_t*>(pssh.str().c_str());
const char *psshHex = DrmUUID::arrayToHex(psshPtr, psshSize).string();
ALOGV("retrieveDrmInfo: MEDIA_DRM_INFO PSSH: size: %u %s", psshSize, psshHex);
// 1) Write PSSH bytes
drmInfo.write(reinterpret_cast<const char *>(&psshSize), sizeof(psshSize));
drmInfo.write(reinterpret_cast<const char *>(pssh.str().c_str()), psshSize);
// 2) Write supportedDRMs
uint32_t numentries = psshInfo->numentries;
drmInfo.write(reinterpret_cast<const char *>(&numentries), sizeof(numentries));
for (size_t i = 0; i < numentries; i++) {
PsshEntry *entry = &psshInfo->entries[i];
drmInfo.write(reinterpret_cast<const char *>(&entry->uuid), sizeof(entry->uuid));
ALOGV("retrieveDrmInfo: MEDIA_DRM_INFO supportedScheme[%zu] %s", i,
DrmUUID::arrayToHex((const uint8_t*)&entry->uuid, sizeof(AMediaUUID)).string());
}
sp<ABuffer> drmInfoBuf = ABuffer::CreateAsCopy(drmInfo.str().c_str(), drmInfo.tellp());
drmInfoBuf->setRange(0, drmInfo.tellp());
return drmInfoBuf;
}
} // namespace android