/*
**
** Copyright 2007, 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_TAG "IMediaExtractorService"
//#define LOG_NDEBUG 0
#include <utils/Log.h>
#include <stdint.h>
#include <sys/types.h>
#include <binder/Parcel.h>
#include <media/IMediaExtractorService.h>
#include <media/MediaExtractor.h>
namespace android {
enum {
MAKE_EXTRACTOR = IBinder::FIRST_CALL_TRANSACTION,
MAKE_IDATA_SOURCE_FD,
};
class BpMediaExtractorService : public BpInterface<IMediaExtractorService>
{
public:
explicit BpMediaExtractorService(const sp<IBinder>& impl)
: BpInterface<IMediaExtractorService>(impl)
{
}
virtual sp<IMediaExtractor> makeExtractor(const sp<IDataSource> &source, const char *mime) {
Parcel data, reply;
data.writeInterfaceToken(IMediaExtractorService::getInterfaceDescriptor());
data.writeStrongBinder(IInterface::asBinder(source));
if (mime != NULL) {
data.writeCString(mime);
}
status_t ret = remote()->transact(MAKE_EXTRACTOR, data, &reply);
if (ret == NO_ERROR) {
return interface_cast<IMediaExtractor>(reply.readStrongBinder());
}
return NULL;
}
virtual sp<IDataSource> makeIDataSource(int fd, int64_t offset, int64_t length)
{
Parcel data, reply;
data.writeInterfaceToken(IMediaExtractorService::getInterfaceDescriptor());
data.writeFileDescriptor(fd);
data.writeInt64(offset);
data.writeInt64(length);
status_t ret = remote()->transact(MAKE_IDATA_SOURCE_FD, data, &reply);
ALOGV("fd:%d offset:%lld length:%lld ret:%d",
fd, (long long)offset, (long long)length, ret);
if (ret == NO_ERROR) {
return interface_cast<IDataSource>(reply.readStrongBinder());
}
return nullptr;
}
};
IMPLEMENT_META_INTERFACE(MediaExtractorService, "android.media.IMediaExtractorService");
// ----------------------------------------------------------------------
status_t BnMediaExtractorService::onTransact(
uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
{
switch (code) {
case MAKE_EXTRACTOR: {
CHECK_INTERFACE(IMediaExtractorService, data, reply);
sp<IBinder> b;
status_t ret = data.readStrongBinder(&b);
if (ret != NO_ERROR || b == NULL) {
ALOGE("Error reading source from parcel");
return ret;
}
// If we make an extractor through Binder, enabled shared memory
// for MediaBuffers for this process.
MediaBuffer::useSharedMemory();
sp<IDataSource> source = interface_cast<IDataSource>(b);
const char *mime = data.readCString();
sp<IMediaExtractor> ex = makeExtractor(source, mime);
reply->writeStrongBinder(IInterface::asBinder(ex));
return NO_ERROR;
}
case MAKE_IDATA_SOURCE_FD: {
CHECK_INTERFACE(IMediaExtractorService, data, reply);
const int fd = dup(data.readFileDescriptor()); // -1 fd checked in makeIDataSource
const int64_t offset = data.readInt64();
const int64_t length = data.readInt64();
ALOGV("fd %d offset%lld length:%lld", fd, (long long)offset, (long long)length);
sp<IDataSource> source = makeIDataSource(fd, offset, length);
reply->writeStrongBinder(IInterface::asBinder(source));
// The FileSource closes the descriptor, so if it is not created
// we need to close the descriptor explicitly.
if (source.get() == nullptr && fd != -1) {
close(fd);
}
return NO_ERROR;
}
default:
return BBinder::onTransact(code, data, reply, flags);
}
}
// ----------------------------------------------------------------------------
} // namespace android