1*ec779b8eSAndroid Build Coastguard Worker /*
2*ec779b8eSAndroid Build Coastguard Worker * Copyright (C) 2018 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 "NdkMediaDataSource"
19*ec779b8eSAndroid Build Coastguard Worker
20*ec779b8eSAndroid Build Coastguard Worker #include "NdkJavaVMHelperPriv.h"
21*ec779b8eSAndroid Build Coastguard Worker #include "NdkMediaDataSourcePriv.h"
22*ec779b8eSAndroid Build Coastguard Worker
23*ec779b8eSAndroid Build Coastguard Worker #include <inttypes.h>
24*ec779b8eSAndroid Build Coastguard Worker #include <jni.h>
25*ec779b8eSAndroid Build Coastguard Worker #include <unistd.h>
26*ec779b8eSAndroid Build Coastguard Worker
27*ec779b8eSAndroid Build Coastguard Worker #include <android_runtime/AndroidRuntime.h>
28*ec779b8eSAndroid Build Coastguard Worker #include <android_util_Binder.h>
29*ec779b8eSAndroid Build Coastguard Worker #include <cutils/properties.h>
30*ec779b8eSAndroid Build Coastguard Worker #include <datasource/DataSourceFactory.h>
31*ec779b8eSAndroid Build Coastguard Worker #include <datasource/HTTPBase.h>
32*ec779b8eSAndroid Build Coastguard Worker #include <datasource/NuCachedSource2.h>
33*ec779b8eSAndroid Build Coastguard Worker #include <media/IMediaHTTPService.h>
34*ec779b8eSAndroid Build Coastguard Worker #include <media/NdkMediaError.h>
35*ec779b8eSAndroid Build Coastguard Worker #include <media/NdkMediaDataSource.h>
36*ec779b8eSAndroid Build Coastguard Worker #include <media/stagefright/InterfaceUtils.h>
37*ec779b8eSAndroid Build Coastguard Worker #include <utils/Log.h>
38*ec779b8eSAndroid Build Coastguard Worker #include <utils/StrongPointer.h>
39*ec779b8eSAndroid Build Coastguard Worker
40*ec779b8eSAndroid Build Coastguard Worker #include "NdkMediaDataSourceCallbacksPriv.h"
41*ec779b8eSAndroid Build Coastguard Worker
42*ec779b8eSAndroid Build Coastguard Worker
43*ec779b8eSAndroid Build Coastguard Worker using namespace android;
44*ec779b8eSAndroid Build Coastguard Worker
45*ec779b8eSAndroid Build Coastguard Worker struct AMediaDataSource {
46*ec779b8eSAndroid Build Coastguard Worker void *userdata;
47*ec779b8eSAndroid Build Coastguard Worker AMediaDataSourceReadAt readAt;
48*ec779b8eSAndroid Build Coastguard Worker AMediaDataSourceGetSize getSize;
49*ec779b8eSAndroid Build Coastguard Worker AMediaDataSourceClose close;
50*ec779b8eSAndroid Build Coastguard Worker AMediaDataSourceGetAvailableSize getAvailableSize;
51*ec779b8eSAndroid Build Coastguard Worker sp<DataSource> mImpl;
52*ec779b8eSAndroid Build Coastguard Worker uint32_t mFlags;
53*ec779b8eSAndroid Build Coastguard Worker };
54*ec779b8eSAndroid Build Coastguard Worker
NdkDataSource(AMediaDataSource * dataSource)55*ec779b8eSAndroid Build Coastguard Worker NdkDataSource::NdkDataSource(AMediaDataSource *dataSource)
56*ec779b8eSAndroid Build Coastguard Worker : mDataSource(AMediaDataSource_new()) {
57*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource_setReadAt(mDataSource, dataSource->readAt);
58*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource_setGetSize(mDataSource, dataSource->getSize);
59*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource_setClose(mDataSource, dataSource->close);
60*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource_setUserdata(mDataSource, dataSource->userdata);
61*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource_setGetAvailableSize(mDataSource, dataSource->getAvailableSize);
62*ec779b8eSAndroid Build Coastguard Worker mDataSource->mImpl = dataSource->mImpl;
63*ec779b8eSAndroid Build Coastguard Worker mDataSource->mFlags = dataSource->mFlags;
64*ec779b8eSAndroid Build Coastguard Worker }
65*ec779b8eSAndroid Build Coastguard Worker
~NdkDataSource()66*ec779b8eSAndroid Build Coastguard Worker NdkDataSource::~NdkDataSource() {
67*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource_delete(mDataSource);
68*ec779b8eSAndroid Build Coastguard Worker }
69*ec779b8eSAndroid Build Coastguard Worker
initCheck() const70*ec779b8eSAndroid Build Coastguard Worker status_t NdkDataSource::initCheck() const {
71*ec779b8eSAndroid Build Coastguard Worker return OK;
72*ec779b8eSAndroid Build Coastguard Worker }
73*ec779b8eSAndroid Build Coastguard Worker
flags()74*ec779b8eSAndroid Build Coastguard Worker uint32_t NdkDataSource::flags() {
75*ec779b8eSAndroid Build Coastguard Worker return mDataSource->mFlags;
76*ec779b8eSAndroid Build Coastguard Worker }
77*ec779b8eSAndroid Build Coastguard Worker
readAt(off64_t offset,void * data,size_t size)78*ec779b8eSAndroid Build Coastguard Worker ssize_t NdkDataSource::readAt(off64_t offset, void *data, size_t size) {
79*ec779b8eSAndroid Build Coastguard Worker Mutex::Autolock l(mLock);
80*ec779b8eSAndroid Build Coastguard Worker if (mDataSource->readAt == NULL || mDataSource->userdata == NULL) {
81*ec779b8eSAndroid Build Coastguard Worker return -1;
82*ec779b8eSAndroid Build Coastguard Worker }
83*ec779b8eSAndroid Build Coastguard Worker return mDataSource->readAt(mDataSource->userdata, offset, data, size);
84*ec779b8eSAndroid Build Coastguard Worker }
85*ec779b8eSAndroid Build Coastguard Worker
getSize(off64_t * size)86*ec779b8eSAndroid Build Coastguard Worker status_t NdkDataSource::getSize(off64_t *size) {
87*ec779b8eSAndroid Build Coastguard Worker Mutex::Autolock l(mLock);
88*ec779b8eSAndroid Build Coastguard Worker if (mDataSource->getSize == NULL || mDataSource->userdata == NULL) {
89*ec779b8eSAndroid Build Coastguard Worker return NO_INIT;
90*ec779b8eSAndroid Build Coastguard Worker }
91*ec779b8eSAndroid Build Coastguard Worker if (size != NULL) {
92*ec779b8eSAndroid Build Coastguard Worker *size = mDataSource->getSize(mDataSource->userdata);
93*ec779b8eSAndroid Build Coastguard Worker }
94*ec779b8eSAndroid Build Coastguard Worker return OK;
95*ec779b8eSAndroid Build Coastguard Worker }
96*ec779b8eSAndroid Build Coastguard Worker
toString()97*ec779b8eSAndroid Build Coastguard Worker String8 NdkDataSource::toString() {
98*ec779b8eSAndroid Build Coastguard Worker return String8::format("NdkDataSource(pid %d, uid %d)", getpid(), getuid());
99*ec779b8eSAndroid Build Coastguard Worker }
100*ec779b8eSAndroid Build Coastguard Worker
getMIMEType() const101*ec779b8eSAndroid Build Coastguard Worker String8 NdkDataSource::getMIMEType() const {
102*ec779b8eSAndroid Build Coastguard Worker return String8("application/octet-stream");
103*ec779b8eSAndroid Build Coastguard Worker }
104*ec779b8eSAndroid Build Coastguard Worker
close()105*ec779b8eSAndroid Build Coastguard Worker void NdkDataSource::close() {
106*ec779b8eSAndroid Build Coastguard Worker if (mDataSource->close != NULL && mDataSource->userdata != NULL) {
107*ec779b8eSAndroid Build Coastguard Worker mDataSource->close(mDataSource->userdata);
108*ec779b8eSAndroid Build Coastguard Worker }
109*ec779b8eSAndroid Build Coastguard Worker }
110*ec779b8eSAndroid Build Coastguard Worker
getAvailableSize(off64_t offset,off64_t * sizeptr)111*ec779b8eSAndroid Build Coastguard Worker status_t NdkDataSource::getAvailableSize(off64_t offset, off64_t *sizeptr) {
112*ec779b8eSAndroid Build Coastguard Worker off64_t size = -1;
113*ec779b8eSAndroid Build Coastguard Worker if (mDataSource->getAvailableSize != NULL
114*ec779b8eSAndroid Build Coastguard Worker && mDataSource->userdata != NULL
115*ec779b8eSAndroid Build Coastguard Worker && sizeptr != NULL) {
116*ec779b8eSAndroid Build Coastguard Worker size = mDataSource->getAvailableSize(mDataSource->userdata, offset);
117*ec779b8eSAndroid Build Coastguard Worker *sizeptr = size;
118*ec779b8eSAndroid Build Coastguard Worker }
119*ec779b8eSAndroid Build Coastguard Worker return size >= 0 ? OK : UNKNOWN_ERROR;
120*ec779b8eSAndroid Build Coastguard Worker }
121*ec779b8eSAndroid Build Coastguard Worker
createMediaHttpServiceFromJavaObj(JNIEnv * env,jobject obj)122*ec779b8eSAndroid Build Coastguard Worker static sp<MediaHTTPService> createMediaHttpServiceFromJavaObj(JNIEnv *env, jobject obj) {
123*ec779b8eSAndroid Build Coastguard Worker if (obj == NULL) {
124*ec779b8eSAndroid Build Coastguard Worker return NULL;
125*ec779b8eSAndroid Build Coastguard Worker }
126*ec779b8eSAndroid Build Coastguard Worker return interface_cast<IMediaHTTPService>(ibinderForJavaObject(env, obj));
127*ec779b8eSAndroid Build Coastguard Worker }
128*ec779b8eSAndroid Build Coastguard Worker
createMediaHttpServiceTemplate(JNIEnv * env,const char * uri,const char * clazz,const char * method,const char * signature)129*ec779b8eSAndroid Build Coastguard Worker static sp<MediaHTTPService> createMediaHttpServiceTemplate(
130*ec779b8eSAndroid Build Coastguard Worker JNIEnv *env,
131*ec779b8eSAndroid Build Coastguard Worker const char *uri,
132*ec779b8eSAndroid Build Coastguard Worker const char *clazz,
133*ec779b8eSAndroid Build Coastguard Worker const char *method,
134*ec779b8eSAndroid Build Coastguard Worker const char *signature) {
135*ec779b8eSAndroid Build Coastguard Worker jobject service = NULL;
136*ec779b8eSAndroid Build Coastguard Worker if (env == NULL) {
137*ec779b8eSAndroid Build Coastguard Worker ALOGE("http service must be created from Java thread");
138*ec779b8eSAndroid Build Coastguard Worker return NULL;
139*ec779b8eSAndroid Build Coastguard Worker }
140*ec779b8eSAndroid Build Coastguard Worker
141*ec779b8eSAndroid Build Coastguard Worker jclass mediahttpclass = env->FindClass(clazz);
142*ec779b8eSAndroid Build Coastguard Worker if (mediahttpclass == NULL) {
143*ec779b8eSAndroid Build Coastguard Worker ALOGE("can't find Media(2)HttpService");
144*ec779b8eSAndroid Build Coastguard Worker env->ExceptionClear();
145*ec779b8eSAndroid Build Coastguard Worker return NULL;
146*ec779b8eSAndroid Build Coastguard Worker }
147*ec779b8eSAndroid Build Coastguard Worker
148*ec779b8eSAndroid Build Coastguard Worker jmethodID mediaHttpCreateMethod = env->GetStaticMethodID(mediahttpclass, method, signature);
149*ec779b8eSAndroid Build Coastguard Worker if (mediaHttpCreateMethod == NULL) {
150*ec779b8eSAndroid Build Coastguard Worker ALOGE("can't find method");
151*ec779b8eSAndroid Build Coastguard Worker env->ExceptionClear();
152*ec779b8eSAndroid Build Coastguard Worker return NULL;
153*ec779b8eSAndroid Build Coastguard Worker }
154*ec779b8eSAndroid Build Coastguard Worker
155*ec779b8eSAndroid Build Coastguard Worker jstring juri = env->NewStringUTF(uri);
156*ec779b8eSAndroid Build Coastguard Worker
157*ec779b8eSAndroid Build Coastguard Worker service = env->CallStaticObjectMethod(mediahttpclass, mediaHttpCreateMethod, juri);
158*ec779b8eSAndroid Build Coastguard Worker env->DeleteLocalRef(juri);
159*ec779b8eSAndroid Build Coastguard Worker
160*ec779b8eSAndroid Build Coastguard Worker env->ExceptionClear();
161*ec779b8eSAndroid Build Coastguard Worker sp<MediaHTTPService> httpService = createMediaHttpServiceFromJavaObj(env, service);
162*ec779b8eSAndroid Build Coastguard Worker return httpService;
163*ec779b8eSAndroid Build Coastguard Worker
164*ec779b8eSAndroid Build Coastguard Worker }
165*ec779b8eSAndroid Build Coastguard Worker
createMediaHttpService(const char * uri)166*ec779b8eSAndroid Build Coastguard Worker sp<MediaHTTPService> createMediaHttpService(const char *uri) {
167*ec779b8eSAndroid Build Coastguard Worker
168*ec779b8eSAndroid Build Coastguard Worker JNIEnv *env;
169*ec779b8eSAndroid Build Coastguard Worker const char *clazz, *method, *signature;
170*ec779b8eSAndroid Build Coastguard Worker
171*ec779b8eSAndroid Build Coastguard Worker env = NdkJavaVMHelper::getJNIEnv();
172*ec779b8eSAndroid Build Coastguard Worker
173*ec779b8eSAndroid Build Coastguard Worker clazz = "android/media/MediaHTTPService";
174*ec779b8eSAndroid Build Coastguard Worker method = "createHttpServiceBinderIfNecessary";
175*ec779b8eSAndroid Build Coastguard Worker signature = "(Ljava/lang/String;)Landroid/os/IBinder;";
176*ec779b8eSAndroid Build Coastguard Worker
177*ec779b8eSAndroid Build Coastguard Worker return createMediaHttpServiceTemplate(env, uri, clazz, method, signature);
178*ec779b8eSAndroid Build Coastguard Worker
179*ec779b8eSAndroid Build Coastguard Worker }
180*ec779b8eSAndroid Build Coastguard Worker
181*ec779b8eSAndroid Build Coastguard Worker extern "C" {
182*ec779b8eSAndroid Build Coastguard Worker
183*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_new()184*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource* AMediaDataSource_new() {
185*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource *mSource = new AMediaDataSource();
186*ec779b8eSAndroid Build Coastguard Worker mSource->userdata = NULL;
187*ec779b8eSAndroid Build Coastguard Worker mSource->readAt = NULL;
188*ec779b8eSAndroid Build Coastguard Worker mSource->getSize = NULL;
189*ec779b8eSAndroid Build Coastguard Worker mSource->close = NULL;
190*ec779b8eSAndroid Build Coastguard Worker return mSource;
191*ec779b8eSAndroid Build Coastguard Worker }
192*ec779b8eSAndroid Build Coastguard Worker
193*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_newUri(const char * uri,int numheaders,const char * const * key_values)194*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource* AMediaDataSource_newUri(
195*ec779b8eSAndroid Build Coastguard Worker const char *uri,
196*ec779b8eSAndroid Build Coastguard Worker int numheaders,
197*ec779b8eSAndroid Build Coastguard Worker const char * const *key_values) {
198*ec779b8eSAndroid Build Coastguard Worker
199*ec779b8eSAndroid Build Coastguard Worker sp<MediaHTTPService> service = createMediaHttpService(uri);
200*ec779b8eSAndroid Build Coastguard Worker KeyedVector<String8, String8> headers;
201*ec779b8eSAndroid Build Coastguard Worker for (int i = 0; i < numheaders; ++i) {
202*ec779b8eSAndroid Build Coastguard Worker String8 key8(key_values[i * 2]);
203*ec779b8eSAndroid Build Coastguard Worker String8 value8(key_values[i * 2 + 1]);
204*ec779b8eSAndroid Build Coastguard Worker headers.add(key8, value8);
205*ec779b8eSAndroid Build Coastguard Worker }
206*ec779b8eSAndroid Build Coastguard Worker
207*ec779b8eSAndroid Build Coastguard Worker sp<DataSource> source = DataSourceFactory::getInstance()->CreateFromURI(service, uri, &headers);
208*ec779b8eSAndroid Build Coastguard Worker if (source == NULL) {
209*ec779b8eSAndroid Build Coastguard Worker ALOGE("AMediaDataSource_newUri source is null");
210*ec779b8eSAndroid Build Coastguard Worker return NULL;
211*ec779b8eSAndroid Build Coastguard Worker }
212*ec779b8eSAndroid Build Coastguard Worker ALOGI("AMediaDataSource_newUri source %s flags %u", source->toString().c_str(), source->flags());
213*ec779b8eSAndroid Build Coastguard Worker AMediaDataSource* aSource = convertDataSourceToAMediaDataSource(source);
214*ec779b8eSAndroid Build Coastguard Worker aSource->mImpl = source;
215*ec779b8eSAndroid Build Coastguard Worker aSource->mFlags = source->flags();
216*ec779b8eSAndroid Build Coastguard Worker return aSource;
217*ec779b8eSAndroid Build Coastguard Worker }
218*ec779b8eSAndroid Build Coastguard Worker
219*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_delete(AMediaDataSource * mSource)220*ec779b8eSAndroid Build Coastguard Worker void AMediaDataSource_delete(AMediaDataSource *mSource) {
221*ec779b8eSAndroid Build Coastguard Worker ALOGV("dtor");
222*ec779b8eSAndroid Build Coastguard Worker if (mSource != NULL) {
223*ec779b8eSAndroid Build Coastguard Worker delete mSource;
224*ec779b8eSAndroid Build Coastguard Worker }
225*ec779b8eSAndroid Build Coastguard Worker }
226*ec779b8eSAndroid Build Coastguard Worker
227*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_setUserdata(AMediaDataSource * mSource,void * userdata)228*ec779b8eSAndroid Build Coastguard Worker void AMediaDataSource_setUserdata(AMediaDataSource *mSource, void *userdata) {
229*ec779b8eSAndroid Build Coastguard Worker mSource->userdata = userdata;
230*ec779b8eSAndroid Build Coastguard Worker }
231*ec779b8eSAndroid Build Coastguard Worker
232*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_setReadAt(AMediaDataSource * mSource,AMediaDataSourceReadAt readAt)233*ec779b8eSAndroid Build Coastguard Worker void AMediaDataSource_setReadAt(AMediaDataSource *mSource, AMediaDataSourceReadAt readAt) {
234*ec779b8eSAndroid Build Coastguard Worker mSource->readAt = readAt;
235*ec779b8eSAndroid Build Coastguard Worker }
236*ec779b8eSAndroid Build Coastguard Worker
237*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_setGetSize(AMediaDataSource * mSource,AMediaDataSourceGetSize getSize)238*ec779b8eSAndroid Build Coastguard Worker void AMediaDataSource_setGetSize(AMediaDataSource *mSource, AMediaDataSourceGetSize getSize) {
239*ec779b8eSAndroid Build Coastguard Worker mSource->getSize = getSize;
240*ec779b8eSAndroid Build Coastguard Worker }
241*ec779b8eSAndroid Build Coastguard Worker
242*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_setClose(AMediaDataSource * mSource,AMediaDataSourceClose close)243*ec779b8eSAndroid Build Coastguard Worker void AMediaDataSource_setClose(AMediaDataSource *mSource, AMediaDataSourceClose close) {
244*ec779b8eSAndroid Build Coastguard Worker mSource->close = close;
245*ec779b8eSAndroid Build Coastguard Worker }
246*ec779b8eSAndroid Build Coastguard Worker
247*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_close(AMediaDataSource * mSource)248*ec779b8eSAndroid Build Coastguard Worker void AMediaDataSource_close(AMediaDataSource *mSource) {
249*ec779b8eSAndroid Build Coastguard Worker return mSource->close(mSource->userdata);
250*ec779b8eSAndroid Build Coastguard Worker }
251*ec779b8eSAndroid Build Coastguard Worker
252*ec779b8eSAndroid Build Coastguard Worker EXPORT
AMediaDataSource_setGetAvailableSize(AMediaDataSource * mSource,AMediaDataSourceGetAvailableSize getAvailableSize)253*ec779b8eSAndroid Build Coastguard Worker void AMediaDataSource_setGetAvailableSize(AMediaDataSource *mSource,
254*ec779b8eSAndroid Build Coastguard Worker AMediaDataSourceGetAvailableSize getAvailableSize) {
255*ec779b8eSAndroid Build Coastguard Worker mSource->getAvailableSize = getAvailableSize;
256*ec779b8eSAndroid Build Coastguard Worker }
257*ec779b8eSAndroid Build Coastguard Worker
258*ec779b8eSAndroid Build Coastguard Worker } // extern "C"
259*ec779b8eSAndroid Build Coastguard Worker
260