1*ec779b8eSAndroid Build Coastguard Worker /*
2*ec779b8eSAndroid Build Coastguard Worker * Copyright (C) 2009 The Android Open Source Project
3*ec779b8eSAndroid Build Coastguard Worker *
4*ec779b8eSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*ec779b8eSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*ec779b8eSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*ec779b8eSAndroid Build Coastguard Worker *
8*ec779b8eSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*ec779b8eSAndroid Build Coastguard Worker *
10*ec779b8eSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*ec779b8eSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*ec779b8eSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*ec779b8eSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*ec779b8eSAndroid Build Coastguard Worker * limitations under the License.
15*ec779b8eSAndroid Build Coastguard Worker */
16*ec779b8eSAndroid Build Coastguard Worker
17*ec779b8eSAndroid Build Coastguard Worker //#define LOG_NDEBUG 0
18*ec779b8eSAndroid Build Coastguard Worker #define LOG_TAG "PlayerServiceFileSource"
19*ec779b8eSAndroid Build Coastguard Worker #include <utils/Log.h>
20*ec779b8eSAndroid Build Coastguard Worker
21*ec779b8eSAndroid Build Coastguard Worker #include <datasource/PlayerServiceFileSource.h>
22*ec779b8eSAndroid Build Coastguard Worker #include <media/stagefright/foundation/ADebug.h>
23*ec779b8eSAndroid Build Coastguard Worker #include <private/android_filesystem_config.h>
24*ec779b8eSAndroid Build Coastguard Worker
25*ec779b8eSAndroid Build Coastguard Worker namespace android {
26*ec779b8eSAndroid Build Coastguard Worker
PlayerServiceFileSource(const char * filename)27*ec779b8eSAndroid Build Coastguard Worker PlayerServiceFileSource::PlayerServiceFileSource(const char *filename)
28*ec779b8eSAndroid Build Coastguard Worker : FileSource(filename),
29*ec779b8eSAndroid Build Coastguard Worker mDecryptHandle(NULL),
30*ec779b8eSAndroid Build Coastguard Worker mDrmManagerClient(NULL),
31*ec779b8eSAndroid Build Coastguard Worker mDrmBufOffset(0),
32*ec779b8eSAndroid Build Coastguard Worker mDrmBufSize(0),
33*ec779b8eSAndroid Build Coastguard Worker mDrmBuf(NULL){
34*ec779b8eSAndroid Build Coastguard Worker (void) DrmInitialization(nullptr);
35*ec779b8eSAndroid Build Coastguard Worker }
36*ec779b8eSAndroid Build Coastguard Worker
PlayerServiceFileSource(int fd,int64_t offset,int64_t length)37*ec779b8eSAndroid Build Coastguard Worker PlayerServiceFileSource::PlayerServiceFileSource(int fd, int64_t offset, int64_t length)
38*ec779b8eSAndroid Build Coastguard Worker : FileSource(fd, offset, length),
39*ec779b8eSAndroid Build Coastguard Worker mDecryptHandle(NULL),
40*ec779b8eSAndroid Build Coastguard Worker mDrmManagerClient(NULL),
41*ec779b8eSAndroid Build Coastguard Worker mDrmBufOffset(0),
42*ec779b8eSAndroid Build Coastguard Worker mDrmBufSize(0),
43*ec779b8eSAndroid Build Coastguard Worker mDrmBuf(NULL) {
44*ec779b8eSAndroid Build Coastguard Worker (void) DrmInitialization(nullptr);
45*ec779b8eSAndroid Build Coastguard Worker }
46*ec779b8eSAndroid Build Coastguard Worker
~PlayerServiceFileSource()47*ec779b8eSAndroid Build Coastguard Worker PlayerServiceFileSource::~PlayerServiceFileSource() {
48*ec779b8eSAndroid Build Coastguard Worker if (mDrmBuf != NULL) {
49*ec779b8eSAndroid Build Coastguard Worker delete[] mDrmBuf;
50*ec779b8eSAndroid Build Coastguard Worker mDrmBuf = NULL;
51*ec779b8eSAndroid Build Coastguard Worker }
52*ec779b8eSAndroid Build Coastguard Worker
53*ec779b8eSAndroid Build Coastguard Worker if (mDecryptHandle != NULL) {
54*ec779b8eSAndroid Build Coastguard Worker // To release mDecryptHandle
55*ec779b8eSAndroid Build Coastguard Worker CHECK(mDrmManagerClient);
56*ec779b8eSAndroid Build Coastguard Worker mDrmManagerClient->closeDecryptSession(mDecryptHandle);
57*ec779b8eSAndroid Build Coastguard Worker mDecryptHandle = NULL;
58*ec779b8eSAndroid Build Coastguard Worker }
59*ec779b8eSAndroid Build Coastguard Worker
60*ec779b8eSAndroid Build Coastguard Worker if (mDrmManagerClient != NULL) {
61*ec779b8eSAndroid Build Coastguard Worker delete mDrmManagerClient;
62*ec779b8eSAndroid Build Coastguard Worker mDrmManagerClient = NULL;
63*ec779b8eSAndroid Build Coastguard Worker }
64*ec779b8eSAndroid Build Coastguard Worker }
65*ec779b8eSAndroid Build Coastguard Worker
readAt(off64_t offset,void * data,size_t size)66*ec779b8eSAndroid Build Coastguard Worker ssize_t PlayerServiceFileSource::readAt(off64_t offset, void *data, size_t size) {
67*ec779b8eSAndroid Build Coastguard Worker if (mFd < 0) {
68*ec779b8eSAndroid Build Coastguard Worker return NO_INIT;
69*ec779b8eSAndroid Build Coastguard Worker }
70*ec779b8eSAndroid Build Coastguard Worker
71*ec779b8eSAndroid Build Coastguard Worker Mutex::Autolock autoLock(mLock);
72*ec779b8eSAndroid Build Coastguard Worker
73*ec779b8eSAndroid Build Coastguard Worker if (mLength >= 0) {
74*ec779b8eSAndroid Build Coastguard Worker if (offset < 0) {
75*ec779b8eSAndroid Build Coastguard Worker return UNKNOWN_ERROR;
76*ec779b8eSAndroid Build Coastguard Worker }
77*ec779b8eSAndroid Build Coastguard Worker if (offset >= mLength) {
78*ec779b8eSAndroid Build Coastguard Worker return 0; // read beyond EOF.
79*ec779b8eSAndroid Build Coastguard Worker }
80*ec779b8eSAndroid Build Coastguard Worker uint64_t numAvailable = mLength - offset;
81*ec779b8eSAndroid Build Coastguard Worker if ((uint64_t)size > numAvailable) {
82*ec779b8eSAndroid Build Coastguard Worker size = numAvailable;
83*ec779b8eSAndroid Build Coastguard Worker }
84*ec779b8eSAndroid Build Coastguard Worker }
85*ec779b8eSAndroid Build Coastguard Worker
86*ec779b8eSAndroid Build Coastguard Worker if (mDecryptHandle != NULL && DecryptApiType::CONTAINER_BASED
87*ec779b8eSAndroid Build Coastguard Worker == mDecryptHandle->decryptApiType) {
88*ec779b8eSAndroid Build Coastguard Worker return readAtDRM_l(offset, data, size);
89*ec779b8eSAndroid Build Coastguard Worker } else {
90*ec779b8eSAndroid Build Coastguard Worker return readAt_l(offset, data, size);
91*ec779b8eSAndroid Build Coastguard Worker }
92*ec779b8eSAndroid Build Coastguard Worker }
93*ec779b8eSAndroid Build Coastguard Worker
DrmInitialization(const char * mime)94*ec779b8eSAndroid Build Coastguard Worker sp<DecryptHandle> PlayerServiceFileSource::DrmInitialization(const char *mime) {
95*ec779b8eSAndroid Build Coastguard Worker if (getuid() == AID_MEDIA_EX) {
96*ec779b8eSAndroid Build Coastguard Worker return NULL; // no DRM in media extractor
97*ec779b8eSAndroid Build Coastguard Worker }
98*ec779b8eSAndroid Build Coastguard Worker if (mDrmManagerClient == NULL) {
99*ec779b8eSAndroid Build Coastguard Worker mDrmManagerClient = new DrmManagerClient();
100*ec779b8eSAndroid Build Coastguard Worker }
101*ec779b8eSAndroid Build Coastguard Worker
102*ec779b8eSAndroid Build Coastguard Worker if (mDrmManagerClient == NULL) {
103*ec779b8eSAndroid Build Coastguard Worker return NULL;
104*ec779b8eSAndroid Build Coastguard Worker }
105*ec779b8eSAndroid Build Coastguard Worker
106*ec779b8eSAndroid Build Coastguard Worker if (mDecryptHandle == NULL) {
107*ec779b8eSAndroid Build Coastguard Worker mDecryptHandle = mDrmManagerClient->openDecryptSession(
108*ec779b8eSAndroid Build Coastguard Worker mFd, mOffset, mLength, mime);
109*ec779b8eSAndroid Build Coastguard Worker }
110*ec779b8eSAndroid Build Coastguard Worker
111*ec779b8eSAndroid Build Coastguard Worker if (mDecryptHandle == NULL) {
112*ec779b8eSAndroid Build Coastguard Worker delete mDrmManagerClient;
113*ec779b8eSAndroid Build Coastguard Worker mDrmManagerClient = NULL;
114*ec779b8eSAndroid Build Coastguard Worker }
115*ec779b8eSAndroid Build Coastguard Worker
116*ec779b8eSAndroid Build Coastguard Worker return mDecryptHandle;
117*ec779b8eSAndroid Build Coastguard Worker }
118*ec779b8eSAndroid Build Coastguard Worker
readAtDRM_l(off64_t offset,void * data,size_t size)119*ec779b8eSAndroid Build Coastguard Worker ssize_t PlayerServiceFileSource::readAtDRM_l(off64_t offset, void *data, size_t size) {
120*ec779b8eSAndroid Build Coastguard Worker size_t DRM_CACHE_SIZE = 1024;
121*ec779b8eSAndroid Build Coastguard Worker if (mDrmBuf == NULL) {
122*ec779b8eSAndroid Build Coastguard Worker mDrmBuf = new unsigned char[DRM_CACHE_SIZE];
123*ec779b8eSAndroid Build Coastguard Worker }
124*ec779b8eSAndroid Build Coastguard Worker
125*ec779b8eSAndroid Build Coastguard Worker if (mDrmBuf != NULL && mDrmBufSize > 0 && (offset + mOffset) >= mDrmBufOffset
126*ec779b8eSAndroid Build Coastguard Worker && (offset + mOffset + size) <= static_cast<size_t>(mDrmBufOffset + mDrmBufSize)) {
127*ec779b8eSAndroid Build Coastguard Worker /* Use buffered data */
128*ec779b8eSAndroid Build Coastguard Worker memcpy(data, (void*)(mDrmBuf+(offset+mOffset-mDrmBufOffset)), size);
129*ec779b8eSAndroid Build Coastguard Worker return size;
130*ec779b8eSAndroid Build Coastguard Worker } else if (size <= DRM_CACHE_SIZE) {
131*ec779b8eSAndroid Build Coastguard Worker /* Buffer new data */
132*ec779b8eSAndroid Build Coastguard Worker mDrmBufOffset = offset + mOffset;
133*ec779b8eSAndroid Build Coastguard Worker mDrmBufSize = mDrmManagerClient->pread(mDecryptHandle, mDrmBuf,
134*ec779b8eSAndroid Build Coastguard Worker DRM_CACHE_SIZE, offset + mOffset);
135*ec779b8eSAndroid Build Coastguard Worker if (mDrmBufSize > 0) {
136*ec779b8eSAndroid Build Coastguard Worker int64_t dataRead = 0;
137*ec779b8eSAndroid Build Coastguard Worker dataRead = size > static_cast<size_t>(mDrmBufSize) ? mDrmBufSize : size;
138*ec779b8eSAndroid Build Coastguard Worker memcpy(data, (void*)mDrmBuf, dataRead);
139*ec779b8eSAndroid Build Coastguard Worker return dataRead;
140*ec779b8eSAndroid Build Coastguard Worker } else {
141*ec779b8eSAndroid Build Coastguard Worker return mDrmBufSize;
142*ec779b8eSAndroid Build Coastguard Worker }
143*ec779b8eSAndroid Build Coastguard Worker } else {
144*ec779b8eSAndroid Build Coastguard Worker /* Too big chunk to cache. Call DRM directly */
145*ec779b8eSAndroid Build Coastguard Worker return mDrmManagerClient->pread(mDecryptHandle, data, size, offset + mOffset);
146*ec779b8eSAndroid Build Coastguard Worker }
147*ec779b8eSAndroid Build Coastguard Worker }
148*ec779b8eSAndroid Build Coastguard Worker
149*ec779b8eSAndroid Build Coastguard Worker /* static */
requiresDrm(int fd,int64_t offset,int64_t length,const char * mime)150*ec779b8eSAndroid Build Coastguard Worker bool PlayerServiceFileSource::requiresDrm(int fd, int64_t offset, int64_t length, const char *mime) {
151*ec779b8eSAndroid Build Coastguard Worker std::unique_ptr<DrmManagerClient> drmClient(new DrmManagerClient());
152*ec779b8eSAndroid Build Coastguard Worker sp<DecryptHandle> decryptHandle =
153*ec779b8eSAndroid Build Coastguard Worker drmClient->openDecryptSession(fd, offset, length, mime);
154*ec779b8eSAndroid Build Coastguard Worker bool requiresDrm = false;
155*ec779b8eSAndroid Build Coastguard Worker if (decryptHandle != nullptr) {
156*ec779b8eSAndroid Build Coastguard Worker requiresDrm = decryptHandle->decryptApiType == DecryptApiType::CONTAINER_BASED;
157*ec779b8eSAndroid Build Coastguard Worker drmClient->closeDecryptSession(decryptHandle);
158*ec779b8eSAndroid Build Coastguard Worker }
159*ec779b8eSAndroid Build Coastguard Worker return requiresDrm;
160*ec779b8eSAndroid Build Coastguard Worker }
161*ec779b8eSAndroid Build Coastguard Worker
162*ec779b8eSAndroid Build Coastguard Worker } // namespace android
163