You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
260 lines
7.3 KiB
260 lines
7.3 KiB
/*
|
|
* Copyright (C) 2018 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 "NdkMediaDataSource"
|
|
|
|
#include "NdkJavaVMHelperPriv.h"
|
|
#include "NdkMediaDataSourcePriv.h"
|
|
|
|
#include <inttypes.h>
|
|
#include <jni.h>
|
|
#include <unistd.h>
|
|
|
|
#include <android_runtime/AndroidRuntime.h>
|
|
#include <android_util_Binder.h>
|
|
#include <cutils/properties.h>
|
|
#include <datasource/DataSourceFactory.h>
|
|
#include <datasource/HTTPBase.h>
|
|
#include <datasource/NuCachedSource2.h>
|
|
#include <media/IMediaHTTPService.h>
|
|
#include <media/NdkMediaError.h>
|
|
#include <media/NdkMediaDataSource.h>
|
|
#include <media/stagefright/InterfaceUtils.h>
|
|
#include <utils/Log.h>
|
|
#include <utils/StrongPointer.h>
|
|
|
|
#include "NdkMediaDataSourceCallbacksPriv.h"
|
|
|
|
|
|
using namespace android;
|
|
|
|
struct AMediaDataSource {
|
|
void *userdata;
|
|
AMediaDataSourceReadAt readAt;
|
|
AMediaDataSourceGetSize getSize;
|
|
AMediaDataSourceClose close;
|
|
AMediaDataSourceGetAvailableSize getAvailableSize;
|
|
sp<DataSource> mImpl;
|
|
uint32_t mFlags;
|
|
};
|
|
|
|
NdkDataSource::NdkDataSource(AMediaDataSource *dataSource)
|
|
: mDataSource(AMediaDataSource_new()) {
|
|
AMediaDataSource_setReadAt(mDataSource, dataSource->readAt);
|
|
AMediaDataSource_setGetSize(mDataSource, dataSource->getSize);
|
|
AMediaDataSource_setClose(mDataSource, dataSource->close);
|
|
AMediaDataSource_setUserdata(mDataSource, dataSource->userdata);
|
|
AMediaDataSource_setGetAvailableSize(mDataSource, dataSource->getAvailableSize);
|
|
mDataSource->mImpl = dataSource->mImpl;
|
|
mDataSource->mFlags = dataSource->mFlags;
|
|
}
|
|
|
|
NdkDataSource::~NdkDataSource() {
|
|
AMediaDataSource_delete(mDataSource);
|
|
}
|
|
|
|
status_t NdkDataSource::initCheck() const {
|
|
return OK;
|
|
}
|
|
|
|
uint32_t NdkDataSource::flags() {
|
|
return mDataSource->mFlags;
|
|
}
|
|
|
|
ssize_t NdkDataSource::readAt(off64_t offset, void *data, size_t size) {
|
|
Mutex::Autolock l(mLock);
|
|
if (mDataSource->readAt == NULL || mDataSource->userdata == NULL) {
|
|
return -1;
|
|
}
|
|
return mDataSource->readAt(mDataSource->userdata, offset, data, size);
|
|
}
|
|
|
|
status_t NdkDataSource::getSize(off64_t *size) {
|
|
Mutex::Autolock l(mLock);
|
|
if (mDataSource->getSize == NULL || mDataSource->userdata == NULL) {
|
|
return NO_INIT;
|
|
}
|
|
if (size != NULL) {
|
|
*size = mDataSource->getSize(mDataSource->userdata);
|
|
}
|
|
return OK;
|
|
}
|
|
|
|
String8 NdkDataSource::toString() {
|
|
return String8::format("NdkDataSource(pid %d, uid %d)", getpid(), getuid());
|
|
}
|
|
|
|
String8 NdkDataSource::getMIMEType() const {
|
|
return String8("application/octet-stream");
|
|
}
|
|
|
|
void NdkDataSource::close() {
|
|
if (mDataSource->close != NULL && mDataSource->userdata != NULL) {
|
|
mDataSource->close(mDataSource->userdata);
|
|
}
|
|
}
|
|
|
|
status_t NdkDataSource::getAvailableSize(off64_t offset, off64_t *sizeptr) {
|
|
off64_t size = -1;
|
|
if (mDataSource->getAvailableSize != NULL
|
|
&& mDataSource->userdata != NULL
|
|
&& sizeptr != NULL) {
|
|
size = mDataSource->getAvailableSize(mDataSource->userdata, offset);
|
|
*sizeptr = size;
|
|
}
|
|
return size >= 0 ? OK : UNKNOWN_ERROR;
|
|
}
|
|
|
|
static sp<MediaHTTPService> createMediaHttpServiceFromJavaObj(JNIEnv *env, jobject obj) {
|
|
if (obj == NULL) {
|
|
return NULL;
|
|
}
|
|
return interface_cast<IMediaHTTPService>(ibinderForJavaObject(env, obj));
|
|
}
|
|
|
|
static sp<MediaHTTPService> createMediaHttpServiceTemplate(
|
|
JNIEnv *env,
|
|
const char *uri,
|
|
const char *clazz,
|
|
const char *method,
|
|
const char *signature) {
|
|
jobject service = NULL;
|
|
if (env == NULL) {
|
|
ALOGE("http service must be created from Java thread");
|
|
return NULL;
|
|
}
|
|
|
|
jclass mediahttpclass = env->FindClass(clazz);
|
|
if (mediahttpclass == NULL) {
|
|
ALOGE("can't find Media(2)HttpService");
|
|
env->ExceptionClear();
|
|
return NULL;
|
|
}
|
|
|
|
jmethodID mediaHttpCreateMethod = env->GetStaticMethodID(mediahttpclass, method, signature);
|
|
if (mediaHttpCreateMethod == NULL) {
|
|
ALOGE("can't find method");
|
|
env->ExceptionClear();
|
|
return NULL;
|
|
}
|
|
|
|
jstring juri = env->NewStringUTF(uri);
|
|
|
|
service = env->CallStaticObjectMethod(mediahttpclass, mediaHttpCreateMethod, juri);
|
|
env->DeleteLocalRef(juri);
|
|
|
|
env->ExceptionClear();
|
|
sp<MediaHTTPService> httpService = createMediaHttpServiceFromJavaObj(env, service);
|
|
return httpService;
|
|
|
|
}
|
|
|
|
sp<MediaHTTPService> createMediaHttpService(const char *uri) {
|
|
|
|
JNIEnv *env;
|
|
const char *clazz, *method, *signature;
|
|
|
|
env = NdkJavaVMHelper::getJNIEnv();
|
|
|
|
clazz = "android/media/MediaHTTPService";
|
|
method = "createHttpServiceBinderIfNecessary";
|
|
signature = "(Ljava/lang/String;)Landroid/os/IBinder;";
|
|
|
|
return createMediaHttpServiceTemplate(env, uri, clazz, method, signature);
|
|
|
|
}
|
|
|
|
extern "C" {
|
|
|
|
EXPORT
|
|
AMediaDataSource* AMediaDataSource_new() {
|
|
AMediaDataSource *mSource = new AMediaDataSource();
|
|
mSource->userdata = NULL;
|
|
mSource->readAt = NULL;
|
|
mSource->getSize = NULL;
|
|
mSource->close = NULL;
|
|
return mSource;
|
|
}
|
|
|
|
EXPORT
|
|
AMediaDataSource* AMediaDataSource_newUri(
|
|
const char *uri,
|
|
int numheaders,
|
|
const char * const *key_values) {
|
|
|
|
sp<MediaHTTPService> service = createMediaHttpService(uri);
|
|
KeyedVector<String8, String8> headers;
|
|
for (int i = 0; i < numheaders; ++i) {
|
|
String8 key8(key_values[i * 2]);
|
|
String8 value8(key_values[i * 2 + 1]);
|
|
headers.add(key8, value8);
|
|
}
|
|
|
|
sp<DataSource> source = DataSourceFactory::getInstance()->CreateFromURI(service, uri, &headers);
|
|
if (source == NULL) {
|
|
ALOGE("AMediaDataSource_newUri source is null");
|
|
return NULL;
|
|
}
|
|
ALOGI("AMediaDataSource_newUri source %s flags %u", source->toString().c_str(), source->flags());
|
|
AMediaDataSource* aSource = convertDataSourceToAMediaDataSource(source);
|
|
aSource->mImpl = source;
|
|
aSource->mFlags = source->flags();
|
|
return aSource;
|
|
}
|
|
|
|
EXPORT
|
|
void AMediaDataSource_delete(AMediaDataSource *mSource) {
|
|
ALOGV("dtor");
|
|
if (mSource != NULL) {
|
|
delete mSource;
|
|
}
|
|
}
|
|
|
|
EXPORT
|
|
void AMediaDataSource_setUserdata(AMediaDataSource *mSource, void *userdata) {
|
|
mSource->userdata = userdata;
|
|
}
|
|
|
|
EXPORT
|
|
void AMediaDataSource_setReadAt(AMediaDataSource *mSource, AMediaDataSourceReadAt readAt) {
|
|
mSource->readAt = readAt;
|
|
}
|
|
|
|
EXPORT
|
|
void AMediaDataSource_setGetSize(AMediaDataSource *mSource, AMediaDataSourceGetSize getSize) {
|
|
mSource->getSize = getSize;
|
|
}
|
|
|
|
EXPORT
|
|
void AMediaDataSource_setClose(AMediaDataSource *mSource, AMediaDataSourceClose close) {
|
|
mSource->close = close;
|
|
}
|
|
|
|
EXPORT
|
|
void AMediaDataSource_close(AMediaDataSource *mSource) {
|
|
return mSource->close(mSource->userdata);
|
|
}
|
|
|
|
EXPORT
|
|
void AMediaDataSource_setGetAvailableSize(AMediaDataSource *mSource,
|
|
AMediaDataSourceGetAvailableSize getAvailableSize) {
|
|
mSource->getAvailableSize = getAvailableSize;
|
|
}
|
|
|
|
} // extern "C"
|
|
|