/* * Copyright 2017, 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 "NdkWrapper" #include <media/NdkWrapper.h> #include <android/native_window.h> #include <log/log.h> #include <media/NdkMediaCodec.h> #include <media/NdkMediaCrypto.h> #include <media/NdkMediaDrm.h> #include <media/NdkMediaFormat.h> #include <media/NdkMediaExtractor.h> #include <media/stagefright/MetaData.h> #include <media/stagefright/foundation/ABuffer.h> #include <media/stagefright/foundation/AMessage.h> #include <utils/Errors.h> // TODO: remove forward declaration when AMediaExtractor_disconnect is offcially added to NDK #ifdef __cplusplus extern "C" { #endif media_status_t AMediaExtractor_disconnect(AMediaExtractor *); #ifdef __cplusplus } // extern "C" #endif namespace android { static const size_t kAESBlockSize = 16; // AES_BLOCK_SIZE static const char *AMediaFormatKeyGroupInt32[] = { AMEDIAFORMAT_KEY_AAC_DRC_ATTENUATION_FACTOR, AMEDIAFORMAT_KEY_AAC_DRC_BOOST_FACTOR, AMEDIAFORMAT_KEY_AAC_DRC_HEAVY_COMPRESSION, AMEDIAFORMAT_KEY_AAC_DRC_TARGET_REFERENCE_LEVEL, AMEDIAFORMAT_KEY_AAC_ENCODED_TARGET_LEVEL, AMEDIAFORMAT_KEY_AAC_MAX_OUTPUT_CHANNEL_COUNT, AMEDIAFORMAT_KEY_AAC_PROFILE, AMEDIAFORMAT_KEY_AAC_SBR_MODE, AMEDIAFORMAT_KEY_AUDIO_SESSION_ID, AMEDIAFORMAT_KEY_BITRATE_MODE, AMEDIAFORMAT_KEY_BIT_RATE, AMEDIAFORMAT_KEY_CAPTURE_RATE, AMEDIAFORMAT_KEY_CHANNEL_COUNT, AMEDIAFORMAT_KEY_CHANNEL_MASK, AMEDIAFORMAT_KEY_COLOR_FORMAT, AMEDIAFORMAT_KEY_COLOR_RANGE, AMEDIAFORMAT_KEY_COLOR_STANDARD, AMEDIAFORMAT_KEY_COLOR_TRANSFER, AMEDIAFORMAT_KEY_COMPLEXITY, AMEDIAFORMAT_KEY_FLAC_COMPRESSION_LEVEL, AMEDIAFORMAT_KEY_GRID_COLUMNS, AMEDIAFORMAT_KEY_GRID_ROWS, AMEDIAFORMAT_KEY_HEIGHT, AMEDIAFORMAT_KEY_INTRA_REFRESH_PERIOD, AMEDIAFORMAT_KEY_IS_ADTS, AMEDIAFORMAT_KEY_IS_AUTOSELECT, AMEDIAFORMAT_KEY_IS_DEFAULT, AMEDIAFORMAT_KEY_IS_FORCED_SUBTITLE, AMEDIAFORMAT_KEY_LATENCY, AMEDIAFORMAT_KEY_LEVEL, AMEDIAFORMAT_KEY_MAX_HEIGHT, AMEDIAFORMAT_KEY_MAX_INPUT_SIZE, AMEDIAFORMAT_KEY_MAX_WIDTH, AMEDIAFORMAT_KEY_PCM_ENCODING, AMEDIAFORMAT_KEY_PRIORITY, AMEDIAFORMAT_KEY_PROFILE, AMEDIAFORMAT_KEY_PUSH_BLANK_BUFFERS_ON_STOP, AMEDIAFORMAT_KEY_ROTATION, AMEDIAFORMAT_KEY_SAMPLE_RATE, AMEDIAFORMAT_KEY_SLICE_HEIGHT, AMEDIAFORMAT_KEY_STRIDE, AMEDIAFORMAT_KEY_TRACK_ID, AMEDIAFORMAT_KEY_WIDTH, AMEDIAFORMAT_KEY_DISPLAY_HEIGHT, AMEDIAFORMAT_KEY_DISPLAY_WIDTH, AMEDIAFORMAT_KEY_TEMPORAL_LAYER_ID, AMEDIAFORMAT_KEY_TILE_HEIGHT, AMEDIAFORMAT_KEY_TILE_WIDTH, AMEDIAFORMAT_KEY_TRACK_INDEX, }; static const char *AMediaFormatKeyGroupInt64[] = { AMEDIAFORMAT_KEY_DURATION, AMEDIAFORMAT_KEY_REPEAT_PREVIOUS_FRAME_AFTER, AMEDIAFORMAT_KEY_TIME_US, }; static const char *AMediaFormatKeyGroupString[] = { AMEDIAFORMAT_KEY_LANGUAGE, AMEDIAFORMAT_KEY_MIME, AMEDIAFORMAT_KEY_TEMPORAL_LAYERING, }; static const char *AMediaFormatKeyGroupBuffer[] = { AMEDIAFORMAT_KEY_HDR_STATIC_INFO, AMEDIAFORMAT_KEY_SEI, AMEDIAFORMAT_KEY_MPEG_USER_DATA, }; static const char *AMediaFormatKeyGroupCsd[] = { AMEDIAFORMAT_KEY_CSD_0, AMEDIAFORMAT_KEY_CSD_1, AMEDIAFORMAT_KEY_CSD_2, }; static const char *AMediaFormatKeyGroupRect[] = { AMEDIAFORMAT_KEY_DISPLAY_CROP, }; static const char *AMediaFormatKeyGroupFloatInt32[] = { AMEDIAFORMAT_KEY_FRAME_RATE, AMEDIAFORMAT_KEY_I_FRAME_INTERVAL, AMEDIAFORMAT_KEY_OPERATING_RATE, }; static status_t translateErrorCode(media_status_t err) { if (err == AMEDIA_OK) { return OK; } else if (err == AMEDIA_ERROR_END_OF_STREAM) { return ERROR_END_OF_STREAM; } else if (err == AMEDIA_ERROR_IO) { return ERROR_IO; } else if (err == AMEDIACODEC_INFO_TRY_AGAIN_LATER) { return -EAGAIN; } ALOGE("ndk error code: %d", err); return UNKNOWN_ERROR; } static int32_t translateActionCode(int32_t actionCode) { if (AMediaCodecActionCode_isTransient(actionCode)) { return ACTION_CODE_TRANSIENT; } else if (AMediaCodecActionCode_isRecoverable(actionCode)) { return ACTION_CODE_RECOVERABLE; } return ACTION_CODE_FATAL; } static CryptoPlugin::Mode translateToCryptoPluginMode(cryptoinfo_mode_t mode) { CryptoPlugin::Mode ret = CryptoPlugin::kMode_Unencrypted; switch (mode) { case AMEDIACODECRYPTOINFO_MODE_AES_CTR: { ret = CryptoPlugin::kMode_AES_CTR; break; } case AMEDIACODECRYPTOINFO_MODE_AES_WV: { ret = CryptoPlugin::kMode_AES_WV; break; } case AMEDIACODECRYPTOINFO_MODE_AES_CBC: { ret = CryptoPlugin::kMode_AES_CBC; break; } default: break; } return ret; } static cryptoinfo_mode_t translateToCryptoInfoMode(CryptoPlugin::Mode mode) { cryptoinfo_mode_t ret = AMEDIACODECRYPTOINFO_MODE_CLEAR; switch (mode) { case CryptoPlugin::kMode_AES_CTR: { ret = AMEDIACODECRYPTOINFO_MODE_AES_CTR; break; } case CryptoPlugin::kMode_AES_WV: { ret = AMEDIACODECRYPTOINFO_MODE_AES_WV; break; } case CryptoPlugin::kMode_AES_CBC: { ret = AMEDIACODECRYPTOINFO_MODE_AES_CBC; break; } default: break; } return ret; } //////////// AMediaFormatWrapper // static sp<AMediaFormatWrapper> AMediaFormatWrapper::Create(const sp<AMessage> &message) { sp<AMediaFormatWrapper> aMediaFormat = new AMediaFormatWrapper(); for (size_t i = 0; i < message->countEntries(); ++i) { AMessage::Type valueType; const char *key = message->getEntryNameAt(i, &valueType); switch (valueType) { case AMessage::kTypeInt32: { int32_t val; if (!message->findInt32(key, &val)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setInt32(key, val); break; } case AMessage::kTypeInt64: { int64_t val; if (!message->findInt64(key, &val)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setInt64(key, val); break; } case AMessage::kTypeFloat: { float val; if (!message->findFloat(key, &val)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setFloat(key, val); break; } case AMessage::kTypeDouble: { double val; if (!message->findDouble(key, &val)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setDouble(key, val); break; } case AMessage::kTypeSize: { size_t val; if (!message->findSize(key, &val)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setSize(key, val); break; } case AMessage::kTypeRect: { int32_t left, top, right, bottom; if (!message->findRect(key, &left, &top, &right, &bottom)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setRect(key, left, top, right, bottom); break; } case AMessage::kTypeString: { AString val; if (!message->findString(key, &val)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setString(key, val); break; } case AMessage::kTypeBuffer: { sp<ABuffer> val; if (!message->findBuffer(key, &val)) { ALOGE("AMediaFormatWrapper::Create: error at item %zu", i); continue; } aMediaFormat->setBuffer(key, val->data(), val->size()); break; } default: { break; } } } return aMediaFormat; } AMediaFormatWrapper::AMediaFormatWrapper() { mAMediaFormat = AMediaFormat_new(); } AMediaFormatWrapper::AMediaFormatWrapper(AMediaFormat *aMediaFormat) : mAMediaFormat(aMediaFormat) { } AMediaFormatWrapper::~AMediaFormatWrapper() { release(); } status_t AMediaFormatWrapper::release() { if (mAMediaFormat != NULL) { media_status_t err = AMediaFormat_delete(mAMediaFormat); mAMediaFormat = NULL; return translateErrorCode(err); } return OK; } AMediaFormat *AMediaFormatWrapper::getAMediaFormat() const { return mAMediaFormat; } sp<AMessage> AMediaFormatWrapper::toAMessage() const { sp<AMessage> msg; writeToAMessage(msg); return msg; } void AMediaFormatWrapper::writeToAMessage(sp<AMessage> &msg) const { if (mAMediaFormat == NULL) { msg = NULL; } if (msg == NULL) { msg = new AMessage; } for (auto& key : AMediaFormatKeyGroupInt32) { int32_t val; if (getInt32(key, &val)) { msg->setInt32(key, val); } } for (auto& key : AMediaFormatKeyGroupInt64) { int64_t val; if (getInt64(key, &val)) { msg->setInt64(key, val); } } for (auto& key : AMediaFormatKeyGroupString) { AString val; if (getString(key, &val)) { msg->setString(key, val); } } for (auto& key : AMediaFormatKeyGroupBuffer) { void *data; size_t size; if (getBuffer(key, &data, &size)) { sp<ABuffer> buffer = ABuffer::CreateAsCopy(data, size); msg->setBuffer(key, buffer); } } for (auto& key : AMediaFormatKeyGroupCsd) { void *data; size_t size; if (getBuffer(key, &data, &size)) { sp<ABuffer> buffer = ABuffer::CreateAsCopy(data, size); buffer->meta()->setInt32(AMEDIAFORMAT_KEY_CSD, 1); buffer->meta()->setInt64(AMEDIAFORMAT_KEY_TIME_US, 0); msg->setBuffer(key, buffer); } } for (auto& key : AMediaFormatKeyGroupRect) { int32_t left, top, right, bottom; if (getRect(key, &left, &top, &right, &bottom)) { msg->setRect(key, left, top, right, bottom); } } for (auto& key : AMediaFormatKeyGroupFloatInt32) { float valFloat; if (getFloat(key, &valFloat)) { msg->setFloat(key, valFloat); } else { int32_t valInt32; if (getInt32(key, &valInt32)) { msg->setFloat(key, (float)valInt32); } } } } const char* AMediaFormatWrapper::toString() const { if (mAMediaFormat == NULL) { return NULL; } return AMediaFormat_toString(mAMediaFormat); } bool AMediaFormatWrapper::getInt32(const char *name, int32_t *out) const { if (mAMediaFormat == NULL) { return false; } return AMediaFormat_getInt32(mAMediaFormat, name, out); } bool AMediaFormatWrapper::getInt64(const char *name, int64_t *out) const { if (mAMediaFormat == NULL) { return false; } return AMediaFormat_getInt64(mAMediaFormat, name, out); } bool AMediaFormatWrapper::getFloat(const char *name, float *out) const { if (mAMediaFormat == NULL) { return false; } return AMediaFormat_getFloat(mAMediaFormat, name, out); } bool AMediaFormatWrapper::getDouble(const char *name, double *out) const { if (mAMediaFormat == NULL) { return false; } return AMediaFormat_getDouble(mAMediaFormat, name, out); } bool AMediaFormatWrapper::getSize(const char *name, size_t *out) const { if (mAMediaFormat == NULL) { return false; } return AMediaFormat_getSize(mAMediaFormat, name, out); } bool AMediaFormatWrapper::getRect( const char *name, int32_t *left, int32_t *top, int32_t *right, int32_t *bottom) const { if (mAMediaFormat == NULL) { return false; } return AMediaFormat_getRect(mAMediaFormat, name, left, top, right, bottom); } bool AMediaFormatWrapper::getBuffer(const char *name, void** data, size_t *outSize) const { if (mAMediaFormat == NULL) { return false; } return AMediaFormat_getBuffer(mAMediaFormat, name, data, outSize); } bool AMediaFormatWrapper::getString(const char *name, AString *out) const { if (mAMediaFormat == NULL) { return false; } const char *outChar = NULL; bool ret = AMediaFormat_getString(mAMediaFormat, name, &outChar); if (ret) { *out = AString(outChar); } return ret; } void AMediaFormatWrapper::setInt32(const char* name, int32_t value) { if (mAMediaFormat != NULL) { AMediaFormat_setInt32(mAMediaFormat, name, value); } } void AMediaFormatWrapper::setInt64(const char* name, int64_t value) { if (mAMediaFormat != NULL) { AMediaFormat_setInt64(mAMediaFormat, name, value); } } void AMediaFormatWrapper::setFloat(const char* name, float value) { if (mAMediaFormat != NULL) { AMediaFormat_setFloat(mAMediaFormat, name, value); } } void AMediaFormatWrapper::setDouble(const char* name, double value) { if (mAMediaFormat != NULL) { AMediaFormat_setDouble(mAMediaFormat, name, value); } } void AMediaFormatWrapper::setSize(const char* name, size_t value) { if (mAMediaFormat != NULL) { AMediaFormat_setSize(mAMediaFormat, name, value); } } void AMediaFormatWrapper::setRect( const char* name, int32_t left, int32_t top, int32_t right, int32_t bottom) { if (mAMediaFormat != NULL) { AMediaFormat_setRect(mAMediaFormat, name, left, top, right, bottom); } } void AMediaFormatWrapper::setString(const char* name, const AString &value) { if (mAMediaFormat != NULL) { AMediaFormat_setString(mAMediaFormat, name, value.c_str()); } } void AMediaFormatWrapper::setBuffer(const char* name, void* data, size_t size) { if (mAMediaFormat != NULL) { AMediaFormat_setBuffer(mAMediaFormat, name, data, size); } } //////////// ANativeWindowWrapper ANativeWindowWrapper::ANativeWindowWrapper(ANativeWindow *aNativeWindow) : mANativeWindow(aNativeWindow) { if (aNativeWindow != NULL) { ANativeWindow_acquire(aNativeWindow); } } ANativeWindowWrapper::~ANativeWindowWrapper() { release(); } status_t ANativeWindowWrapper::release() { if (mANativeWindow != NULL) { ANativeWindow_release(mANativeWindow); mANativeWindow = NULL; } return OK; } ANativeWindow *ANativeWindowWrapper::getANativeWindow() const { return mANativeWindow; } //////////// AMediaDrmWrapper AMediaDrmWrapper::AMediaDrmWrapper(const uint8_t uuid[16]) { mAMediaDrm = AMediaDrm_createByUUID(uuid); } AMediaDrmWrapper::AMediaDrmWrapper(AMediaDrm *aMediaDrm) : mAMediaDrm(aMediaDrm) { } AMediaDrmWrapper::~AMediaDrmWrapper() { release(); } status_t AMediaDrmWrapper::release() { if (mAMediaDrm != NULL) { AMediaDrm_release(mAMediaDrm); mAMediaDrm = NULL; } return OK; } AMediaDrm *AMediaDrmWrapper::getAMediaDrm() const { return mAMediaDrm; } // static bool AMediaDrmWrapper::isCryptoSchemeSupported( const uint8_t uuid[16], const char *mimeType) { return AMediaDrm_isCryptoSchemeSupported(uuid, mimeType); } //////////// AMediaCryptoWrapper AMediaCryptoWrapper::AMediaCryptoWrapper( const uint8_t uuid[16], const void *initData, size_t initDataSize) { mAMediaCrypto = AMediaCrypto_new(uuid, initData, initDataSize); } AMediaCryptoWrapper::AMediaCryptoWrapper(AMediaCrypto *aMediaCrypto) : mAMediaCrypto(aMediaCrypto) { } AMediaCryptoWrapper::~AMediaCryptoWrapper() { release(); } status_t AMediaCryptoWrapper::release() { if (mAMediaCrypto != NULL) { AMediaCrypto_delete(mAMediaCrypto); mAMediaCrypto = NULL; } return OK; } AMediaCrypto *AMediaCryptoWrapper::getAMediaCrypto() const { return mAMediaCrypto; } bool AMediaCryptoWrapper::isCryptoSchemeSupported(const uint8_t uuid[16]) { if (mAMediaCrypto == NULL) { return false; } return AMediaCrypto_isCryptoSchemeSupported(uuid); } bool AMediaCryptoWrapper::requiresSecureDecoderComponent(const char *mime) { if (mAMediaCrypto == NULL) { return false; } return AMediaCrypto_requiresSecureDecoderComponent(mime); } //////////// AMediaCodecCryptoInfoWrapper // static sp<AMediaCodecCryptoInfoWrapper> AMediaCodecCryptoInfoWrapper::Create(MetaDataBase &meta) { uint32_t type; const void *crypteddata; size_t cryptedsize; if (!meta.findData(kKeyEncryptedSizes, &type, &crypteddata, &cryptedsize)) { return NULL; } int numSubSamples = cryptedsize / sizeof(size_t); if (numSubSamples <= 0) { ALOGE("Create: INVALID numSubSamples: %d", numSubSamples); return NULL; } const void *cleardata; size_t clearsize; if (meta.findData(kKeyPlainSizes, &type, &cleardata, &clearsize)) { if (clearsize != cryptedsize) { // The two must be of the same length. ALOGE("Create: mismatch cryptedsize: %zu != clearsize: %zu", cryptedsize, clearsize); return NULL; } } const void *key; size_t keysize; if (meta.findData(kKeyCryptoKey, &type, &key, &keysize)) { if (keysize != kAESBlockSize) { // Keys must be 16 bytes in length. ALOGE("Create: Keys must be %zu bytes in length: %zu", kAESBlockSize, keysize); return NULL; } } const void *iv; size_t ivsize; if (meta.findData(kKeyCryptoIV, &type, &iv, &ivsize)) { if (ivsize != kAESBlockSize) { // IVs must be 16 bytes in length. ALOGE("Create: IV must be %zu bytes in length: %zu", kAESBlockSize, ivsize); return NULL; } } int32_t mode; if (!meta.findInt32(kKeyCryptoMode, &mode)) { mode = CryptoPlugin::kMode_AES_CTR; } return new AMediaCodecCryptoInfoWrapper( numSubSamples, (uint8_t*) key, (uint8_t*) iv, (CryptoPlugin::Mode)mode, (size_t*) cleardata, (size_t*) crypteddata); } AMediaCodecCryptoInfoWrapper::AMediaCodecCryptoInfoWrapper( int numsubsamples, uint8_t key[16], uint8_t iv[16], CryptoPlugin::Mode mode, size_t *clearbytes, size_t *encryptedbytes) { mAMediaCodecCryptoInfo = AMediaCodecCryptoInfo_new(numsubsamples, key, iv, translateToCryptoInfoMode(mode), clearbytes, encryptedbytes); } AMediaCodecCryptoInfoWrapper::AMediaCodecCryptoInfoWrapper( AMediaCodecCryptoInfo *aMediaCodecCryptoInfo) : mAMediaCodecCryptoInfo(aMediaCodecCryptoInfo) { } AMediaCodecCryptoInfoWrapper::~AMediaCodecCryptoInfoWrapper() { release(); } status_t AMediaCodecCryptoInfoWrapper::release() { if (mAMediaCodecCryptoInfo != NULL) { media_status_t err = AMediaCodecCryptoInfo_delete(mAMediaCodecCryptoInfo); mAMediaCodecCryptoInfo = NULL; return translateErrorCode(err); } return OK; } AMediaCodecCryptoInfo *AMediaCodecCryptoInfoWrapper::getAMediaCodecCryptoInfo() const { return mAMediaCodecCryptoInfo; } void AMediaCodecCryptoInfoWrapper::setPattern(CryptoPlugin::Pattern *pattern) { if (mAMediaCodecCryptoInfo == NULL || pattern == NULL) { return; } cryptoinfo_pattern_t ndkPattern = {(int32_t)pattern->mEncryptBlocks, (int32_t)pattern->mSkipBlocks }; return AMediaCodecCryptoInfo_setPattern(mAMediaCodecCryptoInfo, &ndkPattern); } size_t AMediaCodecCryptoInfoWrapper::getNumSubSamples() { if (mAMediaCodecCryptoInfo == NULL) { return 0; } return AMediaCodecCryptoInfo_getNumSubSamples(mAMediaCodecCryptoInfo); } status_t AMediaCodecCryptoInfoWrapper::getKey(uint8_t *dst) { if (mAMediaCodecCryptoInfo == NULL) { return DEAD_OBJECT; } if (dst == NULL) { return BAD_VALUE; } return translateErrorCode( AMediaCodecCryptoInfo_getKey(mAMediaCodecCryptoInfo, dst)); } status_t AMediaCodecCryptoInfoWrapper::getIV(uint8_t *dst) { if (mAMediaCodecCryptoInfo == NULL) { return DEAD_OBJECT; } if (dst == NULL) { return BAD_VALUE; } return translateErrorCode( AMediaCodecCryptoInfo_getIV(mAMediaCodecCryptoInfo, dst)); } CryptoPlugin::Mode AMediaCodecCryptoInfoWrapper::getMode() { if (mAMediaCodecCryptoInfo == NULL) { return CryptoPlugin::kMode_Unencrypted; } return translateToCryptoPluginMode( AMediaCodecCryptoInfo_getMode(mAMediaCodecCryptoInfo)); } status_t AMediaCodecCryptoInfoWrapper::getClearBytes(size_t *dst) { if (mAMediaCodecCryptoInfo == NULL) { return DEAD_OBJECT; } if (dst == NULL) { return BAD_VALUE; } return translateErrorCode( AMediaCodecCryptoInfo_getClearBytes(mAMediaCodecCryptoInfo, dst)); } status_t AMediaCodecCryptoInfoWrapper::getEncryptedBytes(size_t *dst) { if (mAMediaCodecCryptoInfo == NULL) { return DEAD_OBJECT; } if (dst == NULL) { return BAD_VALUE; } return translateErrorCode( AMediaCodecCryptoInfo_getEncryptedBytes(mAMediaCodecCryptoInfo, dst)); } //////////// AMediaCodecWrapper // static sp<AMediaCodecWrapper> AMediaCodecWrapper::CreateCodecByName(const AString &name) { AMediaCodec *aMediaCodec = AMediaCodec_createCodecByName(name.c_str()); return new AMediaCodecWrapper(aMediaCodec); } // static sp<AMediaCodecWrapper> AMediaCodecWrapper::CreateDecoderByType(const AString &mimeType) { AMediaCodec *aMediaCodec = AMediaCodec_createDecoderByType(mimeType.c_str()); return new AMediaCodecWrapper(aMediaCodec); } // static void AMediaCodecWrapper::OnInputAvailableCB( AMediaCodec * /* aMediaCodec */, void *userdata, int32_t index) { ALOGV("OnInputAvailableCB: index(%d)", index); sp<AMessage> msg = sp<AMessage>((AMessage *)userdata)->dup(); msg->setInt32("callbackID", CB_INPUT_AVAILABLE); msg->setInt32("index", index); msg->post(); } // static void AMediaCodecWrapper::OnOutputAvailableCB( AMediaCodec * /* aMediaCodec */, void *userdata, int32_t index, AMediaCodecBufferInfo *bufferInfo) { ALOGV("OnOutputAvailableCB: index(%d), (%d, %d, %lld, 0x%x)", index, bufferInfo->offset, bufferInfo->size, (long long)bufferInfo->presentationTimeUs, bufferInfo->flags); sp<AMessage> msg = sp<AMessage>((AMessage *)userdata)->dup(); msg->setInt32("callbackID", CB_OUTPUT_AVAILABLE); msg->setInt32("index", index); msg->setSize("offset", (size_t)(bufferInfo->offset)); msg->setSize("size", (size_t)(bufferInfo->size)); msg->setInt64("timeUs", bufferInfo->presentationTimeUs); msg->setInt32("flags", (int32_t)(bufferInfo->flags)); msg->post(); } // static void AMediaCodecWrapper::OnFormatChangedCB( AMediaCodec * /* aMediaCodec */, void *userdata, AMediaFormat *format) { sp<AMediaFormatWrapper> formatWrapper = new AMediaFormatWrapper(format); sp<AMessage> outputFormat = formatWrapper->toAMessage(); ALOGV("OnFormatChangedCB: format(%s)", outputFormat->debugString().c_str()); sp<AMessage> msg = sp<AMessage>((AMessage *)userdata)->dup(); msg->setInt32("callbackID", CB_OUTPUT_FORMAT_CHANGED); msg->setMessage("format", outputFormat); msg->post(); } // static void AMediaCodecWrapper::OnErrorCB( AMediaCodec * /* aMediaCodec */, void *userdata, media_status_t err, int32_t actionCode, const char *detail) { ALOGV("OnErrorCB: err(%d), actionCode(%d), detail(%s)", err, actionCode, detail); sp<AMessage> msg = sp<AMessage>((AMessage *)userdata)->dup(); msg->setInt32("callbackID", CB_ERROR); msg->setInt32("err", translateErrorCode(err)); msg->setInt32("actionCode", translateActionCode(actionCode)); msg->setString("detail", detail); msg->post(); } AMediaCodecWrapper::AMediaCodecWrapper(AMediaCodec *aMediaCodec) : mAMediaCodec(aMediaCodec) { } AMediaCodecWrapper::~AMediaCodecWrapper() { release(); } status_t AMediaCodecWrapper::release() { if (mAMediaCodec != NULL) { AMediaCodecOnAsyncNotifyCallback aCB = {}; AMediaCodec_setAsyncNotifyCallback(mAMediaCodec, aCB, NULL); mCallback = NULL; media_status_t err = AMediaCodec_delete(mAMediaCodec); mAMediaCodec = NULL; return translateErrorCode(err); } return OK; } AMediaCodec *AMediaCodecWrapper::getAMediaCodec() const { return mAMediaCodec; } status_t AMediaCodecWrapper::getName(AString *outComponentName) const { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } char *name = NULL; media_status_t err = AMediaCodec_getName(mAMediaCodec, &name); if (err != AMEDIA_OK) { return translateErrorCode(err); } *outComponentName = AString(name); AMediaCodec_releaseName(mAMediaCodec, name); return OK; } status_t AMediaCodecWrapper::configure( const sp<AMediaFormatWrapper> &format, const sp<ANativeWindowWrapper> &nww, const sp<AMediaCryptoWrapper> &crypto, uint32_t flags) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } media_status_t err = AMediaCodec_configure( mAMediaCodec, format->getAMediaFormat(), (nww == NULL ? NULL : nww->getANativeWindow()), crypto == NULL ? NULL : crypto->getAMediaCrypto(), flags); return translateErrorCode(err); } status_t AMediaCodecWrapper::setCallback(const sp<AMessage> &callback) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } mCallback = callback; AMediaCodecOnAsyncNotifyCallback aCB = { OnInputAvailableCB, OnOutputAvailableCB, OnFormatChangedCB, OnErrorCB }; return translateErrorCode( AMediaCodec_setAsyncNotifyCallback(mAMediaCodec, aCB, callback.get())); } status_t AMediaCodecWrapper::releaseCrypto() { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaCodec_releaseCrypto(mAMediaCodec)); } status_t AMediaCodecWrapper::start() { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaCodec_start(mAMediaCodec)); } status_t AMediaCodecWrapper::stop() { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaCodec_stop(mAMediaCodec)); } status_t AMediaCodecWrapper::flush() { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaCodec_flush(mAMediaCodec)); } uint8_t* AMediaCodecWrapper::getInputBuffer(size_t idx, size_t *out_size) { if (mAMediaCodec == NULL) { return NULL; } return AMediaCodec_getInputBuffer(mAMediaCodec, idx, out_size); } uint8_t* AMediaCodecWrapper::getOutputBuffer(size_t idx, size_t *out_size) { if (mAMediaCodec == NULL) { return NULL; } return AMediaCodec_getOutputBuffer(mAMediaCodec, idx, out_size); } status_t AMediaCodecWrapper::queueInputBuffer( size_t idx, size_t offset, size_t size, uint64_t time, uint32_t flags) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode( AMediaCodec_queueInputBuffer(mAMediaCodec, idx, offset, size, time, flags)); } status_t AMediaCodecWrapper::queueSecureInputBuffer( size_t idx, size_t offset, sp<AMediaCodecCryptoInfoWrapper> &codecCryptoInfo, uint64_t time, uint32_t flags) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode( AMediaCodec_queueSecureInputBuffer( mAMediaCodec, idx, offset, codecCryptoInfo->getAMediaCodecCryptoInfo(), time, flags)); } sp<AMediaFormatWrapper> AMediaCodecWrapper::getOutputFormat() { if (mAMediaCodec == NULL) { return NULL; } return new AMediaFormatWrapper(AMediaCodec_getOutputFormat(mAMediaCodec)); } sp<AMediaFormatWrapper> AMediaCodecWrapper::getInputFormat() { if (mAMediaCodec == NULL) { return NULL; } return new AMediaFormatWrapper(AMediaCodec_getInputFormat(mAMediaCodec)); } status_t AMediaCodecWrapper::releaseOutputBuffer(size_t idx, bool render) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode( AMediaCodec_releaseOutputBuffer(mAMediaCodec, idx, render)); } status_t AMediaCodecWrapper::setOutputSurface(const sp<ANativeWindowWrapper> &nww) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode( AMediaCodec_setOutputSurface(mAMediaCodec, (nww == NULL ? NULL : nww->getANativeWindow()))); } status_t AMediaCodecWrapper::releaseOutputBufferAtTime(size_t idx, int64_t timestampNs) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode( AMediaCodec_releaseOutputBufferAtTime(mAMediaCodec, idx, timestampNs)); } status_t AMediaCodecWrapper::setParameters(const sp<AMediaFormatWrapper> ¶ms) { if (mAMediaCodec == NULL) { return DEAD_OBJECT; } return translateErrorCode( AMediaCodec_setParameters(mAMediaCodec, params->getAMediaFormat())); } //////////// AMediaExtractorWrapper AMediaExtractorWrapper::AMediaExtractorWrapper(AMediaExtractor *aMediaExtractor) : mAMediaExtractor(aMediaExtractor) { } AMediaExtractorWrapper::~AMediaExtractorWrapper() { release(); } status_t AMediaExtractorWrapper::release() { if (mAMediaExtractor != NULL) { media_status_t err = AMediaExtractor_delete(mAMediaExtractor); mAMediaExtractor = NULL; return translateErrorCode(err); } return OK; } status_t AMediaExtractorWrapper::disconnect() { if (mAMediaExtractor != NULL) { media_status_t err = AMediaExtractor_disconnect(mAMediaExtractor); return translateErrorCode(err); } return DEAD_OBJECT; } AMediaExtractor *AMediaExtractorWrapper::getAMediaExtractor() const { return mAMediaExtractor; } status_t AMediaExtractorWrapper::setDataSource(int fd, off64_t offset, off64_t length) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaExtractor_setDataSourceFd( mAMediaExtractor, fd, offset, length)); } status_t AMediaExtractorWrapper::setDataSource(const char *location) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaExtractor_setDataSource(mAMediaExtractor, location)); } status_t AMediaExtractorWrapper::setDataSource(AMediaDataSource *source) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaExtractor_setDataSourceCustom(mAMediaExtractor, source)); } size_t AMediaExtractorWrapper::getTrackCount() { if (mAMediaExtractor == NULL) { return 0; } return AMediaExtractor_getTrackCount(mAMediaExtractor); } sp<AMediaFormatWrapper> AMediaExtractorWrapper::getFormat() { if (mAMediaExtractor == NULL) { return NULL; } return new AMediaFormatWrapper(AMediaExtractor_getFileFormat(mAMediaExtractor)); } sp<AMediaFormatWrapper> AMediaExtractorWrapper::getTrackFormat(size_t idx) { if (mAMediaExtractor == NULL) { return NULL; } return new AMediaFormatWrapper(AMediaExtractor_getTrackFormat(mAMediaExtractor, idx)); } status_t AMediaExtractorWrapper::selectTrack(size_t idx) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaExtractor_selectTrack(mAMediaExtractor, idx)); } status_t AMediaExtractorWrapper::unselectTrack(size_t idx) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } return translateErrorCode(AMediaExtractor_unselectTrack(mAMediaExtractor, idx)); } status_t AMediaExtractorWrapper::selectSingleTrack(size_t idx) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } for (size_t i = 0; i < AMediaExtractor_getTrackCount(mAMediaExtractor); ++i) { if (i == idx) { media_status_t err = AMediaExtractor_selectTrack(mAMediaExtractor, i); if (err != AMEDIA_OK) { return translateErrorCode(err); } } else { media_status_t err = AMediaExtractor_unselectTrack(mAMediaExtractor, i); if (err != AMEDIA_OK) { return translateErrorCode(err); } } } return OK; } ssize_t AMediaExtractorWrapper::readSampleData(const sp<ABuffer> &buffer) { if (mAMediaExtractor == NULL) { return -1; } return AMediaExtractor_readSampleData(mAMediaExtractor, buffer->data(), buffer->capacity()); } ssize_t AMediaExtractorWrapper::getSampleSize() { if (mAMediaExtractor == NULL) { return 0; } return AMediaExtractor_getSampleSize(mAMediaExtractor); } uint32_t AMediaExtractorWrapper::getSampleFlags() { if (mAMediaExtractor == NULL) { return 0; } return AMediaExtractor_getSampleFlags(mAMediaExtractor); } int AMediaExtractorWrapper::getSampleTrackIndex() { if (mAMediaExtractor == NULL) { return -1; } return AMediaExtractor_getSampleTrackIndex(mAMediaExtractor); } int64_t AMediaExtractorWrapper::getSampleTime() { if (mAMediaExtractor == NULL) { return -1; } return AMediaExtractor_getSampleTime(mAMediaExtractor); } status_t AMediaExtractorWrapper::getSampleFormat(sp<AMediaFormatWrapper> &formatWrapper) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } AMediaFormat *format = AMediaFormat_new(); formatWrapper = new AMediaFormatWrapper(format); return translateErrorCode(AMediaExtractor_getSampleFormat(mAMediaExtractor, format)); } int64_t AMediaExtractorWrapper::getCachedDuration() { if (mAMediaExtractor == NULL) { return -1; } return AMediaExtractor_getCachedDuration(mAMediaExtractor); } bool AMediaExtractorWrapper::advance() { if (mAMediaExtractor == NULL) { return false; } return AMediaExtractor_advance(mAMediaExtractor); } status_t AMediaExtractorWrapper::seekTo(int64_t seekPosUs, MediaSource::ReadOptions::SeekMode mode) { if (mAMediaExtractor == NULL) { return DEAD_OBJECT; } SeekMode aMode; switch (mode) { case MediaSource::ReadOptions::SEEK_PREVIOUS_SYNC: { aMode = AMEDIAEXTRACTOR_SEEK_PREVIOUS_SYNC; break; } case MediaSource::ReadOptions::SEEK_NEXT_SYNC: { aMode = AMEDIAEXTRACTOR_SEEK_NEXT_SYNC; break; } default: { aMode = AMEDIAEXTRACTOR_SEEK_CLOSEST_SYNC; break; } } return AMediaExtractor_seekTo(mAMediaExtractor, seekPosUs, aMode); } PsshInfo* AMediaExtractorWrapper::getPsshInfo() { if (mAMediaExtractor == NULL) { return NULL; } return AMediaExtractor_getPsshInfo(mAMediaExtractor); } sp<AMediaCodecCryptoInfoWrapper> AMediaExtractorWrapper::getSampleCryptoInfo() { if (mAMediaExtractor == NULL) { return NULL; } return new AMediaCodecCryptoInfoWrapper(AMediaExtractor_getSampleCryptoInfo(mAMediaExtractor)); } ssize_t AMediaDataSourceWrapper::AMediaDataSourceWrapper_getSize(void *userdata) { DataSource *source = static_cast<DataSource *>(userdata); off64_t size = -1; source->getSize(&size); return size; } ssize_t AMediaDataSourceWrapper::AMediaDataSourceWrapper_readAt(void *userdata, off64_t offset, void * buf, size_t size) { DataSource *source = static_cast<DataSource *>(userdata); return source->readAt(offset, buf, size); } void AMediaDataSourceWrapper::AMediaDataSourceWrapper_close(void *userdata) { DataSource *source = static_cast<DataSource *>(userdata); source->close(); } AMediaDataSourceWrapper::AMediaDataSourceWrapper(const sp<DataSource> &dataSource) : mDataSource(dataSource), mAMediaDataSource(AMediaDataSource_new()) { ALOGV("setDataSource (source: %p)", dataSource.get()); AMediaDataSource_setUserdata(mAMediaDataSource, dataSource.get()); AMediaDataSource_setReadAt(mAMediaDataSource, AMediaDataSourceWrapper_readAt); AMediaDataSource_setGetSize(mAMediaDataSource, AMediaDataSourceWrapper_getSize); AMediaDataSource_setClose(mAMediaDataSource, AMediaDataSourceWrapper_close); } AMediaDataSourceWrapper::~AMediaDataSourceWrapper() { if (mAMediaDataSource == NULL) { return; } AMediaDataSource_delete(mAMediaDataSource); mAMediaDataSource = NULL; } AMediaDataSource* AMediaDataSourceWrapper::getAMediaDataSource() { return mAMediaDataSource; } } // namespace android