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.
495 lines
15 KiB
495 lines
15 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.
|
|
*/
|
|
|
|
#ifndef MEDIA_EXTRACTOR_PLUGIN_HELPER_H_
|
|
|
|
#define MEDIA_EXTRACTOR_PLUGIN_HELPER_H_
|
|
|
|
#include <arpa/inet.h>
|
|
#include <stdio.h>
|
|
#include <map>
|
|
|
|
#include <utils/Errors.h>
|
|
#include <utils/Log.h>
|
|
#include <utils/RefBase.h>
|
|
#include <media/MediaExtractorPluginApi.h>
|
|
#include <media/NdkMediaFormat.h>
|
|
|
|
namespace android {
|
|
|
|
class DataSourceBase;
|
|
class MetaDataBase;
|
|
struct MediaTrack;
|
|
|
|
|
|
class MediaTrackHelper;
|
|
|
|
class MediaBufferHelper {
|
|
private:
|
|
friend CMediaTrack *wrap(MediaTrackHelper *);
|
|
CMediaBuffer *mBuffer;
|
|
public:
|
|
MediaBufferHelper(CMediaBuffer *buf) {
|
|
mBuffer = buf;
|
|
}
|
|
|
|
virtual ~MediaBufferHelper() {}
|
|
|
|
virtual void release() {
|
|
mBuffer->release(mBuffer->handle);
|
|
}
|
|
|
|
virtual void* data() {
|
|
return mBuffer->data(mBuffer->handle);
|
|
}
|
|
|
|
virtual size_t size() {
|
|
return mBuffer->size(mBuffer->handle);
|
|
}
|
|
|
|
virtual size_t range_offset() {
|
|
return mBuffer->range_offset(mBuffer->handle);
|
|
}
|
|
|
|
virtual size_t range_length() {
|
|
return mBuffer->range_length(mBuffer->handle);
|
|
}
|
|
|
|
virtual void set_range(size_t offset, size_t length) {
|
|
mBuffer->set_range(mBuffer->handle, offset, length);
|
|
}
|
|
virtual AMediaFormat *meta_data() {
|
|
return mBuffer->meta_data(mBuffer->handle);
|
|
}
|
|
};
|
|
|
|
class MediaBufferGroupHelper {
|
|
private:
|
|
CMediaBufferGroup *mGroup;
|
|
std::map<CMediaBuffer*, MediaBufferHelper*> mBufferHelpers;
|
|
public:
|
|
MediaBufferGroupHelper(CMediaBufferGroup *group) {
|
|
mGroup = group;
|
|
}
|
|
~MediaBufferGroupHelper() {
|
|
// delete all entries in map
|
|
ALOGV("buffergroup %p map has %zu entries", this, mBufferHelpers.size());
|
|
for (auto it = mBufferHelpers.begin(); it != mBufferHelpers.end(); ++it) {
|
|
delete it->second;
|
|
}
|
|
}
|
|
bool init(size_t buffers, size_t buffer_size, size_t growthLimit = 0) {
|
|
return mGroup->init(mGroup->handle, buffers, buffer_size, growthLimit);
|
|
}
|
|
void add_buffer(size_t size) {
|
|
mGroup->add_buffer(mGroup->handle, size);
|
|
}
|
|
media_status_t acquire_buffer(
|
|
MediaBufferHelper **buffer, bool nonBlocking = false, size_t requestedSize = 0) {
|
|
CMediaBuffer *buf = nullptr;
|
|
media_status_t ret =
|
|
mGroup->acquire_buffer(mGroup->handle, &buf, nonBlocking, requestedSize);
|
|
if (ret == AMEDIA_OK && buf != nullptr) {
|
|
auto helper = mBufferHelpers.find(buf);
|
|
if (helper == mBufferHelpers.end()) {
|
|
MediaBufferHelper* newHelper = new MediaBufferHelper(buf);
|
|
mBufferHelpers.insert(std::make_pair(buf, newHelper));
|
|
*buffer = newHelper;
|
|
} else {
|
|
*buffer = helper->second;
|
|
}
|
|
} else {
|
|
*buffer = nullptr;
|
|
}
|
|
return ret;
|
|
}
|
|
bool has_buffers() {
|
|
return mGroup->has_buffers(mGroup->handle);
|
|
}
|
|
};
|
|
|
|
class MediaTrackHelper {
|
|
public:
|
|
MediaTrackHelper() : mBufferGroup(nullptr) {
|
|
}
|
|
virtual ~MediaTrackHelper() {
|
|
delete mBufferGroup;
|
|
}
|
|
virtual media_status_t start() = 0;
|
|
virtual media_status_t stop() = 0;
|
|
virtual media_status_t getFormat(AMediaFormat *format) = 0;
|
|
|
|
class ReadOptions {
|
|
public:
|
|
enum SeekMode : int32_t {
|
|
SEEK_PREVIOUS_SYNC,
|
|
SEEK_NEXT_SYNC,
|
|
SEEK_CLOSEST_SYNC,
|
|
SEEK_CLOSEST,
|
|
SEEK_FRAME_INDEX,
|
|
};
|
|
|
|
ReadOptions(uint32_t options, int64_t seekPosUs) {
|
|
mOptions = options;
|
|
mSeekPosUs = seekPosUs;
|
|
}
|
|
bool getSeekTo(int64_t *time_us, SeekMode *mode) const {
|
|
if ((mOptions & CMediaTrackReadOptions::SEEK) == 0) {
|
|
return false;
|
|
}
|
|
*time_us = mSeekPosUs;
|
|
*mode = (SeekMode) (mOptions & 7);
|
|
return true;
|
|
}
|
|
bool getNonBlocking() const {
|
|
return mOptions & CMediaTrackReadOptions::NONBLOCKING;
|
|
}
|
|
private:
|
|
uint32_t mOptions;
|
|
int64_t mSeekPosUs;
|
|
};
|
|
|
|
virtual media_status_t read(
|
|
MediaBufferHelper **buffer, const ReadOptions *options = NULL) = 0;
|
|
virtual bool supportsNonBlockingRead() { return false; }
|
|
protected:
|
|
friend CMediaTrack *wrap(MediaTrackHelper *track);
|
|
MediaBufferGroupHelper *mBufferGroup;
|
|
};
|
|
|
|
inline CMediaTrack *wrap(MediaTrackHelper *track) {
|
|
if (track == nullptr) {
|
|
return nullptr;
|
|
}
|
|
CMediaTrack *wrapper = (CMediaTrack*) malloc(sizeof(CMediaTrack));
|
|
wrapper->data = track;
|
|
wrapper->free = [](void *data) -> void {
|
|
delete (MediaTrackHelper*)(data);
|
|
};
|
|
wrapper->start = [](void *data, CMediaBufferGroup *bufferGroup) -> media_status_t {
|
|
if (((MediaTrackHelper*)data)->mBufferGroup) {
|
|
// this shouldn't happen, but handle it anyway
|
|
delete ((MediaTrackHelper*)data)->mBufferGroup;
|
|
}
|
|
((MediaTrackHelper*)data)->mBufferGroup = new MediaBufferGroupHelper(bufferGroup);
|
|
return ((MediaTrackHelper*)data)->start();
|
|
};
|
|
wrapper->stop = [](void *data) -> media_status_t {
|
|
return ((MediaTrackHelper*)data)->stop();
|
|
};
|
|
wrapper->getFormat = [](void *data, AMediaFormat *meta) -> media_status_t {
|
|
return ((MediaTrackHelper*)data)->getFormat(meta);
|
|
};
|
|
wrapper->read = [](void *data, CMediaBuffer **buffer, uint32_t options, int64_t seekPosUs)
|
|
-> media_status_t {
|
|
MediaTrackHelper::ReadOptions opts(options, seekPosUs);
|
|
MediaBufferHelper *buf = NULL;
|
|
media_status_t ret = ((MediaTrackHelper*)data)->read(&buf, &opts);
|
|
if (ret == AMEDIA_OK && buf != nullptr) {
|
|
*buffer = buf->mBuffer;
|
|
}
|
|
return ret;
|
|
};
|
|
wrapper->supportsNonBlockingRead = [](void *data) -> bool {
|
|
return ((MediaTrackHelper*)data)->supportsNonBlockingRead();
|
|
};
|
|
return wrapper;
|
|
}
|
|
|
|
|
|
// extractor plugins can derive from this class which looks remarkably
|
|
// like MediaExtractor and can be easily wrapped in the required C API
|
|
class MediaExtractorPluginHelper
|
|
{
|
|
public:
|
|
virtual ~MediaExtractorPluginHelper() {}
|
|
virtual size_t countTracks() = 0;
|
|
virtual MediaTrackHelper *getTrack(size_t index) = 0;
|
|
|
|
enum GetTrackMetaDataFlags {
|
|
kIncludeExtensiveMetaData = 1
|
|
};
|
|
virtual media_status_t getTrackMetaData(
|
|
AMediaFormat *meta,
|
|
size_t index, uint32_t flags = 0) = 0;
|
|
|
|
// Return container specific meta-data. The default implementation
|
|
// returns an empty metadata object.
|
|
virtual media_status_t getMetaData(AMediaFormat *meta) = 0;
|
|
|
|
enum Flags {
|
|
CAN_SEEK_BACKWARD = 1, // the "seek 10secs back button"
|
|
CAN_SEEK_FORWARD = 2, // the "seek 10secs forward button"
|
|
CAN_PAUSE = 4,
|
|
CAN_SEEK = 8, // the "seek bar"
|
|
};
|
|
|
|
// If subclasses do _not_ override this, the default is
|
|
// CAN_SEEK_BACKWARD | CAN_SEEK_FORWARD | CAN_SEEK | CAN_PAUSE
|
|
virtual uint32_t flags() const {
|
|
return CAN_SEEK_BACKWARD | CAN_SEEK_FORWARD | CAN_SEEK | CAN_PAUSE;
|
|
};
|
|
|
|
virtual media_status_t setMediaCas(const uint8_t* /*casToken*/, size_t /*size*/) {
|
|
return AMEDIA_ERROR_INVALID_OPERATION;
|
|
}
|
|
|
|
virtual const char * name() { return "<unspecified>"; }
|
|
|
|
protected:
|
|
MediaExtractorPluginHelper() {}
|
|
|
|
private:
|
|
MediaExtractorPluginHelper(const MediaExtractorPluginHelper &);
|
|
MediaExtractorPluginHelper &operator=(const MediaExtractorPluginHelper &);
|
|
};
|
|
|
|
inline CMediaExtractor *wrap(MediaExtractorPluginHelper *extractor) {
|
|
CMediaExtractor *wrapper = (CMediaExtractor*) malloc(sizeof(CMediaExtractor));
|
|
wrapper->data = extractor;
|
|
wrapper->free = [](void *data) -> void {
|
|
delete (MediaExtractorPluginHelper*)(data);
|
|
};
|
|
wrapper->countTracks = [](void *data) -> size_t {
|
|
return ((MediaExtractorPluginHelper*)data)->countTracks();
|
|
};
|
|
wrapper->getTrack = [](void *data, size_t index) -> CMediaTrack* {
|
|
return wrap(((MediaExtractorPluginHelper*)data)->getTrack(index));
|
|
};
|
|
wrapper->getTrackMetaData = [](
|
|
void *data,
|
|
AMediaFormat *meta,
|
|
size_t index, uint32_t flags) -> media_status_t {
|
|
return ((MediaExtractorPluginHelper*)data)->getTrackMetaData(meta, index, flags);
|
|
};
|
|
wrapper->getMetaData = [](
|
|
void *data,
|
|
AMediaFormat *meta) -> media_status_t {
|
|
return ((MediaExtractorPluginHelper*)data)->getMetaData(meta);
|
|
};
|
|
wrapper->flags = [](
|
|
void *data) -> uint32_t {
|
|
return ((MediaExtractorPluginHelper*)data)->flags();
|
|
};
|
|
wrapper->setMediaCas = [](
|
|
void *data, const uint8_t *casToken, size_t size) -> media_status_t {
|
|
return ((MediaExtractorPluginHelper*)data)->setMediaCas(casToken, size);
|
|
};
|
|
wrapper->name = [](
|
|
void *data) -> const char * {
|
|
return ((MediaExtractorPluginHelper*)data)->name();
|
|
};
|
|
return wrapper;
|
|
}
|
|
|
|
/* adds some convience methods */
|
|
class DataSourceHelper {
|
|
public:
|
|
explicit DataSourceHelper(CDataSource *csource) {
|
|
mSource = csource;
|
|
}
|
|
|
|
explicit DataSourceHelper(DataSourceHelper *source) {
|
|
mSource = source->mSource;
|
|
}
|
|
|
|
virtual ~DataSourceHelper() {}
|
|
|
|
virtual ssize_t readAt(off64_t offset, void *data, size_t size) {
|
|
return mSource->readAt(mSource->handle, offset, data, size);
|
|
}
|
|
|
|
virtual status_t getSize(off64_t *size) {
|
|
return mSource->getSize(mSource->handle, size);
|
|
}
|
|
|
|
bool getUri(char *uriString, size_t bufferSize) {
|
|
return mSource->getUri(mSource->handle, uriString, bufferSize);
|
|
}
|
|
|
|
virtual uint32_t flags() {
|
|
return mSource->flags(mSource->handle);
|
|
}
|
|
|
|
// Convenience methods:
|
|
bool getUInt16(off64_t offset, uint16_t *x) {
|
|
*x = 0;
|
|
|
|
uint8_t byte[2];
|
|
if (readAt(offset, byte, 2) != 2) {
|
|
return false;
|
|
}
|
|
|
|
*x = (byte[0] << 8) | byte[1];
|
|
|
|
return true;
|
|
}
|
|
|
|
// 3 byte int, returned as a 32-bit int
|
|
bool getUInt24(off64_t offset, uint32_t *x) {
|
|
*x = 0;
|
|
|
|
uint8_t byte[3];
|
|
if (readAt(offset, byte, 3) != 3) {
|
|
return false;
|
|
}
|
|
|
|
*x = (byte[0] << 16) | (byte[1] << 8) | byte[2];
|
|
|
|
return true;
|
|
}
|
|
|
|
bool getUInt32(off64_t offset, uint32_t *x) {
|
|
*x = 0;
|
|
|
|
uint32_t tmp;
|
|
if (readAt(offset, &tmp, 4) != 4) {
|
|
return false;
|
|
}
|
|
|
|
*x = ntohl(tmp);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool getUInt64(off64_t offset, uint64_t *x) {
|
|
*x = 0;
|
|
|
|
uint64_t tmp;
|
|
if (readAt(offset, &tmp, 8) != 8) {
|
|
return false;
|
|
}
|
|
|
|
*x = ((uint64_t)ntohl(tmp & 0xffffffff) << 32) | ntohl(tmp >> 32);
|
|
|
|
return true;
|
|
}
|
|
|
|
// read either int<N> or int<2N> into a uint<2N>_t, size is the int size in bytes.
|
|
bool getUInt16Var(off64_t offset, uint16_t *x, size_t size) {
|
|
if (size == 2) {
|
|
return getUInt16(offset, x);
|
|
}
|
|
if (size == 1) {
|
|
uint8_t tmp;
|
|
if (readAt(offset, &tmp, 1) == 1) {
|
|
*x = tmp;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool getUInt32Var(off64_t offset, uint32_t *x, size_t size) {
|
|
if (size == 4) {
|
|
return getUInt32(offset, x);
|
|
}
|
|
if (size == 2) {
|
|
uint16_t tmp;
|
|
if (getUInt16(offset, &tmp)) {
|
|
*x = tmp;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool getUInt64Var(off64_t offset, uint64_t *x, size_t size) {
|
|
if (size == 8) {
|
|
return getUInt64(offset, x);
|
|
}
|
|
if (size == 4) {
|
|
uint32_t tmp;
|
|
if (getUInt32(offset, &tmp)) {
|
|
*x = tmp;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
protected:
|
|
CDataSource *mSource;
|
|
};
|
|
|
|
|
|
|
|
// helpers to create a media_uuid_t from a string literal
|
|
|
|
// purposely not defined anywhere so that this will fail to link if
|
|
// expressions below are not evaluated at compile time
|
|
int invalid_uuid_string(const char *);
|
|
|
|
template <typename T, size_t N>
|
|
constexpr uint8_t _digitAt_(const T (&s)[N], const size_t n) {
|
|
return s[n] >= '0' && s[n] <= '9' ? s[n] - '0'
|
|
: s[n] >= 'a' && s[n] <= 'f' ? s[n] - 'a' + 10
|
|
: s[n] >= 'A' && s[n] <= 'F' ? s[n] - 'A' + 10
|
|
: invalid_uuid_string("uuid: bad digits");
|
|
}
|
|
|
|
template <typename T, size_t N>
|
|
constexpr uint8_t _hexByteAt_(const T (&s)[N], size_t n) {
|
|
return (_digitAt_(s, n) << 4) + _digitAt_(s, n + 1);
|
|
}
|
|
|
|
constexpr bool _assertIsDash_(char c) {
|
|
return c == '-' ? true : invalid_uuid_string("Wrong format");
|
|
}
|
|
|
|
template <size_t N>
|
|
constexpr media_uuid_t constUUID(const char (&s) [N]) {
|
|
static_assert(N == 37, "uuid: wrong length");
|
|
return
|
|
_assertIsDash_(s[8]),
|
|
_assertIsDash_(s[13]),
|
|
_assertIsDash_(s[18]),
|
|
_assertIsDash_(s[23]),
|
|
media_uuid_t {{
|
|
_hexByteAt_(s, 0),
|
|
_hexByteAt_(s, 2),
|
|
_hexByteAt_(s, 4),
|
|
_hexByteAt_(s, 6),
|
|
_hexByteAt_(s, 9),
|
|
_hexByteAt_(s, 11),
|
|
_hexByteAt_(s, 14),
|
|
_hexByteAt_(s, 16),
|
|
_hexByteAt_(s, 19),
|
|
_hexByteAt_(s, 21),
|
|
_hexByteAt_(s, 24),
|
|
_hexByteAt_(s, 26),
|
|
_hexByteAt_(s, 28),
|
|
_hexByteAt_(s, 30),
|
|
_hexByteAt_(s, 32),
|
|
_hexByteAt_(s, 34),
|
|
}};
|
|
}
|
|
// Convenience macro to create a media_uuid_t from a string literal, which should
|
|
// be formatted as "12345678-1234-1234-1234-123456789abc", as generated by
|
|
// e.g. https://www.uuidgenerator.net/ or the 'uuidgen' linux command.
|
|
// Hex digits may be upper or lower case.
|
|
//
|
|
// The macro call is otherwise equivalent to specifying the structure directly
|
|
// (e.g. UUID("7d613858-5837-4a38-84c5-332d1cddee27") is the same as
|
|
// {{0x7d, 0x61, 0x38, 0x58, 0x58, 0x37, 0x4a, 0x38,
|
|
// 0x84, 0xc5, 0x33, 0x2d, 0x1c, 0xdd, 0xee, 0x27}})
|
|
|
|
#define UUID(str) []{ constexpr media_uuid_t uuid = constUUID(str); return uuid; }()
|
|
|
|
} // namespace android
|
|
|
|
#endif // MEDIA_EXTRACTOR_PLUGIN_HELPER_H_
|