diff options
author | Android (Google) Code Review <android-gerrit@google.com> | 2009-11-17 10:46:41 -0800 |
---|---|---|
committer | Android (Google) Code Review <android-gerrit@google.com> | 2009-11-17 10:46:41 -0800 |
commit | 2c7664989f7388c0c6cf0ec7d80ffa2704c8c2be (patch) | |
tree | eca65a517e48f75ff22ea7d60a0a42553cc5f889 /media | |
parent | a35f1c63a67df0832d9b3f1aed3e2376367d4398 (diff) | |
parent | ea6a38c63b9e9aeb45aa22587c069bd3c5d83df8 (diff) | |
download | frameworks_base-2c7664989f7388c0c6cf0ec7d80ffa2704c8c2be.zip frameworks_base-2c7664989f7388c0c6cf0ec7d80ffa2704c8c2be.tar.gz frameworks_base-2c7664989f7388c0c6cf0ec7d80ffa2704c8c2be.tar.bz2 |
Merge change Iea6a38c6 into eclair-mr2
* changes:
Squashed commit of the following:
Diffstat (limited to 'media')
-rw-r--r-- | media/libmediaplayerservice/Android.mk | 3 | ||||
-rw-r--r-- | media/libmediaplayerservice/MediaRecorderClient.cpp | 16 | ||||
-rw-r--r-- | media/libmediaplayerservice/MediaRecorderClient.h | 5 | ||||
-rw-r--r-- | media/libmediaplayerservice/StagefrightRecorder.cpp | 241 | ||||
-rw-r--r-- | media/libmediaplayerservice/StagefrightRecorder.h | 76 | ||||
-rw-r--r-- | media/libstagefright/CameraSource.cpp | 28 | ||||
-rw-r--r-- | media/libstagefright/MPEG4Writer.cpp | 96 | ||||
-rw-r--r-- | media/libstagefright/OMXCodec.cpp | 83 |
8 files changed, 530 insertions, 18 deletions
diff --git a/media/libmediaplayerservice/Android.mk b/media/libmediaplayerservice/Android.mk index 6fc9fa2..4784b8e 100644 --- a/media/libmediaplayerservice/Android.mk +++ b/media/libmediaplayerservice/Android.mk @@ -19,8 +19,9 @@ LOCAL_SRC_FILES:= \ ifeq ($(BUILD_WITH_FULL_STAGEFRIGHT),true) LOCAL_SRC_FILES += \ + StagefrightMetadataRetriever.cpp \ StagefrightPlayer.cpp \ - StagefrightMetadataRetriever.cpp + StagefrightRecorder.cpp LOCAL_CFLAGS += -DBUILD_WITH_FULL_STAGEFRIGHT=1 diff --git a/media/libmediaplayerservice/MediaRecorderClient.cpp b/media/libmediaplayerservice/MediaRecorderClient.cpp index 95ee3e4..2ea7cc3 100644 --- a/media/libmediaplayerservice/MediaRecorderClient.cpp +++ b/media/libmediaplayerservice/MediaRecorderClient.cpp @@ -24,6 +24,7 @@ #include <unistd.h> #include <string.h> #include <cutils/atomic.h> +#include <cutils/properties.h> // for property_get #include <android_runtime/ActivityManager.h> #include <binder/IPCThreadState.h> #include <binder/IServiceManager.h> @@ -37,6 +38,8 @@ #include "MediaRecorderClient.h" #include "MediaPlayerService.h" +#include "StagefrightRecorder.h" + namespace android { const char* cameraPermission = "android.permission.CAMERA"; @@ -286,7 +289,18 @@ MediaRecorderClient::MediaRecorderClient(const sp<MediaPlayerService>& service, { LOGV("Client constructor"); mPid = pid; - mRecorder = new PVMediaRecorder(); + +#if BUILD_WITH_FULL_STAGEFRIGHT + char value[PROPERTY_VALUE_MAX]; + if (property_get("media.stagefright.enable-record", value, NULL) + && (!strcmp(value, "1") || !strcasecmp(value, "true"))) { + mRecorder = new StagefrightRecorder; + } else +#endif + { + mRecorder = new PVMediaRecorder(); + } + mMediaPlayerService = service; } diff --git a/media/libmediaplayerservice/MediaRecorderClient.h b/media/libmediaplayerservice/MediaRecorderClient.h index 6260441..e07306b 100644 --- a/media/libmediaplayerservice/MediaRecorderClient.h +++ b/media/libmediaplayerservice/MediaRecorderClient.h @@ -22,8 +22,7 @@ namespace android { -class PVMediaRecorder; -class ISurface; +class MediaRecorderBase; class MediaPlayerService; class MediaRecorderClient : public BnMediaRecorder @@ -59,7 +58,7 @@ private: pid_t mPid; Mutex mLock; - PVMediaRecorder *mRecorder; + MediaRecorderBase *mRecorder; sp<MediaPlayerService> mMediaPlayerService; }; diff --git a/media/libmediaplayerservice/StagefrightRecorder.cpp b/media/libmediaplayerservice/StagefrightRecorder.cpp new file mode 100644 index 0000000..a55273d --- /dev/null +++ b/media/libmediaplayerservice/StagefrightRecorder.cpp @@ -0,0 +1,241 @@ +/* + * 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 "StagefrightRecorder" +#include <utils/Log.h> + +#include "StagefrightRecorder.h" + +#include <media/stagefright/CameraSource.h> +#include <media/stagefright/MPEG4Writer.h> +#include <media/stagefright/MediaDebug.h> +#include <media/stagefright/MediaDefs.h> +#include <media/stagefright/MetaData.h> +#include <media/stagefright/OMXClient.h> +#include <media/stagefright/OMXCodec.h> +#include <ui/ICamera.h> +#include <ui/ISurface.h> +#include <utils/Errors.h> + +namespace android { + +StagefrightRecorder::StagefrightRecorder() { + reset(); +} + +StagefrightRecorder::~StagefrightRecorder() { + stop(); + + if (mOutputFd >= 0) { + ::close(mOutputFd); + mOutputFd = -1; + } +} + +status_t StagefrightRecorder::init() { + return OK; +} + +status_t StagefrightRecorder::setAudioSource(audio_source as) { + mAudioSource = as; + + return OK; +} + +status_t StagefrightRecorder::setVideoSource(video_source vs) { + mVideoSource = vs; + + return OK; +} + +status_t StagefrightRecorder::setOutputFormat(output_format of) { + mOutputFormat = of; + + return OK; +} + +status_t StagefrightRecorder::setAudioEncoder(audio_encoder ae) { + mAudioEncoder = ae; + + return OK; +} + +status_t StagefrightRecorder::setVideoEncoder(video_encoder ve) { + mVideoEncoder = ve; + + return OK; +} + +status_t StagefrightRecorder::setVideoSize(int width, int height) { + mVideoWidth = width; + mVideoHeight = height; + + return OK; +} + +status_t StagefrightRecorder::setVideoFrameRate(int frames_per_second) { + mFrameRate = frames_per_second; + + return OK; +} + +status_t StagefrightRecorder::setCamera(const sp<ICamera> &camera) { + mCamera = camera; + + return OK; +} + +status_t StagefrightRecorder::setPreviewSurface(const sp<ISurface> &surface) { + mPreviewSurface = surface; + + return OK; +} + +status_t StagefrightRecorder::setOutputFile(const char *path) { + // We don't actually support this at all, as the media_server process + // no longer has permissions to create files. + + return UNKNOWN_ERROR; +} + +status_t StagefrightRecorder::setOutputFile(int fd, int64_t offset, int64_t length) { + // These don't make any sense, do they? + CHECK_EQ(offset, 0); + CHECK_EQ(length, 0); + + if (mOutputFd >= 0) { + ::close(mOutputFd); + } + mOutputFd = dup(fd); + + return OK; +} + +status_t StagefrightRecorder::setParameters(const String8 ¶ms) { + mParams = params; + + return OK; +} + +status_t StagefrightRecorder::setListener(const sp<IMediaPlayerClient> &listener) { + mListener = listener; + + return OK; +} + +status_t StagefrightRecorder::prepare() { + return OK; +} + +status_t StagefrightRecorder::start() { + if (mWriter != NULL) { + return UNKNOWN_ERROR; + } + + if (mVideoSource == VIDEO_SOURCE_CAMERA) { + CHECK(mCamera != NULL); + + sp<CameraSource> cameraSource = + CameraSource::CreateFromICamera(mCamera); + + CHECK(cameraSource != NULL); + + cameraSource->setPreviewSurface(mPreviewSurface); + + sp<MetaData> enc_meta = new MetaData; + switch (mVideoEncoder) { + case VIDEO_ENCODER_H263: + enc_meta->setCString(kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_H263); + break; + + case VIDEO_ENCODER_MPEG_4_SP: + enc_meta->setCString(kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_MPEG4); + break; + + case VIDEO_ENCODER_H264: + enc_meta->setCString(kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_AVC); + break; + + default: + CHECK(!"Should not be here, unsupported video encoding."); + break; + } + + sp<MetaData> meta = cameraSource->getFormat(); + + int32_t width, height; + CHECK(meta->findInt32(kKeyWidth, &width)); + CHECK(meta->findInt32(kKeyHeight, &height)); + + enc_meta->setInt32(kKeyWidth, width); + enc_meta->setInt32(kKeyHeight, height); + + OMXClient client; + CHECK_EQ(client.connect(), OK); + + sp<MediaSource> encoder = + OMXCodec::Create( + client.interface(), enc_meta, + true /* createEncoder */, cameraSource); + + CHECK(mOutputFd >= 0); + mWriter = new MPEG4Writer(dup(mOutputFd)); + mWriter->addSource(encoder); + mWriter->start(); + } + + return OK; +} + +status_t StagefrightRecorder::stop() { + if (mWriter == NULL) { + return UNKNOWN_ERROR; + } + + mWriter->stop(); + mWriter = NULL; + + return OK; +} + +status_t StagefrightRecorder::close() { + stop(); + + return OK; +} + +status_t StagefrightRecorder::reset() { + stop(); + + mAudioSource = AUDIO_SOURCE_LIST_END; + mVideoSource = VIDEO_SOURCE_LIST_END; + mOutputFormat = OUTPUT_FORMAT_LIST_END; + mAudioEncoder = AUDIO_ENCODER_LIST_END; + mVideoEncoder = VIDEO_ENCODER_LIST_END; + mVideoWidth = -1; + mVideoHeight = -1; + mFrameRate = -1; + mOutputFd = -1; + + return OK; +} + +status_t StagefrightRecorder::getMaxAmplitude(int *max) { + return UNKNOWN_ERROR; +} + +} // namespace android diff --git a/media/libmediaplayerservice/StagefrightRecorder.h b/media/libmediaplayerservice/StagefrightRecorder.h new file mode 100644 index 0000000..56c4e0e --- /dev/null +++ b/media/libmediaplayerservice/StagefrightRecorder.h @@ -0,0 +1,76 @@ +/* + * 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. + */ + +#ifndef STAGEFRIGHT_RECORDER_H_ + +#define STAGEFRIGHT_RECORDER_H_ + +#include <media/MediaRecorderBase.h> +#include <utils/String8.h> + +namespace android { + +class MPEG4Writer; + +struct StagefrightRecorder : public MediaRecorderBase { + StagefrightRecorder(); + virtual ~StagefrightRecorder(); + + virtual status_t init(); + virtual status_t setAudioSource(audio_source as); + virtual status_t setVideoSource(video_source vs); + virtual status_t setOutputFormat(output_format of); + virtual status_t setAudioEncoder(audio_encoder ae); + virtual status_t setVideoEncoder(video_encoder ve); + virtual status_t setVideoSize(int width, int height); + virtual status_t setVideoFrameRate(int frames_per_second); + virtual status_t setCamera(const sp<ICamera>& camera); + virtual status_t setPreviewSurface(const sp<ISurface>& surface); + virtual status_t setOutputFile(const char *path); + virtual status_t setOutputFile(int fd, int64_t offset, int64_t length); + virtual status_t setParameters(const String8& params); + virtual status_t setListener(const sp<IMediaPlayerClient>& listener); + virtual status_t prepare(); + virtual status_t start(); + virtual status_t stop(); + virtual status_t close(); + virtual status_t reset(); + virtual status_t getMaxAmplitude(int *max); + +private: + sp<ICamera> mCamera; + sp<ISurface> mPreviewSurface; + sp<IMediaPlayerClient> mListener; + sp<MPEG4Writer> mWriter; + + audio_source mAudioSource; + video_source mVideoSource; + output_format mOutputFormat; + audio_encoder mAudioEncoder; + video_encoder mVideoEncoder; + int mVideoWidth, mVideoHeight; + int mFrameRate; + String8 mParams; + int mOutputFd; + + StagefrightRecorder(const StagefrightRecorder &); + StagefrightRecorder &operator=(const StagefrightRecorder &); +}; + +} // namespace android + +#endif // STAGEFRIGHT_RECORDER_H_ + diff --git a/media/libstagefright/CameraSource.cpp b/media/libstagefright/CameraSource.cpp index e9d8557..bd862e0 100644 --- a/media/libstagefright/CameraSource.cpp +++ b/media/libstagefright/CameraSource.cpp @@ -19,6 +19,7 @@ #include <OMX_Component.h> #include <binder/IServiceManager.h> +#include <cutils/properties.h> // for property_get #include <media/stagefright/CameraSource.h> #include <media/stagefright/MediaDebug.h> #include <media/stagefright/MediaDefs.h> @@ -123,6 +124,17 @@ CameraSource *CameraSource::Create() { return new CameraSource(camera); } +// static +CameraSource *CameraSource::CreateFromICamera(const sp<ICamera> &icamera) { + sp<Camera> camera = Camera::create(icamera); + + if (camera.get() == NULL) { + return NULL; + } + + return new CameraSource(camera); +} + CameraSource::CameraSource(const sp<Camera> &camera) : mCamera(camera), mWidth(0), @@ -130,6 +142,13 @@ CameraSource::CameraSource(const sp<Camera> &camera) mFirstFrameTimeUs(0), mNumFrames(0), mStarted(false) { + char value[PROPERTY_VALUE_MAX]; + if (property_get("ro.hardware", value, NULL) && !strcmp(value, "sholes")) { + // The hardware encoder(s) do not support yuv420, but only YCbYCr, + // fortunately the camera also supports this, so we needn't transcode. + mCamera->setParameters(String8("preview-format=yuv422i-yuyv")); + } + String8 s = mCamera->getParameters(); printf("params: \"%s\"\n", s.string()); @@ -143,13 +162,18 @@ CameraSource::~CameraSource() { } } +void CameraSource::setPreviewSurface(const sp<ISurface> &surface) { + mPreviewSurface = surface; +} + status_t CameraSource::start(MetaData *) { CHECK(!mStarted); mCamera->setListener(new CameraSourceListener(this)); - sp<ISurface> dummy = new DummySurface; - status_t err = mCamera->setPreviewDisplay(dummy); + status_t err = + mCamera->setPreviewDisplay( + mPreviewSurface != NULL ? mPreviewSurface : new DummySurface); CHECK_EQ(err, OK); mCamera->setPreviewCallbackFlags( diff --git a/media/libstagefright/MPEG4Writer.cpp b/media/libstagefright/MPEG4Writer.cpp index 9a7a873..367459f 100644 --- a/media/libstagefright/MPEG4Writer.cpp +++ b/media/libstagefright/MPEG4Writer.cpp @@ -75,6 +75,13 @@ MPEG4Writer::MPEG4Writer(const char *filename) CHECK(mFile != NULL); } +MPEG4Writer::MPEG4Writer(int fd) + : mFile(fdopen(fd, "wb")), + mOffset(0), + mMdatOffset(0) { + CHECK(mFile != NULL); +} + MPEG4Writer::~MPEG4Writer() { stop(); @@ -203,6 +210,27 @@ off_t MPEG4Writer::addSample(MediaBuffer *buffer) { return old_offset; } +off_t MPEG4Writer::addLengthPrefixedSample(MediaBuffer *buffer) { + Mutex::Autolock autoLock(mLock); + + off_t old_offset = mOffset; + + size_t length = buffer->range_length(); + CHECK(length < 65536); + + uint8_t x = length >> 8; + fwrite(&x, 1, 1, mFile); + x = length & 0xff; + fwrite(&x, 1, 1, mFile); + + fwrite((const uint8_t *)buffer->data() + buffer->range_offset(), + 1, length, mFile); + + mOffset += length + 2; + + return old_offset; +} + void MPEG4Writer::beginBox(const char *fourcc) { CHECK_EQ(strlen(fourcc), 4); @@ -348,11 +376,12 @@ void *MPEG4Writer::Track::ThreadWrapper(void *me) { } void MPEG4Writer::Track::threadEntry() { - bool is_mpeg4 = false; sp<MetaData> meta = mSource->getFormat(); const char *mime; meta->findCString(kKeyMIMEType, &mime); - is_mpeg4 = !strcasecmp(mime, MEDIA_MIMETYPE_VIDEO_MPEG4); + bool is_mpeg4 = !strcasecmp(mime, MEDIA_MIMETYPE_VIDEO_MPEG4); + bool is_avc = !strcasecmp(mime, MEDIA_MIMETYPE_VIDEO_AVC); + int32_t count = 0; MediaBuffer *buffer; while (!mDone && mSource->read(&buffer) == OK) { @@ -363,6 +392,55 @@ void MPEG4Writer::Track::threadEntry() { continue; } + ++count; + + if (is_avc && count < 3) { + size_t size = buffer->range_length(); + + switch (count) { + case 1: + { + CHECK_EQ(mCodecSpecificData, NULL); + mCodecSpecificData = malloc(size + 8); + uint8_t *header = (uint8_t *)mCodecSpecificData; + header[0] = 1; + header[1] = 0x42; // profile + header[2] = 0x80; + header[3] = 0x1e; // level + header[4] = 0xfc | 3; + header[5] = 0xe0 | 1; + header[6] = size >> 8; + header[7] = size & 0xff; + memcpy(&header[8], + (const uint8_t *)buffer->data() + buffer->range_offset(), + size); + + mCodecSpecificDataSize = size + 8; + break; + } + + case 2: + { + size_t offset = mCodecSpecificDataSize; + mCodecSpecificDataSize += size + 3; + mCodecSpecificData = realloc(mCodecSpecificData, mCodecSpecificDataSize); + uint8_t *header = (uint8_t *)mCodecSpecificData; + header[offset] = 1; + header[offset + 1] = size >> 8; + header[offset + 2] = size & 0xff; + memcpy(&header[offset + 3], + (const uint8_t *)buffer->data() + buffer->range_offset(), + size); + break; + } + } + + buffer->release(); + buffer = NULL; + + continue; + } + if (mCodecSpecificData == NULL && is_mpeg4) { const uint8_t *data = (const uint8_t *)buffer->data() + buffer->range_offset(); @@ -393,10 +471,11 @@ void MPEG4Writer::Track::threadEntry() { buffer->set_range(buffer->range_offset() + offset, size - offset); } - off_t offset = mOwner->addSample(buffer); + off_t offset = is_avc ? mOwner->addLengthPrefixedSample(buffer) + : mOwner->addSample(buffer); SampleInfo info; - info.size = buffer->range_length(); + info.size = is_avc ? buffer->range_length() + 2 : buffer->range_length(); info.offset = offset; int64_t timestampUs; @@ -556,6 +635,8 @@ void MPEG4Writer::Track::writeTrackHeader(int32_t trackID) { mOwner->beginBox("mp4v"); } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_H263, mime)) { mOwner->beginBox("s263"); + } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mime)) { + mOwner->beginBox("avc1"); } else { LOGE("Unknown mime type '%s'.", mime); CHECK(!"should not be here, unknown mime type."); @@ -631,8 +712,13 @@ void MPEG4Writer::Track::writeTrackHeader(int32_t trackID) { mOwner->writeInt8(0); // profile: 0 mOwner->endBox(); // d263 + } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mime)) { + mOwner->beginBox("avcC"); + mOwner->write(mCodecSpecificData, mCodecSpecificDataSize); + mOwner->endBox(); // avcC } - mOwner->endBox(); // mp4v or s263 + + mOwner->endBox(); // mp4v, s263 or avc1 } mOwner->endBox(); // stsd diff --git a/media/libstagefright/OMXCodec.cpp b/media/libstagefright/OMXCodec.cpp index 6c0367a..d36653e 100644 --- a/media/libstagefright/OMXCodec.cpp +++ b/media/libstagefright/OMXCodec.cpp @@ -594,11 +594,9 @@ void OMXCodec::setVideoInputFormat( CHECK(!"Should not be here. Not a supported video mime type."); } - OMX_COLOR_FORMATTYPE colorFormat = - 0 ? OMX_COLOR_FormatYCbYCr : OMX_COLOR_FormatCbYCrY; - - if (!strncmp("OMX.qcom.video.encoder.", mComponentName, 23)) { - colorFormat = OMX_COLOR_FormatYUV420SemiPlanar; + OMX_COLOR_FORMATTYPE colorFormat = OMX_COLOR_FormatYUV420SemiPlanar; + if (!strcasecmp("OMX.TI.Video.encoder", mComponentName)) { + colorFormat = OMX_COLOR_FormatYCbYCr; } CHECK_EQ(setVideoPortFormatType( @@ -666,6 +664,12 @@ void OMXCodec::setVideoInputFormat( case OMX_VIDEO_CodingH263: break; + case OMX_VIDEO_CodingAVC: + { + CHECK_EQ(setupAVCEncoderParameters(), OK); + break; + } + default: CHECK(!"Support for this compressionFormat to be implemented."); break; @@ -749,6 +753,64 @@ status_t OMXCodec::setupMPEG4EncoderParameters() { return OK; } +status_t OMXCodec::setupAVCEncoderParameters() { + OMX_VIDEO_PARAM_AVCTYPE h264type; + InitOMXParams(&h264type); + h264type.nPortIndex = kPortIndexOutput; + + status_t err = mOMX->getParameter( + mNode, OMX_IndexParamVideoAvc, &h264type, sizeof(h264type)); + CHECK_EQ(err, OK); + + h264type.nAllowedPictureTypes = + OMX_VIDEO_PictureTypeI | OMX_VIDEO_PictureTypeP; + + h264type.nSliceHeaderSpacing = 0; + h264type.nBFrames = 0; + h264type.bUseHadamard = OMX_TRUE; + h264type.nRefFrames = 1; + h264type.nRefIdx10ActiveMinus1 = 0; + h264type.nRefIdx11ActiveMinus1 = 0; + h264type.bEnableUEP = OMX_FALSE; + h264type.bEnableFMO = OMX_FALSE; + h264type.bEnableASO = OMX_FALSE; + h264type.bEnableRS = OMX_FALSE; + h264type.eProfile = OMX_VIDEO_AVCProfileBaseline; + h264type.eLevel = OMX_VIDEO_AVCLevel1b; + h264type.bFrameMBsOnly = OMX_TRUE; + h264type.bMBAFF = OMX_FALSE; + h264type.bEntropyCodingCABAC = OMX_FALSE; + h264type.bWeightedPPrediction = OMX_FALSE; + h264type.bconstIpred = OMX_FALSE; + h264type.bDirect8x8Inference = OMX_FALSE; + h264type.bDirectSpatialTemporal = OMX_FALSE; + h264type.nCabacInitIdc = 0; + h264type.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterEnable; + + err = mOMX->setParameter( + mNode, OMX_IndexParamVideoAvc, &h264type, sizeof(h264type)); + CHECK_EQ(err, OK); + + OMX_VIDEO_PARAM_BITRATETYPE bitrateType; + InitOMXParams(&bitrateType); + bitrateType.nPortIndex = kPortIndexOutput; + + err = mOMX->getParameter( + mNode, OMX_IndexParamVideoBitrate, + &bitrateType, sizeof(bitrateType)); + CHECK_EQ(err, OK); + + bitrateType.eControlRate = OMX_Video_ControlRateVariable; + bitrateType.nTargetBitrate = 1000000; + + err = mOMX->setParameter( + mNode, OMX_IndexParamVideoBitrate, + &bitrateType, sizeof(bitrateType)); + CHECK_EQ(err, OK); + + return OK; +} + void OMXCodec::setVideoOutputFormat( const char *mime, OMX_U32 width, OMX_U32 height) { CODEC_LOGV("setVideoOutputFormat width=%ld, height=%ld", width, height); @@ -849,7 +911,6 @@ void OMXCodec::setVideoOutputFormat( CHECK_EQ(err, OK); } - OMXCodec::OMXCodec( const sp<IOMX> &omx, IOMX::node_id node, uint32_t quirks, bool isEncoder, @@ -1178,6 +1239,9 @@ void OMXCodec::on_message(const omx_message &msg) { if (msg.u.extended_buffer_data.flags & OMX_BUFFERFLAG_SYNCFRAME) { buffer->meta_data()->setInt32(kKeyIsSyncFrame, true); } + if (msg.u.extended_buffer_data.flags & OMX_BUFFERFLAG_CODECCONFIG) { + buffer->meta_data()->setInt32(kKeyIsCodecConfig, true); + } buffer->meta_data()->setPointer( kKeyPlatformPrivate, @@ -1738,6 +1802,13 @@ void OMXCodec::drainInputBuffer(BufferInfo *info) { } info->mOwnedByComponent = true; + + // This component does not ever signal the EOS flag on output buffers, + // Thanks for nothing. + if (mSignalledEOS && !strcmp(mComponentName, "OMX.TI.Video.encoder")) { + mNoMoreOutputData = true; + mBufferFilled.signal(); + } } void OMXCodec::fillOutputBuffer(BufferInfo *info) { |