/*
* Copyright (C) 2009 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 "BpMediaExtractor"
#include <utils/Log.h>
#include <stdint.h>
#include <sys/types.h>
#include <binder/IPCThreadState.h>
#include <binder/Parcel.h>
#include <binder/PermissionCache.h>
#include <media/IMediaExtractor.h>
#include <media/stagefright/MetaData.h>
namespace android {
enum {
COUNTTRACKS = IBinder::FIRST_CALL_TRANSACTION,
GETTRACK,
GETTRACKMETADATA,
GETMETADATA,
FLAGS,
SETMEDIACAS,
NAME,
GETMETRICS
};
class BpMediaExtractor : public BpInterface<IMediaExtractor> {
public:
explicit BpMediaExtractor(const sp<IBinder>& impl)
: BpInterface<IMediaExtractor>(impl)
{
}
virtual size_t countTracks() {
ALOGV("countTracks");
Parcel data, reply;
data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
status_t ret = remote()->transact(COUNTTRACKS, data, &reply);
size_t numTracks = 0;
if (ret == NO_ERROR) {
numTracks = reply.readUint32();
}
return numTracks;
}
virtual sp<IMediaSource> getTrack(size_t index) {
ALOGV("getTrack(%zu)", index);
Parcel data, reply;
data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
data.writeUint32(index);
status_t ret = remote()->transact(GETTRACK, data, &reply);
if (ret == NO_ERROR) {
return interface_cast<IMediaSource>(reply.readStrongBinder());
}
return NULL;
}
virtual sp<MetaData> getTrackMetaData(
size_t index, uint32_t flags) {
ALOGV("getTrackMetaData(%zu, %u)", index, flags);
Parcel data, reply;
data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
data.writeUint32(index);
data.writeUint32(flags);
status_t ret = remote()->transact(GETTRACKMETADATA, data, &reply);
if (ret == NO_ERROR) {
return MetaData::createFromParcel(reply);
}
return NULL;
}
virtual sp<MetaData> getMetaData() {
ALOGV("getMetaData");
Parcel data, reply;
data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
status_t ret = remote()->transact(GETMETADATA, data, &reply);
if (ret == NO_ERROR) {
return MetaData::createFromParcel(reply);
}
return NULL;
}
virtual status_t getMetrics(Parcel * reply) {
Parcel data;
data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
status_t ret = remote()->transact(GETMETRICS, data, reply);
if (ret == NO_ERROR) {
return OK;
}
return UNKNOWN_ERROR;
}
virtual uint32_t flags() const {
ALOGV("flags NOT IMPLEMENTED");
return 0;
}
virtual status_t setMediaCas(const HInterfaceToken &casToken) {
ALOGV("setMediaCas");
Parcel data, reply;
data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
data.writeByteVector(casToken);
status_t err = remote()->transact(SETMEDIACAS, data, &reply);
if (err != NO_ERROR) {
return err;
}
return reply.readInt32();
}
virtual const char * name() {
ALOGV("name NOT IMPLEMENTED");
return NULL;
}
};
IMPLEMENT_META_INTERFACE(MediaExtractor, "android.media.IMediaExtractor");
#undef LOG_TAG
#define LOG_TAG "BnMediaExtractor"
status_t BnMediaExtractor::onTransact(
uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
{
switch (code) {
case COUNTTRACKS: {
ALOGV("countTracks");
CHECK_INTERFACE(IMediaExtractor, data, reply);
size_t numTracks = countTracks();
if (numTracks > INT32_MAX) {
numTracks = 0;
}
reply->writeUint32(uint32_t(numTracks));
return NO_ERROR;
}
case GETTRACK: {
ALOGV("getTrack()");
CHECK_INTERFACE(IMediaExtractor, data, reply);
uint32_t idx;
if (data.readUint32(&idx) == NO_ERROR) {
const sp<IMediaSource> track = getTrack(size_t(idx));
registerMediaSource(this, track);
return reply->writeStrongBinder(IInterface::asBinder(track));
}
return UNKNOWN_ERROR;
}
case GETTRACKMETADATA: {
ALOGV("getTrackMetaData");
CHECK_INTERFACE(IMediaExtractor, data, reply);
uint32_t idx;
uint32_t flags;
if (data.readUint32(&idx) == NO_ERROR &&
data.readUint32(&flags) == NO_ERROR) {
sp<MetaData> meta = getTrackMetaData(idx, flags);
if (meta == NULL) {
return UNKNOWN_ERROR;
}
meta->writeToParcel(*reply);
return NO_ERROR;
}
return UNKNOWN_ERROR;
}
case GETMETADATA: {
ALOGV("getMetaData");
CHECK_INTERFACE(IMediaExtractor, data, reply);
sp<MetaData> meta = getMetaData();
if (meta != NULL) {
meta->writeToParcel(*reply);
return NO_ERROR;
}
return UNKNOWN_ERROR;
}
case GETMETRICS: {
CHECK_INTERFACE(IMediaExtractor, data, reply);
status_t ret = getMetrics(reply);
return ret;
}
case SETMEDIACAS: {
ALOGV("setMediaCas");
CHECK_INTERFACE(IMediaExtractor, data, reply);
HInterfaceToken casToken;
status_t err = data.readByteVector(&casToken);
if (err != NO_ERROR) {
ALOGE("Error reading casToken from parcel");
return err;
}
reply->writeInt32(setMediaCas(casToken));
return OK;
}
default:
return BBinder::onTransact(code, data, reply, flags);
}
}
typedef struct {
String8 mime;
String8 name;
String8 sourceDescription;
pid_t owner;
wp<IMediaExtractor> extractor;
Vector<wp<IMediaSource>> tracks;
Vector<String8> trackDescriptions;
String8 toString() const;
} ExtractorInstance;
String8 ExtractorInstance::toString() const {
String8 str = name;
str.append(" for mime ");
str.append(mime);
str.append(", source ");
str.append(sourceDescription);
str.append(String8::format(", pid %d: ", owner));
if (extractor.promote() == NULL) {
str.append("deleted\n");
} else {
str.append("active\n");
}
for (size_t i = 0; i < tracks.size(); i++) {
const String8 desc = trackDescriptions.itemAt(i);
str.appendFormat(" track {%s} ", desc.string());
wp<IMediaSource> wSource = tracks.itemAt(i);
if (wSource == NULL) {
str.append(": null\n");
} else {
const sp<IMediaSource> source = wSource.promote();
if (source == NULL) {
str.append(": deleted\n");
} else {
str.appendFormat(": active\n");
}
}
}
return str;
}
static Vector<ExtractorInstance> sExtractors;
static Mutex sExtractorsLock;
void registerMediaSource(
const sp<IMediaExtractor> &ex,
const sp<IMediaSource> &source) {
Mutex::Autolock lock(sExtractorsLock);
for (size_t i = 0; i < sExtractors.size(); i++) {
ExtractorInstance &instance = sExtractors.editItemAt(i);
sp<IMediaExtractor> extractor = instance.extractor.promote();
if (extractor != NULL && extractor == ex) {
if (instance.tracks.size() > 5) {
instance.tracks.resize(5);
instance.trackDescriptions.resize(5);
}
instance.tracks.push_front(source);
if (source != NULL) {
instance.trackDescriptions.push_front(source->getFormat()->toString());
} else {
instance.trackDescriptions.push_front(String8::empty());
}
break;
}
}
}
void registerMediaExtractor(
const sp<IMediaExtractor> &extractor,
const sp<DataSource> &source,
const char *mime) {
ExtractorInstance ex;
ex.mime = mime == NULL ? "NULL" : mime;
ex.name = extractor->name();
ex.sourceDescription = source->toString();
ex.owner = IPCThreadState::self()->getCallingPid();
ex.extractor = extractor;
{
Mutex::Autolock lock(sExtractorsLock);
if (sExtractors.size() > 10) {
sExtractors.resize(10);
}
sExtractors.push_front(ex);
}
}
status_t dumpExtractors(int fd, const Vector<String16>&) {
String8 out;
const IPCThreadState* ipc = IPCThreadState::self();
const int pid = ipc->getCallingPid();
const int uid = ipc->getCallingUid();
if (!PermissionCache::checkPermission(String16("android.permission.DUMP"), pid, uid)) {
out.appendFormat("Permission Denial: "
"can't dump MediaExtractor from pid=%d, uid=%d\n", pid, uid);
} else {
out.append("Recent extractors, most recent first:\n");
{
Mutex::Autolock lock(sExtractorsLock);
for (size_t i = 0; i < sExtractors.size(); i++) {
const ExtractorInstance &instance = sExtractors.itemAt(i);
out.append(" ");
out.append(instance.toString());
}
}
}
write(fd, out.string(), out.size());
return OK;
}
} // namespace android