/*
**
** 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