OMXCodec.cpp revision 8edb8e82fa886564ee8e72178a1969e2437dd525
1693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber/*
2693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * Copyright (C) 2009 The Android Open Source Project
3693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber *
4693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * Licensed under the Apache License, Version 2.0 (the "License");
5693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * you may not use this file except in compliance with the License.
6693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * You may obtain a copy of the License at
7693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber *
8693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber *      http://www.apache.org/licenses/LICENSE-2.0
9693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber *
10693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * Unless required by applicable law or agreed to in writing, software
11693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * distributed under the License is distributed on an "AS IS" BASIS,
12693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * See the License for the specific language governing permissions and
14693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber * limitations under the License.
15693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber */
16693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
17693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber//#define LOG_NDEBUG 0
18693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#define LOG_TAG "OMXCodec"
19693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <utils/Log.h>
20693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
21956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong#include "include/AACEncoder.h"
22b841f14f8e51f2365945281fbfa54ef6a1b1b5a6Andreas Huber#include "include/AMRNBEncoder.h"
23956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong#include "include/AMRWBEncoder.h"
2429a84457aed4c45bc900998b5e11c03023264208James Dong#include "include/AVCEncoder.h"
2559f566c4ec3dfc097ad8163523e522280b27e5c3James Dong#include "include/M4vH263Encoder.h"
2692616b5655b7aef260480f60f2aabf98e821c8f3Andreas Huber
2789e69da4d86348409994c9dafbbb2634ccd7c196Andreas Huber#include "include/ESDS.h"
2889e69da4d86348409994c9dafbbb2634ccd7c196Andreas Huber
29693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <binder/IServiceManager.h>
30693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <binder/MemoryDealer.h>
31693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <binder/ProcessState.h>
32f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber#include <media/stagefright/foundation/ADebug.h>
33693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <media/IMediaPlayerService.h>
346a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis#include <media/stagefright/HardwareAPI.h>
35693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <media/stagefright/MediaBuffer.h>
36693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <media/stagefright/MediaBufferGroup.h>
3718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber#include <media/stagefright/MediaDefs.h>
38693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <media/stagefright/MediaExtractor.h>
39693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <media/stagefright/MetaData.h>
40693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <media/stagefright/OMXCodec.h>
41bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber#include <media/stagefright/Utils.h>
42693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <utils/Vector.h>
43693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
44693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <OMX_Audio.h>
45693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#include <OMX_Component.h>
46693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
47bfd41f33c77c66ead48ee378e4ea4b7bfa5fca1fAndreas Huber#include "include/avc_utils.h"
48aae3516293e58c0b015d4109bde58c11d503433cAndreas Huber
49693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubernamespace android {
50693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
51a57a9a491272aa884494b2ec7854960827a73742James Dong// Treat time out as an error if we have not received any output
52a57a9a491272aa884494b2ec7854960827a73742James Dong// buffers after 3 seconds.
538edb8e82fa886564ee8e72178a1969e2437dd525James Dongconst static int64_t kBufferFilledEventTimeOutNs = 3000000000LL;
54a57a9a491272aa884494b2ec7854960827a73742James Dong
55693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstruct CodecInfo {
56693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    const char *mime;
57693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    const char *codec;
58693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber};
59693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
60956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong#define FACTORY_CREATE_ENCODER(name) \
61956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dongstatic sp<MediaSource> Make##name(const sp<MediaSource> &source, const sp<MetaData> &meta) { \
62956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    return new name(source, meta); \
63956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong}
64956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong
651af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber#define FACTORY_REF(name) { #name, Make##name },
661af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
67bbba88cb1bdc34705d1477208990a06904c022e7Andreas HuberFACTORY_CREATE_ENCODER(AMRNBEncoder)
68bbba88cb1bdc34705d1477208990a06904c022e7Andreas HuberFACTORY_CREATE_ENCODER(AMRWBEncoder)
69bbba88cb1bdc34705d1477208990a06904c022e7Andreas HuberFACTORY_CREATE_ENCODER(AACEncoder)
70bbba88cb1bdc34705d1477208990a06904c022e7Andreas HuberFACTORY_CREATE_ENCODER(AVCEncoder)
71bbba88cb1bdc34705d1477208990a06904c022e7Andreas HuberFACTORY_CREATE_ENCODER(M4vH263Encoder)
72bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
73956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dongstatic sp<MediaSource> InstantiateSoftwareEncoder(
74956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        const char *name, const sp<MediaSource> &source,
75956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        const sp<MetaData> &meta) {
76956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    struct FactoryInfo {
77956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        const char *name;
78956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        sp<MediaSource> (*CreateFunc)(const sp<MediaSource> &, const sp<MetaData> &);
79956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    };
80956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong
81956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    static const FactoryInfo kFactoryInfo[] = {
82956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        FACTORY_REF(AMRNBEncoder)
83956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        FACTORY_REF(AMRWBEncoder)
84956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        FACTORY_REF(AACEncoder)
8529a84457aed4c45bc900998b5e11c03023264208James Dong        FACTORY_REF(AVCEncoder)
8659f566c4ec3dfc097ad8163523e522280b27e5c3James Dong        FACTORY_REF(M4vH263Encoder)
87956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    };
88956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    for (size_t i = 0;
89956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong         i < sizeof(kFactoryInfo) / sizeof(kFactoryInfo[0]); ++i) {
90956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        if (!strcmp(name, kFactoryInfo[i].name)) {
91956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return (*kFactoryInfo[i].CreateFunc)(source, meta);
92956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        }
93956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    }
94956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong
95956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    return NULL;
96956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong}
971af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
981af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber#undef FACTORY_REF
991af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber#undef FACTORY_CREATE
1001af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
101693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const CodecInfo kDecoderInfo[] = {
10218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_IMAGE_JPEG, "OMX.TI.JPEG.decode" },
1030b5ba9eeed56a80fed3735f5cd4951477fda79f0James Dong//    { MEDIA_MIMETYPE_AUDIO_MPEG, "OMX.TI.MP3.decode" },
104bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_AUDIO_MPEG, "OMX.google.mp3.decoder" },
1057f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber//    { MEDIA_MIMETYPE_AUDIO_AMR_NB, "OMX.TI.AMR.decode" },
10627fdb181daebec3dbe477080adad94f81ed667adAndreas Huber//    { MEDIA_MIMETYPE_AUDIO_AMR_NB, "OMX.Nvidia.amr.decoder" },
107bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AMR_NB, "OMX.google.amrnb.decoder" },
10827fdb181daebec3dbe477080adad94f81ed667adAndreas Huber//    { MEDIA_MIMETYPE_AUDIO_AMR_NB, "OMX.Nvidia.amrwb.decoder" },
10918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AMR_WB, "OMX.TI.WBAMR.decode" },
110bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AMR_WB, "OMX.google.amrwb.decoder" },
11127fdb181daebec3dbe477080adad94f81ed667adAndreas Huber//    { MEDIA_MIMETYPE_AUDIO_AAC, "OMX.Nvidia.aac.decoder" },
11218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AAC, "OMX.TI.AAC.decode" },
113bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AAC, "OMX.google.aac.decoder" },
114bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_AUDIO_G711_ALAW, "OMX.google.g711.alaw.decoder" },
115bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_AUDIO_G711_MLAW, "OMX.google.g711.mlaw.decoder" },
116cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.TI.DUCATI1.VIDEO.DECODER" },
11709ddead9fcb391efd718738245455398a7ec6887Andreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.Nvidia.mp4.decode" },
118ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.qcom.7x30.video.decoder.mpeg4" },
11918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.qcom.video.decoder.mpeg4" },
12018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.TI.Video.Decoder" },
121095916d92ea2378aea6fc35e80e368172c02021bAndreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.SEC.MPEG4.Decoder" },
122bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.google.mpeg4.decoder" },
123cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.TI.DUCATI1.VIDEO.DECODER" },
12409ddead9fcb391efd718738245455398a7ec6887Andreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.Nvidia.h263.decode" },
125ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.qcom.7x30.video.decoder.h263" },
12618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.qcom.video.decoder.h263" },
127095916d92ea2378aea6fc35e80e368172c02021bAndreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.SEC.H263.Decoder" },
128bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.google.h263.decoder" },
129cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.TI.DUCATI1.VIDEO.DECODER" },
130ea659e51af658d77bef7b88a2fb542ec2d69e032pgudadhe    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.Nvidia.h264.decode" },
131ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.qcom.7x30.video.decoder.avc" },
13218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.qcom.video.decoder.avc" },
13318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.TI.Video.Decoder" },
134095916d92ea2378aea6fc35e80e368172c02021bAndreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.SEC.AVC.Decoder" },
1350c1bc742181ded4930842b46e9507372f0b1b963James Dong    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.google.h264.decoder" },
136bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.google.avc.decoder" },
137bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_AUDIO_VORBIS, "OMX.google.vorbis.decoder" },
138bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    { MEDIA_MIMETYPE_VIDEO_VPX, "OMX.google.vpx.decoder" },
139386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG2, "OMX.Nvidia.mpeg2v.decode" },
140693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber};
141693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
142693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const CodecInfo kEncoderInfo[] = {
14318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AMR_NB, "OMX.TI.AMR.encode" },
144b25e2a948c8b5a96e284069a908c2fddb49efc48Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AMR_NB, "AMRNBEncoder" },
14518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AMR_WB, "OMX.TI.WBAMR.encode" },
146956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    { MEDIA_MIMETYPE_AUDIO_AMR_WB, "AMRWBEncoder" },
14718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_AUDIO_AAC, "OMX.TI.AAC.encode" },
148956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    { MEDIA_MIMETYPE_AUDIO_AAC, "AACEncoder" },
1491374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.TI.DUCATI1.VIDEO.MPEG4E" },
150ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.qcom.7x30.video.encoder.mpeg4" },
15118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.qcom.video.encoder.mpeg4" },
15218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.TI.Video.encoder" },
1536d59aa4bb8e9734a9197290a027356e88552f59aJames Dong    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.Nvidia.mp4.encoder" },
154095916d92ea2378aea6fc35e80e368172c02021bAndreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, "OMX.SEC.MPEG4.Encoder" },
15559f566c4ec3dfc097ad8163523e522280b27e5c3James Dong    { MEDIA_MIMETYPE_VIDEO_MPEG4, "M4vH263Encoder" },
1561374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.TI.DUCATI1.VIDEO.MPEG4E" },
157ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.qcom.7x30.video.encoder.h263" },
15818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.qcom.video.encoder.h263" },
15918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.TI.Video.encoder" },
1606d59aa4bb8e9734a9197290a027356e88552f59aJames Dong    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.Nvidia.h263.encoder" },
161095916d92ea2378aea6fc35e80e368172c02021bAndreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, "OMX.SEC.H263.Encoder" },
16259f566c4ec3dfc097ad8163523e522280b27e5c3James Dong    { MEDIA_MIMETYPE_VIDEO_H263, "M4vH263Encoder" },
1631374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.TI.DUCATI1.VIDEO.H264E" },
164ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.qcom.7x30.video.encoder.avc" },
16503b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.qcom.video.encoder.avc" },
16618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.TI.Video.encoder" },
1674937be23b485ce8edf180ff5e71ebf0884178f17pgudadhe    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.Nvidia.h264.encoder" },
168095916d92ea2378aea6fc35e80e368172c02021bAndreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, "OMX.SEC.AVC.Encoder" },
16929a84457aed4c45bc900998b5e11c03023264208James Dong    { MEDIA_MIMETYPE_VIDEO_AVC, "AVCEncoder" },
170693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber};
171693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1721af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber#undef OPTIONAL
1731af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
1748cd11d23197209e2c2bdb7980bc17e3f1257a118Andreas Huber#define CODEC_LOGI(x, ...) LOGI("[%s] "x, mComponentName, ##__VA_ARGS__)
1757a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber#define CODEC_LOGV(x, ...) LOGV("[%s] "x, mComponentName, ##__VA_ARGS__)
176c017cfcc24dd63c7efde5747a6ed2d4053788abcAndreas Huber#define CODEC_LOGE(x, ...) LOGE("[%s] "x, mComponentName, ##__VA_ARGS__)
1777a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
178693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstruct OMXCodecObserver : public BnOMXObserver {
179318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMXCodecObserver() {
180318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
181318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
182318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    void setCodec(const sp<OMXCodec> &target) {
183318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        mTarget = target;
184693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
185693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
186693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    // from IOMXObserver
187318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    virtual void onMessage(const omx_message &msg) {
188693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        sp<OMXCodec> codec = mTarget.promote();
189693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
190693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (codec.get() != NULL) {
19114c858e80dfe2030c9f343dc0c6e2048e030731bJames Dong            Mutex::Autolock autoLock(codec->mLock);
192693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            codec->on_message(msg);
19314c858e80dfe2030c9f343dc0c6e2048e030731bJames Dong            codec.clear();
194693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
195693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
196693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
197693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberprotected:
198693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    virtual ~OMXCodecObserver() {}
199693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
200693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberprivate:
201693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    wp<OMXCodec> mTarget;
202693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
203693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMXCodecObserver(const OMXCodecObserver &);
204693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMXCodecObserver &operator=(const OMXCodecObserver &);
205693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber};
206693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
207693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const char *GetCodec(const CodecInfo *info, size_t numInfos,
208693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                            const char *mime, int index) {
209693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CHECK(index >= 0);
210693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for(size_t i = 0; i < numInfos; ++i) {
211693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (!strcasecmp(mime, info[i].mime)) {
212693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (index == 0) {
213693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                return info[i].codec;
214693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
215693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
216693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            --index;
217693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
218693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
219693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
220693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return NULL;
221693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
222693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2237a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Hubertemplate<class T>
2247a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huberstatic void InitOMXParams(T *params) {
2257a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    params->nSize = sizeof(T);
2267a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    params->nVersion.s.nVersionMajor = 1;
2277a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    params->nVersion.s.nVersionMinor = 0;
2287a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    params->nVersion.s.nRevision = 0;
2297a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    params->nVersion.s.nStep = 0;
2307a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber}
2317a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
232ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huberstatic bool IsSoftwareCodec(const char *componentName) {
233bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    if (!strncmp("OMX.google.", componentName, 11)) {
234bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        return true;
235bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
236bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
237c810b75fd04eef8af509bb42aa8837566a374b32James Dong    if (!strncmp("OMX.", componentName, 4)) {
238c810b75fd04eef8af509bb42aa8837566a374b32James Dong        return false;
239ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    }
240693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
241c810b75fd04eef8af509bb42aa8837566a374b32James Dong    return true;
242ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber}
243693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
244bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber// A sort order in which OMX software codecs are first, followed
245bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber// by other (non-OMX) software codecs, followed by everything else.
246ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huberstatic int CompareSoftwareCodecsFirst(
247ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        const String8 *elem1, const String8 *elem2) {
248bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    bool isOMX1 = !strncmp(elem1->string(), "OMX.", 4);
249bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    bool isOMX2 = !strncmp(elem2->string(), "OMX.", 4);
2501af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
251ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    bool isSoftwareCodec1 = IsSoftwareCodec(elem1->string());
252ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    bool isSoftwareCodec2 = IsSoftwareCodec(elem2->string());
253693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
254ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    if (isSoftwareCodec1) {
255bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (!isSoftwareCodec2) { return -1; }
256bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
257bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (isOMX1) {
258bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (isOMX2) { return 0; }
259bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
260bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            return -1;
261bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        } else {
262bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (isOMX2) { return 0; }
263bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
264bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            return 1;
265bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
266bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
267ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        return -1;
268ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    }
269693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
270ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    if (isSoftwareCodec2) {
271ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        return 1;
272693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
273693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
274ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    return 0;
275ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber}
276ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
277ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber// static
2782b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huberuint32_t OMXCodec::getComponentQuirks(
2792b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber        const char *componentName, bool isEncoder) {
280693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    uint32_t quirks = 0;
281ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
2828d9dda1ba28bbcd7730a9bec10ce70db4912688eDima Zavin    if (!strcmp(componentName, "OMX.Nvidia.amr.decoder") ||
2838d9dda1ba28bbcd7730a9bec10ce70db4912688eDima Zavin         !strcmp(componentName, "OMX.Nvidia.amrwb.decoder") ||
2848d9dda1ba28bbcd7730a9bec10ce70db4912688eDima Zavin         !strcmp(componentName, "OMX.Nvidia.aac.decoder") ||
2858d9dda1ba28bbcd7730a9bec10ce70db4912688eDima Zavin         !strcmp(componentName, "OMX.Nvidia.mp3.decoder")) {
2868d9dda1ba28bbcd7730a9bec10ce70db4912688eDima Zavin        quirks |= kDecoderLiesAboutNumberOfChannels;
2878d9dda1ba28bbcd7730a9bec10ce70db4912688eDima Zavin    }
2888d9dda1ba28bbcd7730a9bec10ce70db4912688eDima Zavin
289693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (!strcmp(componentName, "OMX.TI.MP3.decode")) {
290693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        quirks |= kNeedsFlushBeforeDisable;
29178d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber        quirks |= kDecoderLiesAboutNumberOfChannels;
292693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
293693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (!strcmp(componentName, "OMX.TI.AAC.decode")) {
294693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        quirks |= kNeedsFlushBeforeDisable;
2951beb760d920561679862ded945a04e370368c7f7Andreas Huber        quirks |= kRequiresFlushCompleteEmulation;
2967f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        quirks |= kSupportsMultipleFramesPerInputBuffer;
297693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
298693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (!strncmp(componentName, "OMX.qcom.video.encoder.", 23)) {
299693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        quirks |= kRequiresLoadedToIdleAfterAllocation;
300693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        quirks |= kRequiresAllocateBufferOnInputPorts;
301888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        quirks |= kRequiresAllocateBufferOnOutputPorts;
302824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong        if (!strncmp(componentName, "OMX.qcom.video.encoder.avc", 26)) {
303824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong
304824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong            // The AVC encoder advertises the size of output buffers
305824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong            // based on the input video resolution and assumes
306824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong            // the worst/least compression ratio is 0.5. It is found that
307824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong            // sometimes, the output buffer size is larger than
308824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong            // size advertised by the encoder.
309824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong            quirks |= kRequiresLargerEncoderOutputBuffer;
310824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong        }
311693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
312ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    if (!strncmp(componentName, "OMX.qcom.7x30.video.encoder.", 28)) {
313ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    }
314b7802d9ee9c9dbb471ae4f010bd67ba1aaa9f264Andreas Huber    if (!strncmp(componentName, "OMX.qcom.video.decoder.", 23)) {
315b7802d9ee9c9dbb471ae4f010bd67ba1aaa9f264Andreas Huber        quirks |= kRequiresAllocateBufferOnOutputPorts;
316213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber        quirks |= kDefersOutputBufferAllocation;
317b7802d9ee9c9dbb471ae4f010bd67ba1aaa9f264Andreas Huber    }
318ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    if (!strncmp(componentName, "OMX.qcom.7x30.video.decoder.", 28)) {
319ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber        quirks |= kRequiresAllocateBufferOnInputPorts;
320ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber        quirks |= kRequiresAllocateBufferOnOutputPorts;
321ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber        quirks |= kDefersOutputBufferAllocation;
322ae1f0036bd9cc52353fb3a8ad4556792e1e8047aAndreas Huber    }
323693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
324cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan    if (!strcmp(componentName, "OMX.TI.DUCATI1.VIDEO.DECODER")) {
325cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan        quirks |= kRequiresAllocateBufferOnInputPorts;
326cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan        quirks |= kRequiresAllocateBufferOnOutputPorts;
3271374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    }
3281374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket
3291374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    // FIXME:
3301374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    // Remove the quirks after the work is done.
3311374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    else if (!strcmp(componentName, "OMX.TI.DUCATI1.VIDEO.MPEG4E") ||
3321374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket             !strcmp(componentName, "OMX.TI.DUCATI1.VIDEO.H264E")) {
3331374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket
3341374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket        quirks |= kRequiresAllocateBufferOnInputPorts;
3351374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket        quirks |= kRequiresAllocateBufferOnOutputPorts;
3361374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    }
3371374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    else if (!strncmp(componentName, "OMX.TI.", 7)) {
338df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber        // Apparently I must not use OMX_UseBuffer on either input or
339df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber        // output ports on any of the TI components or quote:
340df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber        // "(I) may have unexpected problem (sic) which can be timing related
341df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber        //  and hard to reproduce."
342df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber
343df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber        quirks |= kRequiresAllocateBufferOnInputPorts;
344df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber        quirks |= kRequiresAllocateBufferOnOutputPorts;
3458aa8fe5ea704b05d8f0ab3d7bf18de18151f1b50James Dong        if (!strncmp(componentName, "OMX.TI.Video.encoder", 20)) {
346d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong            quirks |= kAvoidMemcopyInputRecordingFrames;
347d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong        }
348df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber    }
349df4de7d7bd0060a84b064ce074d3f86d3a7aa0aeAndreas Huber
35086559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber    if (!strcmp(componentName, "OMX.TI.Video.Decoder")) {
35186559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber        quirks |= kInputBufferSizesAreBogus;
35286559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber    }
35386559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber
3542b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber    if (!strncmp(componentName, "OMX.SEC.", 8) && !isEncoder) {
3552b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber        // These output buffers contain no video data, just some
3562b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber        // opaque information that allows the overlay to display their
3572b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber        // contents.
3582b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber        quirks |= kOutputBuffersAreUnreadable;
3592b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber    }
3602b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber
361ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    return quirks;
362ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber}
363ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
364ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber// static
365ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Hubervoid OMXCodec::findMatchingCodecs(
366ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        const char *mime,
367ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        bool createEncoder, const char *matchComponentName,
368ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        uint32_t flags,
369ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        Vector<String8> *matchingCodecs) {
370ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    matchingCodecs->clear();
371ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
372ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    for (int index = 0;; ++index) {
373ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        const char *componentName;
374ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
375ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        if (createEncoder) {
376ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber            componentName = GetCodec(
377ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber                    kEncoderInfo,
378ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber                    sizeof(kEncoderInfo) / sizeof(kEncoderInfo[0]),
379ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber                    mime, index);
380ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        } else {
381ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber            componentName = GetCodec(
382ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber                    kDecoderInfo,
383ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber                    sizeof(kDecoderInfo) / sizeof(kDecoderInfo[0]),
384ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber                    mime, index);
385ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        }
386ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
387ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        if (!componentName) {
388ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber            break;
389ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        }
390ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
391ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        // If a specific codec is requested, skip the non-matching ones.
392ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        if (matchComponentName && strcmp(componentName, matchComponentName)) {
393ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber            continue;
394ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        }
395ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
396d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong        // When requesting software-only codecs, only push software codecs
397d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong        // When requesting hardware-only codecs, only push hardware codecs
398d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong        // When there is request neither for software-only nor for
399d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong        // hardware-only codecs, push all codecs
400d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong        if (((flags & kSoftwareCodecsOnly) &&   IsSoftwareCodec(componentName)) ||
401d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong            ((flags & kHardwareCodecsOnly) &&  !IsSoftwareCodec(componentName)) ||
402d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong            (!(flags & (kSoftwareCodecsOnly | kHardwareCodecsOnly)))) {
403d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong
404d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong            matchingCodecs->push(String8(componentName));
405d332a72e8f71caea0d3dc898db6a9d7a929fad32James Dong        }
406ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    }
407ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
408ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    if (flags & kPreferSoftwareCodecs) {
409ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        matchingCodecs->sort(CompareSoftwareCodecsFirst);
410ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    }
411ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber}
412ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
413ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber// static
414bf697e37550d9e8376089b0b5e498613bede798cAndreas Hubersp<MediaSource> OMXCodec::Create(
415ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        const sp<IOMX> &omx,
416ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        const sp<MetaData> &meta, bool createEncoder,
417ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        const sp<MediaSource> &source,
418ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        const char *matchComponentName,
4196a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        uint32_t flags,
4206a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        const sp<ANativeWindow> &nativeWindow) {
4210bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    int32_t requiresSecureBuffers;
4220bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if (source->getFormat()->findInt32(
4230bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                kKeyRequiresSecureBuffers,
4240bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                &requiresSecureBuffers)
4250bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            && requiresSecureBuffers) {
4260bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        flags |= kIgnoreCodecSpecificData;
4270bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        flags |= kUseSecureInputBuffers;
4280bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    }
4290bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
430ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    const char *mime;
431ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    bool success = meta->findCString(kKeyMIMEType, &mime);
432ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    CHECK(success);
433ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
434ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    Vector<String8> matchingCodecs;
435ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    findMatchingCodecs(
436ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber            mime, createEncoder, matchComponentName, flags, &matchingCodecs);
437ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
438ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    if (matchingCodecs.isEmpty()) {
439ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        return NULL;
440ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    }
441ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
442ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    sp<OMXCodecObserver> observer = new OMXCodecObserver;
443ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    IOMX::node_id node = 0;
444ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
445ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber    for (size_t i = 0; i < matchingCodecs.size(); ++i) {
446c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber        const char *componentNameBase = matchingCodecs[i].string();
447c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber        const char *componentName = componentNameBase;
448c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber
449c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber        AString tmp;
450c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber        if (flags & kUseSecureInputBuffers) {
451c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber            tmp = componentNameBase;
452c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber            tmp.append(".secure");
453c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber
454c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber            componentName = tmp.c_str();
455c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber        }
456ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
45787e031caf905145cb6e362a8c60f02736b87deefAndreas Huber        if (createEncoder) {
45887e031caf905145cb6e362a8c60f02736b87deefAndreas Huber            sp<MediaSource> softwareCodec =
45987e031caf905145cb6e362a8c60f02736b87deefAndreas Huber                InstantiateSoftwareEncoder(componentName, source, meta);
4601af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
46187e031caf905145cb6e362a8c60f02736b87deefAndreas Huber            if (softwareCodec != NULL) {
46287e031caf905145cb6e362a8c60f02736b87deefAndreas Huber                LOGV("Successfully allocated software codec '%s'", componentName);
4631af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
46487e031caf905145cb6e362a8c60f02736b87deefAndreas Huber                return softwareCodec;
46587e031caf905145cb6e362a8c60f02736b87deefAndreas Huber            }
4661af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber        }
4671af80bc3fefb999756ef4847e72b2dcfd0b88a4aAndreas Huber
468ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        LOGV("Attempting to allocate OMX node '%s'", componentName);
469ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
470c4847184f0d351ca2587ca66dc5ab8eab4dbe207Andreas Huber        uint32_t quirks = getComponentQuirks(componentNameBase, createEncoder);
471f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
472f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        if (!createEncoder
473f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                && (quirks & kOutputBuffersAreUnreadable)
474f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                && (flags & kClientNeedsFramebuffer)) {
475f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber            if (strncmp(componentName, "OMX.SEC.", 8)) {
476f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                // For OMX.SEC.* decoders we can enable a special mode that
477f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                // gives the client access to the framebuffer contents.
478f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
479f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                LOGW("Component '%s' does not give the client access to "
480f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                     "the framebuffer contents. Skipping.",
481f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                     componentName);
482f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
483f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                continue;
484f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber            }
485f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        }
486f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
487ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        status_t err = omx->allocateNode(componentName, observer, &node);
488ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber        if (err == OK) {
489ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber            LOGV("Successfully allocated OMX node '%s'", componentName);
490ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
4912a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            sp<OMXCodec> codec = new OMXCodec(
4920bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    omx, node, quirks, flags,
4932a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber                    createEncoder, mime, componentName,
4946a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                    source, nativeWindow);
495ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
4962a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            observer->setCodec(codec);
4972a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber
4980bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            err = codec->configureCodec(meta);
4992a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber
5002a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            if (err == OK) {
501386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber                if (!strcmp("OMX.Nvidia.mpeg2v.decode", componentName)) {
5020bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    codec->mFlags |= kOnlySubmitOneInputBufferAtOneTime;
503386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber                }
504386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber
5052a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber                return codec;
5062a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            }
507ba229b3faa98b3bb6567cf95040062be2d9c2b3fAndreas Huber
5082a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            LOGV("Failed to configure codec '%s'", componentName);
5092a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        }
5102a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    }
511693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5122a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    return NULL;
5132a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber}
514318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
5150bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huberstatus_t OMXCodec::configureCodec(const sp<MetaData> &meta) {
5160bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    LOGV("configureCodec protected=%d",
5170bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber         (mFlags & kEnableGrallocUsageProtected) ? 1 : 0);
5187757f5010a771fb8824b6fdf9788f588a1577e3fJames Dong
5190bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if (!(mFlags & kIgnoreCodecSpecificData)) {
5202a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber        uint32_t type;
5212a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber        const void *data;
5222a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber        size_t size;
5232a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber        if (meta->findData(kKeyESDS, &type, &data, &size)) {
5242a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            ESDS esds((const char *)data, size);
525f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ(esds.InitCheck(), (status_t)OK);
526693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5272a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            const void *codec_specific_data;
5282a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            size_t codec_specific_data_size;
5292a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            esds.getCodecSpecificInfo(
5302a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                    &codec_specific_data, &codec_specific_data_size);
531693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5322a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            addCodecSpecificData(
5332a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                    codec_specific_data, codec_specific_data_size);
5342a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber        } else if (meta->findData(kKeyAVCC, &type, &data, &size)) {
5352a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            // Parse the AVCDecoderConfigurationRecord
536bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5372a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            const uint8_t *ptr = (const uint8_t *)data;
538bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5392a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            CHECK(size >= 7);
540f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((unsigned)ptr[0], 1u);  // configurationVersion == 1
5412a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            uint8_t profile = ptr[1];
5422a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            uint8_t level = ptr[3];
543bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5442a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            // There is decodable content out there that fails the following
5452a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            // assertion, let's be lenient for now...
5462a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            // CHECK((ptr[4] >> 2) == 0x3f);  // reserved
547bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5482a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            size_t lengthSize = 1 + (ptr[4] & 3);
549bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5502a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            // commented out check below as H264_QVGA_500_NO_AUDIO.3gp
5512a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            // violates it...
5522a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            // CHECK((ptr[5] >> 5) == 7);  // reserved
553bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5542a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            size_t numSeqParameterSets = ptr[5] & 31;
555bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5562a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            ptr += 6;
5572a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            size -= 6;
558bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5592a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            for (size_t i = 0; i < numSeqParameterSets; ++i) {
5602a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                CHECK(size >= 2);
5612a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                size_t length = U16_AT(ptr);
562693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5632a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                ptr += 2;
5642a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                size -= 2;
565693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5662a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                CHECK(size >= length);
567bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5682a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                addCodecSpecificData(ptr, length);
569bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5702a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                ptr += length;
5712a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                size -= length;
5722a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            }
573bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5742a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            CHECK(size >= 1);
5752a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            size_t numPictureParameterSets = *ptr;
5762a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            ++ptr;
5772a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            --size;
578bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5792a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            for (size_t i = 0; i < numPictureParameterSets; ++i) {
5802a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                CHECK(size >= 2);
5812a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                size_t length = U16_AT(ptr);
582bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
5832a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                ptr += 2;
5842a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                size -= 2;
585693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5862a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                CHECK(size >= length);
587693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5882a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                addCodecSpecificData(ptr, length);
589693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5902a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                ptr += length;
5912a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                size -= length;
5922a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            }
593693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
59407e946ce4908eea5eebb30ea89f088fc7c679b5aAndreas Huber            CODEC_LOGI(
5952a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                    "AVC profile = %d (%s), level = %d",
5962a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                    (int)profile, AVCProfileToString(profile), level);
597693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
5982a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            if (!strcmp(mComponentName, "OMX.TI.Video.Decoder")
5992a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                && (profile != kAVCProfileBaseline || level > 30)) {
6002a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                // This stream exceeds the decoder's capabilities. The decoder
6012a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                // does not handle this gracefully and would clobber the heap
6022a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                // and wreak havoc instead...
603bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
6042a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                LOGE("Profile and/or level exceed the decoder's capabilities.");
6052a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber                return ERROR_UNSUPPORTED;
6062a4d22d79e927f2245537921e10fc5fda1c47a29Andreas Huber            }
607bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        } else if (meta->findData(kKeyVorbisInfo, &type, &data, &size)) {
608bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            addCodecSpecificData(data, size);
609bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
610bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            CHECK(meta->findData(kKeyVorbisBooks, &type, &data, &size));
611bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            addCodecSpecificData(data, size);
612693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
613693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
614693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
615956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    int32_t bitRate = 0;
616956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    if (mIsEncoder) {
617956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        CHECK(meta->findInt32(kKeyBitRate, &bitRate));
618956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    }
6192a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    if (!strcasecmp(MEDIA_MIMETYPE_AUDIO_AMR_NB, mMIME)) {
620956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        setAMRFormat(false /* isWAMR */, bitRate);
621bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_AUDIO_AMR_WB, mMIME)) {
622956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        setAMRFormat(true /* isWAMR */, bitRate);
623bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_AUDIO_AAC, mMIME)) {
62437940eefcba982836b579fe1ffec6cada72b0974Andreas Huber        int32_t numChannels, sampleRate;
62537940eefcba982836b579fe1ffec6cada72b0974Andreas Huber        CHECK(meta->findInt32(kKeyChannelCount, &numChannels));
62637940eefcba982836b579fe1ffec6cada72b0974Andreas Huber        CHECK(meta->findInt32(kKeySampleRate, &sampleRate));
62737940eefcba982836b579fe1ffec6cada72b0974Andreas Huber
628956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        setAACFormat(numChannels, sampleRate, bitRate);
629bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_AUDIO_G711_ALAW, mMIME)
630bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            || !strcasecmp(MEDIA_MIMETYPE_AUDIO_G711_MLAW, mMIME)) {
631bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        // These are PCM-like formats with a fixed sample rate but
632bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        // a variable number of channels.
633bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
634bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        int32_t numChannels;
635bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        CHECK(meta->findInt32(kKeyChannelCount, &numChannels));
636bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
637bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        setG711Format(numChannels);
638693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
639050b28a593350047845a45a14cc5026221ac1620James Dong
6402a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    if (!strncasecmp(mMIME, "video/", 6)) {
641693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
6422a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        if (mIsEncoder) {
643ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong            setVideoInputFormat(mMIME, meta);
644693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        } else {
645ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong            int32_t width, height;
646ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong            bool success = meta->findInt32(kKeyWidth, &width);
647ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong            success = success && meta->findInt32(kKeyHeight, &height);
648ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong            CHECK(success);
6492a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            status_t err = setVideoOutputFormat(
6502a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber                    mMIME, width, height);
6512a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber
6522a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            if (err != OK) {
6532a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber                return err;
6542a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            }
655693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
656693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
6577f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
6582a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    if (!strcasecmp(mMIME, MEDIA_MIMETYPE_IMAGE_JPEG)
6592a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        && !strcmp(mComponentName, "OMX.TI.JPEG.decode")) {
660693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        OMX_COLOR_FORMATTYPE format =
661693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_COLOR_Format32bitARGB8888;
662693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            // OMX_COLOR_FormatYUV420PackedPlanar;
663693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            // OMX_COLOR_FormatCbYCrY;
664693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            // OMX_COLOR_FormatYUV411Planar;
665693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
666693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        int32_t width, height;
667693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        bool success = meta->findInt32(kKeyWidth, &width);
668693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        success = success && meta->findInt32(kKeyHeight, &height);
6697530e9c708275c273c134c36c68179f511c1940eAndreas Huber
6707530e9c708275c273c134c36c68179f511c1940eAndreas Huber        int32_t compressedSize;
6717530e9c708275c273c134c36c68179f511c1940eAndreas Huber        success = success && meta->findInt32(
672e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                kKeyMaxInputSize, &compressedSize);
6737530e9c708275c273c134c36c68179f511c1940eAndreas Huber
6747530e9c708275c273c134c36c68179f511c1940eAndreas Huber        CHECK(success);
6757530e9c708275c273c134c36c68179f511c1940eAndreas Huber        CHECK(compressedSize > 0);
676693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
6772a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        setImageOutputFormat(format, width, height);
6782a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        setJPEGInputFormat(width, height, (OMX_U32)compressedSize);
679693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
680693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
681e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    int32_t maxInputSize;
682738c4315859395bfeeaae3d4c9d6fb9f414778f1Andreas Huber    if (meta->findInt32(kKeyMaxInputSize, &maxInputSize)) {
6832a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        setMinBufferSize(kPortIndexInput, (OMX_U32)maxInputSize);
684e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    }
685e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
6862a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    if (!strcmp(mComponentName, "OMX.TI.AMR.encode")
687050b28a593350047845a45a14cc5026221ac1620James Dong        || !strcmp(mComponentName, "OMX.TI.WBAMR.encode")
688050b28a593350047845a45a14cc5026221ac1620James Dong        || !strcmp(mComponentName, "OMX.TI.AAC.encode")) {
6892a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        setMinBufferSize(kPortIndexOutput, 8192);  // XXX
690e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    }
691e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
6922a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    initOutputFormat(meta);
693693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
6940bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if ((mFlags & kClientNeedsFramebuffer)
695f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber            && !strncmp(mComponentName, "OMX.SEC.", 8)) {
696f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        OMX_INDEXTYPE index;
697f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
698f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        status_t err =
699f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber            mOMX->getExtensionIndex(
700f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                    mNode,
701f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                    "OMX.SEC.index.ThumbnailMode",
702f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                    &index);
703f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
704f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        if (err != OK) {
705f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber            return err;
706f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        }
707f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
708f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        OMX_BOOL enable = OMX_TRUE;
709f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        err = mOMX->setConfig(mNode, index, &enable, sizeof(enable));
710f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
711f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        if (err != OK) {
712f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber            CODEC_LOGE("setConfig('OMX.SEC.index.ThumbnailMode') "
713f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber                       "returned error 0x%08x", err);
714f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
715f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber            return err;
716f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        }
717f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
718f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber        mQuirks &= ~kOutputBuffersAreUnreadable;
719f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber    }
720f3712f026aad1fc46b1df18d1dba718281e39726Andreas Huber
721bf2ba2a97927c24d14c0e71158abe7b49c557c68Jamie Gennis    if (mNativeWindow != NULL
722bf2ba2a97927c24d14c0e71158abe7b49c557c68Jamie Gennis        && !mIsEncoder
7236a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        && !strncasecmp(mMIME, "video/", 6)
7246a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        && !strncmp(mComponentName, "OMX.", 4)) {
7256a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        status_t err = initNativeWindow();
7266a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        if (err != OK) {
7276a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            return err;
7286a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        }
7296a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
7306a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
7312a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    return OK;
732693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
733693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
734e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Hubervoid OMXCodec::setMinBufferSize(OMX_U32 portIndex, OMX_U32 size) {
735e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
7367a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
737e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    def.nPortIndex = portIndex;
738e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
739318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->getParameter(
740e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
741f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
742e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
74386559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber    if ((portIndex == kPortIndexInput && (mQuirks & kInputBufferSizesAreBogus))
74486559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber        || (def.nBufferSize < size)) {
745e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        def.nBufferSize = size;
746e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    }
747e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
748318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->setParameter(
749e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
750f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
751738c4315859395bfeeaae3d4c9d6fb9f414778f1Andreas Huber
752738c4315859395bfeeaae3d4c9d6fb9f414778f1Andreas Huber    err = mOMX->getParameter(
753738c4315859395bfeeaae3d4c9d6fb9f414778f1Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
754f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
755738c4315859395bfeeaae3d4c9d6fb9f414778f1Andreas Huber
756738c4315859395bfeeaae3d4c9d6fb9f414778f1Andreas Huber    // Make sure the setting actually stuck.
75786559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber    if (portIndex == kPortIndexInput
75886559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber            && (mQuirks & kInputBufferSizesAreBogus)) {
75986559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber        CHECK_EQ(def.nBufferSize, size);
76086559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber    } else {
76186559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber        CHECK(def.nBufferSize >= size);
76286559de0275b46a9bdba06b749fbd549ad2022d1Andreas Huber    }
763e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber}
764e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
765693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatus_t OMXCodec::setVideoPortFormatType(
766693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        OMX_U32 portIndex,
767693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        OMX_VIDEO_CODINGTYPE compressionFormat,
768693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        OMX_COLOR_FORMATTYPE colorFormat) {
769693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_VIDEO_PARAM_PORTFORMATTYPE format;
7707a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&format);
771693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    format.nPortIndex = portIndex;
772693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    format.nIndex = 0;
773693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    bool found = false;
774693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
775693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_U32 index = 0;
776693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (;;) {
777693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        format.nIndex = index;
778318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        status_t err = mOMX->getParameter(
779693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mNode, OMX_IndexParamVideoPortFormat,
780693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                &format, sizeof(format));
781693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
782693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (err != OK) {
783693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            return err;
784693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
785693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
786693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        // The following assertion is violated by TI's video decoder.
7877530e9c708275c273c134c36c68179f511c1940eAndreas Huber        // CHECK_EQ(format.nIndex, index);
788693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
789693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#if 1
7902a4a7d5af053a17586a262a1267ba993e31790f1Andreas Huber        CODEC_LOGV("portIndex: %ld, index: %ld, eCompressionFormat=%d eColorFormat=%d",
791693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber             portIndex,
792693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber             index, format.eCompressionFormat, format.eColorFormat);
793693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#endif
794693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
795693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (!strcmp("OMX.TI.Video.encoder", mComponentName)) {
796693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (portIndex == kPortIndexInput
797693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    && colorFormat == format.eColorFormat) {
798693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                // eCompressionFormat does not seem right.
799693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                found = true;
800693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                break;
801693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
802693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (portIndex == kPortIndexOutput
803693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    && compressionFormat == format.eCompressionFormat) {
804693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                // eColorFormat does not seem right.
805693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                found = true;
806693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                break;
807693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
808693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
809693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
810693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (format.eCompressionFormat == compressionFormat
8110c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi                && format.eColorFormat == colorFormat) {
812693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            found = true;
813693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
814693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
815693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
816693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        ++index;
817693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
818693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
819693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (!found) {
820693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return UNKNOWN_ERROR;
821693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
822693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
8232a4a7d5af053a17586a262a1267ba993e31790f1Andreas Huber    CODEC_LOGV("found a match.");
824318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->setParameter(
825693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
826693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            &format, sizeof(format));
827693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
828693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return err;
829693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
830693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
831888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huberstatic size_t getFrameSize(
832888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        OMX_COLOR_FORMATTYPE colorFormat, int32_t width, int32_t height) {
833888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    switch (colorFormat) {
834888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        case OMX_COLOR_FormatYCbYCr:
835888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        case OMX_COLOR_FormatCbYCrY:
836888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            return width * height * 2;
837888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
83803b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber        case OMX_COLOR_FormatYUV420Planar:
839888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        case OMX_COLOR_FormatYUV420SemiPlanar:
8401374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket        case OMX_TI_COLOR_FormatYUV420PackedSemiPlanar:
8410c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        /*
8420c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        * FIXME: For the Opaque color format, the frame size does not
8430c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        * need to be (w*h*3)/2. It just needs to
8440c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        * be larger than certain minimum buffer size. However,
8450c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        * currently, this opaque foramt has been tested only on
8460c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        * YUV420 formats. If that is changed, then we need to revisit
8470c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        * this part in the future
8480c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        */
8490c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        case OMX_COLOR_FormatAndroidOpaque:
850888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            return (width * height * 3) / 2;
851888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
852888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        default:
853888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            CHECK(!"Should not be here. Unsupported color format.");
854888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            break;
855888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    }
856888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber}
857888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
858299766cdbf93fba14634e364b177b7f4d5811453James Dongstatus_t OMXCodec::findTargetColorFormat(
859299766cdbf93fba14634e364b177b7f4d5811453James Dong        const sp<MetaData>& meta, OMX_COLOR_FORMATTYPE *colorFormat) {
860299766cdbf93fba14634e364b177b7f4d5811453James Dong    LOGV("findTargetColorFormat");
861299766cdbf93fba14634e364b177b7f4d5811453James Dong    CHECK(mIsEncoder);
862299766cdbf93fba14634e364b177b7f4d5811453James Dong
863299766cdbf93fba14634e364b177b7f4d5811453James Dong    *colorFormat = OMX_COLOR_FormatYUV420SemiPlanar;
864299766cdbf93fba14634e364b177b7f4d5811453James Dong    int32_t targetColorFormat;
865299766cdbf93fba14634e364b177b7f4d5811453James Dong    if (meta->findInt32(kKeyColorFormat, &targetColorFormat)) {
866299766cdbf93fba14634e364b177b7f4d5811453James Dong        *colorFormat = (OMX_COLOR_FORMATTYPE) targetColorFormat;
867299766cdbf93fba14634e364b177b7f4d5811453James Dong    } else {
868299766cdbf93fba14634e364b177b7f4d5811453James Dong        if (!strcasecmp("OMX.TI.Video.encoder", mComponentName)) {
869299766cdbf93fba14634e364b177b7f4d5811453James Dong            *colorFormat = OMX_COLOR_FormatYCbYCr;
870299766cdbf93fba14634e364b177b7f4d5811453James Dong        }
871299766cdbf93fba14634e364b177b7f4d5811453James Dong    }
872299766cdbf93fba14634e364b177b7f4d5811453James Dong
8731374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket
874299766cdbf93fba14634e364b177b7f4d5811453James Dong    // Check whether the target color format is supported.
875299766cdbf93fba14634e364b177b7f4d5811453James Dong    return isColorFormatSupported(*colorFormat, kPortIndexInput);
876299766cdbf93fba14634e364b177b7f4d5811453James Dong}
877299766cdbf93fba14634e364b177b7f4d5811453James Dong
878299766cdbf93fba14634e364b177b7f4d5811453James Dongstatus_t OMXCodec::isColorFormatSupported(
879299766cdbf93fba14634e364b177b7f4d5811453James Dong        OMX_COLOR_FORMATTYPE colorFormat, int portIndex) {
880299766cdbf93fba14634e364b177b7f4d5811453James Dong    LOGV("isColorFormatSupported: %d", static_cast<int>(colorFormat));
881299766cdbf93fba14634e364b177b7f4d5811453James Dong
882299766cdbf93fba14634e364b177b7f4d5811453James Dong    // Enumerate all the color formats supported by
883299766cdbf93fba14634e364b177b7f4d5811453James Dong    // the omx component to see whether the given
884299766cdbf93fba14634e364b177b7f4d5811453James Dong    // color format is supported.
885299766cdbf93fba14634e364b177b7f4d5811453James Dong    OMX_VIDEO_PARAM_PORTFORMATTYPE portFormat;
886299766cdbf93fba14634e364b177b7f4d5811453James Dong    InitOMXParams(&portFormat);
887299766cdbf93fba14634e364b177b7f4d5811453James Dong    portFormat.nPortIndex = portIndex;
888299766cdbf93fba14634e364b177b7f4d5811453James Dong    OMX_U32 index = 0;
889299766cdbf93fba14634e364b177b7f4d5811453James Dong    portFormat.nIndex = index;
890299766cdbf93fba14634e364b177b7f4d5811453James Dong    while (true) {
891299766cdbf93fba14634e364b177b7f4d5811453James Dong        if (OMX_ErrorNone != mOMX->getParameter(
892299766cdbf93fba14634e364b177b7f4d5811453James Dong                mNode, OMX_IndexParamVideoPortFormat,
893299766cdbf93fba14634e364b177b7f4d5811453James Dong                &portFormat, sizeof(portFormat))) {
894a1abc1a76741914c7bc43f1df9e32744f023ab75James Dong            break;
895299766cdbf93fba14634e364b177b7f4d5811453James Dong        }
896299766cdbf93fba14634e364b177b7f4d5811453James Dong        // Make sure that omx component does not overwrite
897299766cdbf93fba14634e364b177b7f4d5811453James Dong        // the incremented index (bug 2897413).
898299766cdbf93fba14634e364b177b7f4d5811453James Dong        CHECK_EQ(index, portFormat.nIndex);
8990c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi        if (portFormat.eColorFormat == colorFormat) {
900299766cdbf93fba14634e364b177b7f4d5811453James Dong            LOGV("Found supported color format: %d", portFormat.eColorFormat);
901299766cdbf93fba14634e364b177b7f4d5811453James Dong            return OK;  // colorFormat is supported!
902299766cdbf93fba14634e364b177b7f4d5811453James Dong        }
903299766cdbf93fba14634e364b177b7f4d5811453James Dong        ++index;
904299766cdbf93fba14634e364b177b7f4d5811453James Dong        portFormat.nIndex = index;
905299766cdbf93fba14634e364b177b7f4d5811453James Dong
906299766cdbf93fba14634e364b177b7f4d5811453James Dong        // OMX Spec defines less than 50 color formats
907299766cdbf93fba14634e364b177b7f4d5811453James Dong        // 1000 is more than enough for us to tell whether the omx
908299766cdbf93fba14634e364b177b7f4d5811453James Dong        // component in question is buggy or not.
909299766cdbf93fba14634e364b177b7f4d5811453James Dong        if (index >= 1000) {
910299766cdbf93fba14634e364b177b7f4d5811453James Dong            LOGE("More than %ld color formats are supported???", index);
911299766cdbf93fba14634e364b177b7f4d5811453James Dong            break;
912299766cdbf93fba14634e364b177b7f4d5811453James Dong        }
913299766cdbf93fba14634e364b177b7f4d5811453James Dong    }
914a1abc1a76741914c7bc43f1df9e32744f023ab75James Dong
915a1abc1a76741914c7bc43f1df9e32744f023ab75James Dong    LOGE("color format %d is not supported", colorFormat);
916299766cdbf93fba14634e364b177b7f4d5811453James Dong    return UNKNOWN_ERROR;
917299766cdbf93fba14634e364b177b7f4d5811453James Dong}
918299766cdbf93fba14634e364b177b7f4d5811453James Dong
919693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::setVideoInputFormat(
920ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong        const char *mime, const sp<MetaData>& meta) {
921ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong
922ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    int32_t width, height, frameRate, bitRate, stride, sliceHeight;
923ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    bool success = meta->findInt32(kKeyWidth, &width);
924ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    success = success && meta->findInt32(kKeyHeight, &height);
925393410a441b6d06daf286ed496470e9d6b2b6ca8James Dong    success = success && meta->findInt32(kKeyFrameRate, &frameRate);
926ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    success = success && meta->findInt32(kKeyBitRate, &bitRate);
927ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    success = success && meta->findInt32(kKeyStride, &stride);
928ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    success = success && meta->findInt32(kKeySliceHeight, &sliceHeight);
929ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    CHECK(success);
930ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    CHECK(stride != 0);
931693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
932693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_VIDEO_CODINGTYPE compressionFormat = OMX_VIDEO_CodingUnused;
93318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mime)) {
934693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        compressionFormat = OMX_VIDEO_CodingAVC;
93518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_MPEG4, mime)) {
936693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        compressionFormat = OMX_VIDEO_CodingMPEG4;
93718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_H263, mime)) {
938693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        compressionFormat = OMX_VIDEO_CodingH263;
939693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
940693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        LOGE("Not a supported video mime type: %s", mime);
941693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        CHECK(!"Should not be here. Not a supported video mime type.");
942693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
943693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
944299766cdbf93fba14634e364b177b7f4d5811453James Dong    OMX_COLOR_FORMATTYPE colorFormat;
945f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((status_t)OK, findTargetColorFormat(meta, &colorFormat));
946693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
947c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    status_t err;
948693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
9497a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    OMX_VIDEO_PORTDEFINITIONTYPE *video_def = &def.format.video;
9507a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
951c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    //////////////////////// Input port /////////////////////////
952c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    CHECK_EQ(setVideoPortFormatType(
953c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong            kPortIndexInput, OMX_VIDEO_CodingUnused,
954f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            colorFormat), (status_t)OK);
955d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong
9567a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
957693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = kPortIndexInput;
958693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
959318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->getParameter(
960693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
961f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
962693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
963ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    def.nBufferSize = getFrameSize(colorFormat,
964ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong            stride > 0? stride: -stride, sliceHeight);
965693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
966f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainVideo);
967693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
968693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->nFrameWidth = width;
969693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->nFrameHeight = height;
970ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    video_def->nStride = stride;
971ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    video_def->nSliceHeight = sliceHeight;
972d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong    video_def->xFramerate = (frameRate << 16);  // Q16 format
973693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->eCompressionFormat = OMX_VIDEO_CodingUnused;
974693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->eColorFormat = colorFormat;
975693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
976318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->setParameter(
977693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
978f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
979888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
980c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    //////////////////////// Output port /////////////////////////
981c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    CHECK_EQ(setVideoPortFormatType(
982c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong            kPortIndexOutput, compressionFormat, OMX_COLOR_FormatUnused),
983f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            (status_t)OK);
984c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    InitOMXParams(&def);
985c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    def.nPortIndex = kPortIndexOutput;
986c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong
98703b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber    err = mOMX->getParameter(
98803b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
989c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong
990f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
991f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainVideo);
992c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong
993c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    video_def->nFrameWidth = width;
994c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    video_def->nFrameHeight = height;
995145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    video_def->xFramerate = 0;      // No need for output port
996d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong    video_def->nBitrate = bitRate;  // Q16 format
997c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    video_def->eCompressionFormat = compressionFormat;
998c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    video_def->eColorFormat = OMX_COLOR_FormatUnused;
999824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong    if (mQuirks & kRequiresLargerEncoderOutputBuffer) {
1000824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong        // Increases the output buffer size
1001824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong        def.nBufferSize = ((def.nBufferSize * 3) >> 1);
1002824c9ff4a55e86faae4f8f158977329909cbfaf6James Dong    }
100303b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber
100403b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber    err = mOMX->setParameter(
100503b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
1006f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
100703b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber
1008c32cd79d9ad4aba7d959b5b3be7361b4715e6f18James Dong    /////////////////// Codec-specific ////////////////////////
1009888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    switch (compressionFormat) {
1010888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        case OMX_VIDEO_CodingMPEG4:
1011888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        {
1012f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ(setupMPEG4EncoderParameters(meta), (status_t)OK);
1013888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            break;
1014888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        }
1015888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1016888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        case OMX_VIDEO_CodingH263:
1017f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ(setupH263EncoderParameters(meta), (status_t)OK);
1018888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            break;
1019888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
102030ab66297501757d745b9ae10da61adcd891f497Andreas Huber        case OMX_VIDEO_CodingAVC:
102130ab66297501757d745b9ae10da61adcd891f497Andreas Huber        {
1022f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ(setupAVCEncoderParameters(meta), (status_t)OK);
102330ab66297501757d745b9ae10da61adcd891f497Andreas Huber            break;
102430ab66297501757d745b9ae10da61adcd891f497Andreas Huber        }
102530ab66297501757d745b9ae10da61adcd891f497Andreas Huber
1026888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        default:
1027888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            CHECK(!"Support for this compressionFormat to be implemented.");
1028888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            break;
1029888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    }
1030888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber}
1031888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1032ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dongstatic OMX_U32 setPFramesSpacing(int32_t iFramesInterval, int32_t frameRate) {
1033ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    if (iFramesInterval < 0) {
1034ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong        return 0xFFFFFFFF;
1035ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    } else if (iFramesInterval == 0) {
1036ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong        return 0;
1037ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    }
1038ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    OMX_U32 ret = frameRate * iFramesInterval;
1039ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    CHECK(ret > 1);
1040ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    return ret;
1041ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong}
1042ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong
1043d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dongstatus_t OMXCodec::setupErrorCorrectionParameters() {
1044d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    OMX_VIDEO_PARAM_ERRORCORRECTIONTYPE errorCorrectionType;
1045d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    InitOMXParams(&errorCorrectionType);
1046d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    errorCorrectionType.nPortIndex = kPortIndexOutput;
1047d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1048d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    status_t err = mOMX->getParameter(
1049d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            mNode, OMX_IndexParamVideoErrorCorrection,
1050d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            &errorCorrectionType, sizeof(errorCorrectionType));
105159f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong    if (err != OK) {
105259f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong        LOGW("Error correction param query is not supported");
105359f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong        return OK;  // Optional feature. Ignore this failure
105459f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong    }
1055d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1056d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    errorCorrectionType.bEnableHEC = OMX_FALSE;
1057d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    errorCorrectionType.bEnableResync = OMX_TRUE;
1058d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    errorCorrectionType.nResynchMarkerSpacing = 256;
1059d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    errorCorrectionType.bEnableDataPartitioning = OMX_FALSE;
1060d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    errorCorrectionType.bEnableRVLC = OMX_FALSE;
1061d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1062d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    err = mOMX->setParameter(
1063d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            mNode, OMX_IndexParamVideoErrorCorrection,
1064d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            &errorCorrectionType, sizeof(errorCorrectionType));
106559f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong    if (err != OK) {
106659f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong        LOGW("Error correction param configuration is not supported");
106759f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong    }
106859f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong
106959f63db5d1cdc91336a2b82e195e0000091cd6e3James Dong    // Optional feature. Ignore the failure.
1070d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    return OK;
1071d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong}
1072d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1073d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dongstatus_t OMXCodec::setupBitRate(int32_t bitRate) {
1074d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    OMX_VIDEO_PARAM_BITRATETYPE bitrateType;
1075d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    InitOMXParams(&bitrateType);
1076d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    bitrateType.nPortIndex = kPortIndexOutput;
1077d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1078d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    status_t err = mOMX->getParameter(
1079d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            mNode, OMX_IndexParamVideoBitrate,
1080d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            &bitrateType, sizeof(bitrateType));
1081f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1082d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1083d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    bitrateType.eControlRate = OMX_Video_ControlRateVariable;
1084d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    bitrateType.nTargetBitrate = bitRate;
1085d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1086d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    err = mOMX->setParameter(
1087d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            mNode, OMX_IndexParamVideoBitrate,
1088d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            &bitrateType, sizeof(bitrateType));
1089f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1090d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    return OK;
1091d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong}
1092d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1093145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dongstatus_t OMXCodec::getVideoProfileLevel(
1094145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        const sp<MetaData>& meta,
1095145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        const CodecProfileLevel& defaultProfileLevel,
1096145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        CodecProfileLevel &profileLevel) {
1097145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    CODEC_LOGV("Default profile: %ld, level %ld",
1098145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong            defaultProfileLevel.mProfile, defaultProfileLevel.mLevel);
1099145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1100145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    // Are the default profile and level overwriten?
1101145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    int32_t profile, level;
1102145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    if (!meta->findInt32(kKeyVideoProfile, &profile)) {
1103145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        profile = defaultProfileLevel.mProfile;
1104145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    }
1105145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    if (!meta->findInt32(kKeyVideoLevel, &level)) {
1106145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        level = defaultProfileLevel.mLevel;
1107145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    }
1108145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    CODEC_LOGV("Target profile: %d, level: %d", profile, level);
1109145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1110145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    // Are the target profile and level supported by the encoder?
1111145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    OMX_VIDEO_PARAM_PROFILELEVELTYPE param;
1112145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    InitOMXParams(&param);
1113145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    param.nPortIndex = kPortIndexOutput;
1114145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    for (param.nProfileIndex = 0;; ++param.nProfileIndex) {
1115145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        status_t err = mOMX->getParameter(
1116145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong                mNode, OMX_IndexParamVideoProfileLevelQuerySupported,
1117145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong                &param, sizeof(param));
1118145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1119f01691f44dcbe38a0ab3914e7c709ddc009d95dcJames Dong        if (err != OK) break;
1120145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1121145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        int32_t supportedProfile = static_cast<int32_t>(param.eProfile);
1122145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        int32_t supportedLevel = static_cast<int32_t>(param.eLevel);
1123b914122eb9cb54bbeae4ec03bfebb194aecdccbdJames Dong        CODEC_LOGV("Supported profile: %d, level %d",
1124145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong            supportedProfile, supportedLevel);
1125145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1126145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        if (profile == supportedProfile &&
1127f01691f44dcbe38a0ab3914e7c709ddc009d95dcJames Dong            level <= supportedLevel) {
1128f01691f44dcbe38a0ab3914e7c709ddc009d95dcJames Dong            // We can further check whether the level is a valid
1129f01691f44dcbe38a0ab3914e7c709ddc009d95dcJames Dong            // value; but we will leave that to the omx encoder component
1130f01691f44dcbe38a0ab3914e7c709ddc009d95dcJames Dong            // via OMX_SetParameter call.
1131145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong            profileLevel.mProfile = profile;
1132145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong            profileLevel.mLevel = level;
1133145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong            return OK;
1134145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        }
1135145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    }
1136145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1137145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    CODEC_LOGE("Target profile (%d) and level (%d) is not supported",
1138145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong            profile, level);
1139145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    return BAD_VALUE;
1140145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong}
1141145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1142d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dongstatus_t OMXCodec::setupH263EncoderParameters(const sp<MetaData>& meta) {
1143d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    int32_t iFramesInterval, frameRate, bitRate;
1144d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    bool success = meta->findInt32(kKeyBitRate, &bitRate);
1145393410a441b6d06daf286ed496470e9d6b2b6ca8James Dong    success = success && meta->findInt32(kKeyFrameRate, &frameRate);
1146d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    success = success && meta->findInt32(kKeyIFramesInterval, &iFramesInterval);
1147d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    CHECK(success);
1148d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    OMX_VIDEO_PARAM_H263TYPE h263type;
1149d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    InitOMXParams(&h263type);
1150d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.nPortIndex = kPortIndexOutput;
1151d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1152d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    status_t err = mOMX->getParameter(
1153d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            mNode, OMX_IndexParamVideoH263, &h263type, sizeof(h263type));
1154f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1155d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1156d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.nAllowedPictureTypes =
1157d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong        OMX_VIDEO_PictureTypeI | OMX_VIDEO_PictureTypeP;
1158d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1159d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.nPFrames = setPFramesSpacing(iFramesInterval, frameRate);
1160d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    if (h263type.nPFrames == 0) {
1161d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong        h263type.nAllowedPictureTypes = OMX_VIDEO_PictureTypeI;
1162d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    }
1163d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.nBFrames = 0;
1164d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1165145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    // Check profile and level parameters
1166145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    CodecProfileLevel defaultProfileLevel, profileLevel;
116797e0fcc0cb69692a6c518e2bfaf9892a164e7f58James Dong    defaultProfileLevel.mProfile = h263type.eProfile;
116897e0fcc0cb69692a6c518e2bfaf9892a164e7f58James Dong    defaultProfileLevel.mLevel = h263type.eLevel;
1169145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    err = getVideoProfileLevel(meta, defaultProfileLevel, profileLevel);
1170145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    if (err != OK) return err;
1171145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    h263type.eProfile = static_cast<OMX_VIDEO_H263PROFILETYPE>(profileLevel.mProfile);
1172145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    h263type.eLevel = static_cast<OMX_VIDEO_H263LEVELTYPE>(profileLevel.mLevel);
1173d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1174d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.bPLUSPTYPEAllowed = OMX_FALSE;
1175d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.bForceRoundingTypeToZero = OMX_FALSE;
1176d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.nPictureHeaderRepetition = 0;
1177d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    h263type.nGOBHeaderInterval = 0;
1178d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1179d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    err = mOMX->setParameter(
1180d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong            mNode, OMX_IndexParamVideoH263, &h263type, sizeof(h263type));
1181f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1182d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1183f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(setupBitRate(bitRate), (status_t)OK);
1184f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(setupErrorCorrectionParameters(), (status_t)OK);
1185d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1186d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong    return OK;
1187d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong}
1188d329e21495eda9dbc531fdd0c26c77f1593ac3f4James Dong
1189ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dongstatus_t OMXCodec::setupMPEG4EncoderParameters(const sp<MetaData>& meta) {
1190ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    int32_t iFramesInterval, frameRate, bitRate;
1191ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    bool success = meta->findInt32(kKeyBitRate, &bitRate);
1192393410a441b6d06daf286ed496470e9d6b2b6ca8James Dong    success = success && meta->findInt32(kKeyFrameRate, &frameRate);
1193ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    success = success && meta->findInt32(kKeyIFramesInterval, &iFramesInterval);
1194ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    CHECK(success);
1195888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    OMX_VIDEO_PARAM_MPEG4TYPE mpeg4type;
1196888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    InitOMXParams(&mpeg4type);
1197888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nPortIndex = kPortIndexOutput;
1198888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1199888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    status_t err = mOMX->getParameter(
1200888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            mNode, OMX_IndexParamVideoMpeg4, &mpeg4type, sizeof(mpeg4type));
1201f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1202888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1203888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nSliceHeaderSpacing = 0;
1204888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.bSVH = OMX_FALSE;
1205888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.bGov = OMX_FALSE;
1206888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1207888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nAllowedPictureTypes =
1208888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber        OMX_VIDEO_PictureTypeI | OMX_VIDEO_PictureTypeP;
1209888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1210ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    mpeg4type.nPFrames = setPFramesSpacing(iFramesInterval, frameRate);
1211ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    if (mpeg4type.nPFrames == 0) {
1212ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong        mpeg4type.nAllowedPictureTypes = OMX_VIDEO_PictureTypeI;
1213ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    }
1214888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nBFrames = 0;
1215888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nIDCVLCThreshold = 0;
1216888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.bACPred = OMX_TRUE;
1217888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nMaxPacketSize = 256;
1218888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nTimeIncRes = 1000;
1219888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.nHeaderExtension = 0;
1220888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    mpeg4type.bReversibleVLC = OMX_FALSE;
1221888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1222145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    // Check profile and level parameters
1223145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    CodecProfileLevel defaultProfileLevel, profileLevel;
122497e0fcc0cb69692a6c518e2bfaf9892a164e7f58James Dong    defaultProfileLevel.mProfile = mpeg4type.eProfile;
122597e0fcc0cb69692a6c518e2bfaf9892a164e7f58James Dong    defaultProfileLevel.mLevel = mpeg4type.eLevel;
1226145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    err = getVideoProfileLevel(meta, defaultProfileLevel, profileLevel);
1227145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    if (err != OK) return err;
1228145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    mpeg4type.eProfile = static_cast<OMX_VIDEO_MPEG4PROFILETYPE>(profileLevel.mProfile);
1229145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    mpeg4type.eLevel = static_cast<OMX_VIDEO_MPEG4LEVELTYPE>(profileLevel.mLevel);
1230888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1231888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    err = mOMX->setParameter(
1232888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber            mNode, OMX_IndexParamVideoMpeg4, &mpeg4type, sizeof(mpeg4type));
1233f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1234888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1235f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(setupBitRate(bitRate), (status_t)OK);
1236f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(setupErrorCorrectionParameters(), (status_t)OK);
1237888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber
1238888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    return OK;
1239693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1240693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1241ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dongstatus_t OMXCodec::setupAVCEncoderParameters(const sp<MetaData>& meta) {
1242ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    int32_t iFramesInterval, frameRate, bitRate;
1243ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    bool success = meta->findInt32(kKeyBitRate, &bitRate);
1244393410a441b6d06daf286ed496470e9d6b2b6ca8James Dong    success = success && meta->findInt32(kKeyFrameRate, &frameRate);
1245ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    success = success && meta->findInt32(kKeyIFramesInterval, &iFramesInterval);
1246ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong    CHECK(success);
1247ddcc4a66d848deef6fb4689e64e30cd9bd2684feJames Dong
124830ab66297501757d745b9ae10da61adcd891f497Andreas Huber    OMX_VIDEO_PARAM_AVCTYPE h264type;
124930ab66297501757d745b9ae10da61adcd891f497Andreas Huber    InitOMXParams(&h264type);
125030ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.nPortIndex = kPortIndexOutput;
125130ab66297501757d745b9ae10da61adcd891f497Andreas Huber
125230ab66297501757d745b9ae10da61adcd891f497Andreas Huber    status_t err = mOMX->getParameter(
125330ab66297501757d745b9ae10da61adcd891f497Andreas Huber            mNode, OMX_IndexParamVideoAvc, &h264type, sizeof(h264type));
1254f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
125530ab66297501757d745b9ae10da61adcd891f497Andreas Huber
125630ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.nAllowedPictureTypes =
125730ab66297501757d745b9ae10da61adcd891f497Andreas Huber        OMX_VIDEO_PictureTypeI | OMX_VIDEO_PictureTypeP;
125830ab66297501757d745b9ae10da61adcd891f497Andreas Huber
1259145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    // Check profile and level parameters
1260145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    CodecProfileLevel defaultProfileLevel, profileLevel;
1261145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    defaultProfileLevel.mProfile = h264type.eProfile;
1262145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    defaultProfileLevel.mLevel = h264type.eLevel;
1263145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    err = getVideoProfileLevel(meta, defaultProfileLevel, profileLevel);
1264145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    if (err != OK) return err;
1265145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    h264type.eProfile = static_cast<OMX_VIDEO_AVCPROFILETYPE>(profileLevel.mProfile);
1266145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    h264type.eLevel = static_cast<OMX_VIDEO_AVCLEVELTYPE>(profileLevel.mLevel);
1267145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
12681374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    // FIXME:
12691374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    // Remove the workaround after the work in done.
12701374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    if (!strncmp(mComponentName, "OMX.TI.DUCATI1", 14)) {
12711374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket        h264type.eProfile = OMX_VIDEO_AVCProfileBaseline;
12721374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket    }
12731374eddc4455b26d1dffdca10fc70534b3f08c1dDandawate Saket
1274145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    if (h264type.eProfile == OMX_VIDEO_AVCProfileBaseline) {
1275d552b88515c6ccd18695e5db5e6032a6425d8c63James Dong        h264type.nSliceHeaderSpacing = 0;
1276145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.bUseHadamard = OMX_TRUE;
1277145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.nRefFrames = 1;
1278d552b88515c6ccd18695e5db5e6032a6425d8c63James Dong        h264type.nBFrames = 0;
1279d552b88515c6ccd18695e5db5e6032a6425d8c63James Dong        h264type.nPFrames = setPFramesSpacing(iFramesInterval, frameRate);
1280d552b88515c6ccd18695e5db5e6032a6425d8c63James Dong        if (h264type.nPFrames == 0) {
1281d552b88515c6ccd18695e5db5e6032a6425d8c63James Dong            h264type.nAllowedPictureTypes = OMX_VIDEO_PictureTypeI;
1282d552b88515c6ccd18695e5db5e6032a6425d8c63James Dong        }
1283145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.nRefIdx10ActiveMinus1 = 0;
1284145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.nRefIdx11ActiveMinus1 = 0;
1285145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.bEntropyCodingCABAC = OMX_FALSE;
1286145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.bWeightedPPrediction = OMX_FALSE;
1287145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.bconstIpred = OMX_FALSE;
1288145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.bDirect8x8Inference = OMX_FALSE;
1289145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.bDirectSpatialTemporal = OMX_FALSE;
1290145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.nCabacInitIdc = 0;
1291145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    }
1292145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
1293145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    if (h264type.nBFrames != 0) {
1294145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong        h264type.nAllowedPictureTypes |= OMX_VIDEO_PictureTypeB;
1295145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong    }
1296145bfe5eb3e08c9689c28f6bf3287a979438b04bJames Dong
129730ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.bEnableUEP = OMX_FALSE;
129830ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.bEnableFMO = OMX_FALSE;
129930ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.bEnableASO = OMX_FALSE;
130030ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.bEnableRS = OMX_FALSE;
130130ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.bFrameMBsOnly = OMX_TRUE;
130230ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.bMBAFF = OMX_FALSE;
130330ab66297501757d745b9ae10da61adcd891f497Andreas Huber    h264type.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterEnable;
130430ab66297501757d745b9ae10da61adcd891f497Andreas Huber
13054937be23b485ce8edf180ff5e71ebf0884178f17pgudadhe    if (!strcasecmp("OMX.Nvidia.h264.encoder", mComponentName)) {
13064937be23b485ce8edf180ff5e71ebf0884178f17pgudadhe        h264type.eLevel = OMX_VIDEO_AVCLevelMax;
13074937be23b485ce8edf180ff5e71ebf0884178f17pgudadhe    }
13084937be23b485ce8edf180ff5e71ebf0884178f17pgudadhe
130930ab66297501757d745b9ae10da61adcd891f497Andreas Huber    err = mOMX->setParameter(
131030ab66297501757d745b9ae10da61adcd891f497Andreas Huber            mNode, OMX_IndexParamVideoAvc, &h264type, sizeof(h264type));
1311f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
131230ab66297501757d745b9ae10da61adcd891f497Andreas Huber
1313f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(setupBitRate(bitRate), (status_t)OK);
131430ab66297501757d745b9ae10da61adcd891f497Andreas Huber
131530ab66297501757d745b9ae10da61adcd891f497Andreas Huber    return OK;
131630ab66297501757d745b9ae10da61adcd891f497Andreas Huber}
131730ab66297501757d745b9ae10da61adcd891f497Andreas Huber
13182a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huberstatus_t OMXCodec::setVideoOutputFormat(
1319693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        const char *mime, OMX_U32 width, OMX_U32 height) {
13202a4a7d5af053a17586a262a1267ba993e31790f1Andreas Huber    CODEC_LOGV("setVideoOutputFormat width=%ld, height=%ld", width, height);
1321693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1322693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_VIDEO_CODINGTYPE compressionFormat = OMX_VIDEO_CodingUnused;
132318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mime)) {
1324693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        compressionFormat = OMX_VIDEO_CodingAVC;
132518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_MPEG4, mime)) {
1326693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        compressionFormat = OMX_VIDEO_CodingMPEG4;
132718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_H263, mime)) {
1328693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        compressionFormat = OMX_VIDEO_CodingH263;
1329bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_VPX, mime)) {
1330bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        compressionFormat = OMX_VIDEO_CodingVPX;
1331386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_MPEG2, mime)) {
1332386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber        compressionFormat = OMX_VIDEO_CodingMPEG2;
1333693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
1334693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        LOGE("Not a supported video mime type: %s", mime);
1335693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        CHECK(!"Should not be here. Not a supported video mime type.");
1336693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1337693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
13382a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    status_t err = setVideoPortFormatType(
1339693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            kPortIndexInput, compressionFormat, OMX_COLOR_FormatUnused);
1340693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
13412a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    if (err != OK) {
13422a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        return err;
13432a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    }
13442a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber
1345693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#if 1
1346693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    {
1347693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        OMX_VIDEO_PARAM_PORTFORMATTYPE format;
13487a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        InitOMXParams(&format);
1349693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        format.nPortIndex = kPortIndexOutput;
1350693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        format.nIndex = 0;
1351693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1352318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        status_t err = mOMX->getParameter(
1353693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mNode, OMX_IndexParamVideoPortFormat,
1354693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                &format, sizeof(format));
1355f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(err, (status_t)OK);
1356f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ((int)format.eCompressionFormat, (int)OMX_VIDEO_CodingUnused);
1357693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1358693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        CHECK(format.eColorFormat == OMX_COLOR_FormatYUV420Planar
1359693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber               || format.eColorFormat == OMX_COLOR_FormatYUV420SemiPlanar
1360693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber               || format.eColorFormat == OMX_COLOR_FormatCbYCrY
1361cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan               || format.eColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar
1362693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber               || format.eColorFormat == OMX_QCOM_COLOR_FormatYVU420SemiPlanar);
1363693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1364318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        err = mOMX->setParameter(
1365693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mNode, OMX_IndexParamVideoPortFormat,
1366693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                &format, sizeof(format));
13672a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber
13682a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        if (err != OK) {
13692a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            return err;
13702a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        }
1371693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1372693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#endif
1373693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1374693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
13757a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
1376693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = kPortIndexInput;
1377693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
13787a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    OMX_VIDEO_PORTDEFINITIONTYPE *video_def = &def.format.video;
13797a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
13802a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    err = mOMX->getParameter(
1381693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
1382693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1383f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1384693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1385693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#if 1
1386693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    // XXX Need a (much) better heuristic to compute input buffer sizes.
1387693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    const size_t X = 64 * 1024;
1388693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (def.nBufferSize < X) {
1389693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        def.nBufferSize = X;
1390693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1391693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#endif
1392693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1393f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainVideo);
1394693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1395693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->nFrameWidth = width;
1396693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->nFrameHeight = height;
1397693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1398888f72a4628eed3c136ff2e346a15d129da0d520Andreas Huber    video_def->eCompressionFormat = compressionFormat;
1399693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->eColorFormat = OMX_COLOR_FormatUnused;
1400693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1401318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->setParameter(
1402693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
14032a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber
14042a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    if (err != OK) {
14052a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        return err;
14062a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    }
1407693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1408693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    ////////////////////////////////////////////////////////////////////////////
1409693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
14107a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
1411693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = kPortIndexOutput;
1412693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1413318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->getParameter(
1414693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
1415f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1416f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainVideo);
1417693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1418693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#if 0
1419693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nBufferSize =
1420693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        (((width + 15) & -16) * ((height + 15) & -16) * 3) / 2;  // YUV420
1421693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#endif
1422693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1423693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->nFrameWidth = width;
1424693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    video_def->nFrameHeight = height;
1425693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1426318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->setParameter(
1427693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
14282a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber
14292a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    return err;
1430693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1431693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1432693d271e62a3726689ff68f4505ba49228eb94b2Andreas HuberOMXCodec::OMXCodec(
14330bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        const sp<IOMX> &omx, IOMX::node_id node,
14340bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        uint32_t quirks, uint32_t flags,
1435bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber        bool isEncoder,
1436693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        const char *mime,
1437693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        const char *componentName,
14386a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        const sp<MediaSource> &source,
14396a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        const sp<ANativeWindow> &nativeWindow)
1440693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    : mOMX(omx),
14417eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber      mOMXLivesLocally(omx->livesLocally(getpid())),
1442693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mNode(node),
1443693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mQuirks(quirks),
14440bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber      mFlags(flags),
1445693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mIsEncoder(isEncoder),
1446693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mMIME(strdup(mime)),
1447693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mComponentName(strdup(componentName)),
1448693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mSource(source),
1449693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mCodecSpecificDataIndex(0),
1450693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mState(LOADED),
1451284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber      mInitialBufferSubmit(true),
1452693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mSignalledEOS(false),
1453693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber      mNoMoreOutputData(false),
14547f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber      mOutputPortSettingsHaveChanged(false),
14557f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber      mSeekTimeUs(-1),
1456abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber      mSeekMode(ReadOptions::SEEK_CLOSEST_SYNC),
1457abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber      mTargetTimeUs(-1),
14589c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber      mOutputPortSettingsChangedPending(false),
1459d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber      mLeftOverBuffer(NULL),
14606a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      mPaused(false),
1461bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber      mNativeWindow(!strncmp(componentName, "OMX.google.", 11)
1462bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                        ? NULL : nativeWindow) {
1463693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mPortStatus[kPortIndexInput] = ENABLED;
1464693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mPortStatus[kPortIndexOutput] = ENABLED;
1465693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
14667a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    setComponentRole();
14677a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber}
14687a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
146918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber// static
147018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Hubervoid OMXCodec::setComponentRole(
147118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        const sp<IOMX> &omx, IOMX::node_id node, bool isEncoder,
147218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        const char *mime) {
14737a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    struct MimeToRole {
14747a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        const char *mime;
14757a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        const char *decoderRole;
14767a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        const char *encoderRole;
14777a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    };
14787a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
14797a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    static const MimeToRole kMimeToRole[] = {
148018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        { MEDIA_MIMETYPE_AUDIO_MPEG,
148118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            "audio_decoder.mp3", "audio_encoder.mp3" },
148218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_NB,
148318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            "audio_decoder.amrnb", "audio_encoder.amrnb" },
148418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_WB,
148518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            "audio_decoder.amrwb", "audio_encoder.amrwb" },
148618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AAC,
148718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            "audio_decoder.aac", "audio_encoder.aac" },
148818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        { MEDIA_MIMETYPE_VIDEO_AVC,
148918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            "video_decoder.avc", "video_encoder.avc" },
149018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        { MEDIA_MIMETYPE_VIDEO_MPEG4,
149118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            "video_decoder.mpeg4", "video_encoder.mpeg4" },
149218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        { MEDIA_MIMETYPE_VIDEO_H263,
149318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            "video_decoder.h263", "video_encoder.h263" },
14947a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    };
14957a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
14967a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    static const size_t kNumMimeToRole =
14977a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        sizeof(kMimeToRole) / sizeof(kMimeToRole[0]);
14987a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
14997a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    size_t i;
15007a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    for (i = 0; i < kNumMimeToRole; ++i) {
150118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        if (!strcasecmp(mime, kMimeToRole[i].mime)) {
15027a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            break;
15037a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        }
15047a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    }
15057a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
15067a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    if (i == kNumMimeToRole) {
15077a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        return;
15087a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    }
15097a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
15107a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    const char *role =
151118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        isEncoder ? kMimeToRole[i].encoderRole
151218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                  : kMimeToRole[i].decoderRole;
15137a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
15147a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    if (role != NULL) {
15157a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        OMX_PARAM_COMPONENTROLETYPE roleParams;
15167a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        InitOMXParams(&roleParams);
15177a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
15187a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        strncpy((char *)roleParams.cRole,
15197a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                role, OMX_MAX_STRINGNAME_SIZE - 1);
15207a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
15217a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        roleParams.cRole[OMX_MAX_STRINGNAME_SIZE - 1] = '\0';
15227a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
1523318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        status_t err = omx->setParameter(
152418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                node, OMX_IndexParamStandardComponentRole,
15257a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                &roleParams, sizeof(roleParams));
15267a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber
15277a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        if (err != OK) {
15287a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            LOGW("Failed to set standard component role '%s'.", role);
15297a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        }
15307a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    }
1531693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1532693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
153318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Hubervoid OMXCodec::setComponentRole() {
153418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    setComponentRole(mOMX, mNode, mIsEncoder, mMIME);
153518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber}
153618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
1537693d271e62a3726689ff68f4505ba49228eb94b2Andreas HuberOMXCodec::~OMXCodec() {
153800998fbb52dfa78ac3a4b3706d64fc612926bfbcAndreas Huber    mSource.clear();
153900998fbb52dfa78ac3a4b3706d64fc612926bfbcAndreas Huber
15404d785ae6b940c62610612baf9d24f1ee607a2714Andreas Huber    CHECK(mState == LOADED || mState == ERROR || mState == LOADED_TO_IDLE);
1541693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1542318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->freeNode(mNode);
1543f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
1544693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1545693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mNode = NULL;
1546693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    setState(DEAD);
1547693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1548693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    clearCodecSpecificData();
1549693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1550693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    free(mComponentName);
1551693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mComponentName = NULL;
1552bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
1553693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    free(mMIME);
1554693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mMIME = NULL;
1555693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1556693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1557693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatus_t OMXCodec::init() {
1558284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber    // mLock is held.
1559693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1560f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)mState, (int)LOADED);
1561693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1562693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err;
1563693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (!(mQuirks & kRequiresLoadedToIdleAfterAllocation)) {
1564318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        err = mOMX->sendCommand(mNode, OMX_CommandStateSet, OMX_StateIdle);
1565f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(err, (status_t)OK);
1566693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        setState(LOADED_TO_IDLE);
1567693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1568693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1569693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    err = allocateBuffers();
157060693ddde0e435def1617848f545c78cec452766Jamie Gennis    if (err != (status_t)OK) {
157160693ddde0e435def1617848f545c78cec452766Jamie Gennis        return err;
157260693ddde0e435def1617848f545c78cec452766Jamie Gennis    }
1573693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1574693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (mQuirks & kRequiresLoadedToIdleAfterAllocation) {
1575318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        err = mOMX->sendCommand(mNode, OMX_CommandStateSet, OMX_StateIdle);
1576f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(err, (status_t)OK);
1577693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1578693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        setState(LOADED_TO_IDLE);
1579693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1580693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1581693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    while (mState != EXECUTING && mState != ERROR) {
1582693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        mAsyncCompletion.wait(mLock);
1583693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1584693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1585693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return mState == ERROR ? UNKNOWN_ERROR : OK;
1586693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1587693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1588693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber// static
1589693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberbool OMXCodec::isIntermediateState(State state) {
1590693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return state == LOADED_TO_IDLE
1591693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        || state == IDLE_TO_EXECUTING
1592693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        || state == EXECUTING_TO_IDLE
1593693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        || state == IDLE_TO_LOADED
1594693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        || state == RECONFIGURING;
1595693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1596693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1597693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatus_t OMXCodec::allocateBuffers() {
1598693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err = allocateBuffersOnPort(kPortIndexInput);
1599693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1600693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (err != OK) {
1601693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return err;
1602693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1603693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1604693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return allocateBuffersOnPort(kPortIndexOutput);
1605693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1606693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1607693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatus_t OMXCodec::allocateBuffersOnPort(OMX_U32 portIndex) {
1608bf2ba2a97927c24d14c0e71158abe7b49c557c68Jamie Gennis    if (mNativeWindow != NULL && portIndex == kPortIndexOutput) {
16096a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        return allocateOutputBuffersFromNativeWindow();
16106a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
16116a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
16120bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if ((mFlags & kEnableGrallocUsageProtected) && portIndex == kPortIndexOutput) {
1613d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        LOGE("protected output buffers must be stent to an ANativeWindow");
1614d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        return PERMISSION_DENIED;
1615d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis    }
1616d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis
1617f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong    status_t err = OK;
16180bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if ((mFlags & kStoreMetaDataInVideoBuffers)
16190bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            && portIndex == kPortIndexInput) {
1620f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong        err = mOMX->storeMetaDataInBuffers(mNode, kPortIndexInput, OMX_TRUE);
1621f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong        if (err != OK) {
1622f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong            LOGE("Storing meta data in video buffers is not supported");
1623f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong            return err;
1624f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong        }
1625f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong    }
1626f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong
1627693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
16287a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
1629693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = portIndex;
1630693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1631f6e1ff257eb3c4f98587170d0c6367bf58cfb115James Dong    err = mOMX->getParameter(
1632693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
1633693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1634693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (err != OK) {
1635693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return err;
1636693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1637693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
163839ddf8e0f18766f7ba1e3246b774aa6ebd93eea8Andreas Huber    CODEC_LOGI("allocating %lu buffers of size %lu on %s port",
163939ddf8e0f18766f7ba1e3246b774aa6ebd93eea8Andreas Huber            def.nBufferCountActual, def.nBufferSize,
164039ddf8e0f18766f7ba1e3246b774aa6ebd93eea8Andreas Huber            portIndex == kPortIndexInput ? "input" : "output");
164139ddf8e0f18766f7ba1e3246b774aa6ebd93eea8Andreas Huber
16427530e9c708275c273c134c36c68179f511c1940eAndreas Huber    size_t totalSize = def.nBufferCountActual * def.nBufferSize;
1643867d2f6ce668968e463eb86b856d21525f12fd67Mathias Agopian    mDealer[portIndex] = new MemoryDealer(totalSize, "OMXCodec");
16447530e9c708275c273c134c36c68179f511c1940eAndreas Huber
1645693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (OMX_U32 i = 0; i < def.nBufferCountActual; ++i) {
16467530e9c708275c273c134c36c68179f511c1940eAndreas Huber        sp<IMemory> mem = mDealer[portIndex]->allocate(def.nBufferSize);
1647693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        CHECK(mem.get() != NULL);
1648693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1649570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        BufferInfo info;
1650570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        info.mData = NULL;
1651570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        info.mSize = def.nBufferSize;
1652570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber
1653693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        IOMX::buffer_id buffer;
1654693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (portIndex == kPortIndexInput
16550bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                && ((mQuirks & kRequiresAllocateBufferOnInputPorts)
16560bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    || (mFlags & kUseSecureInputBuffers))) {
16577eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber            if (mOMXLivesLocally) {
1658570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber                mem.clear();
1659570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber
16607eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber                err = mOMX->allocateBuffer(
1661570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber                        mNode, portIndex, def.nBufferSize, &buffer,
1662570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber                        &info.mData);
16637eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber            } else {
16647eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber                err = mOMX->allocateBufferWithBackup(
16657eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber                        mNode, portIndex, mem, &buffer);
16667eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber            }
1667ea7d15629752256f6ea1d5c6ea796e59aefd030fAndreas Huber        } else if (portIndex == kPortIndexOutput
1668ea7d15629752256f6ea1d5c6ea796e59aefd030fAndreas Huber                && (mQuirks & kRequiresAllocateBufferOnOutputPorts)) {
16697eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber            if (mOMXLivesLocally) {
1670570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber                mem.clear();
1671570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber
16727eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber                err = mOMX->allocateBuffer(
1673570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber                        mNode, portIndex, def.nBufferSize, &buffer,
1674570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber                        &info.mData);
16757eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber            } else {
16767eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber                err = mOMX->allocateBufferWithBackup(
16777eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber                        mNode, portIndex, mem, &buffer);
16787eaa9c9385535b651064e02d05a8ffa4b2359281Andreas Huber            }
1679693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        } else {
1680318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            err = mOMX->useBuffer(mNode, portIndex, mem, &buffer);
1681693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
1682693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1683693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (err != OK) {
1684693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            LOGE("allocate_buffer_with_backup failed");
1685693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            return err;
1686693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
1687693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1688570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        if (mem != NULL) {
1689570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            info.mData = mem->pointer();
1690570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        }
1691570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber
1692693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        info.mBuffer = buffer;
169392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        info.mStatus = OWNED_BY_US;
1694693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        info.mMem = mem;
1695693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        info.mMediaBuffer = NULL;
1696693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1697693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (portIndex == kPortIndexOutput) {
1698213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber            if (!(mOMXLivesLocally
1699213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                        && (mQuirks & kRequiresAllocateBufferOnOutputPorts)
1700213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                        && (mQuirks & kDefersOutputBufferAllocation))) {
1701213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                // If the node does not fill in the buffer ptr at this time,
1702213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                // we will defer creating the MediaBuffer until receiving
1703213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                // the first FILL_BUFFER_DONE notification instead.
1704213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                info.mMediaBuffer = new MediaBuffer(info.mData, info.mSize);
1705213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                info.mMediaBuffer->setObserver(this);
1706213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber            }
1707693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
1708693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1709693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        mPortBuffers[portIndex].push(info);
1710693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
17117a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        CODEC_LOGV("allocated buffer %p on %s port", buffer,
1712693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber             portIndex == kPortIndexInput ? "input" : "output");
1713693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
1714693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
1715134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    // dumpPortStatus(portIndex);
1716693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
17170bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if (portIndex == kPortIndexInput && (mFlags & kUseSecureInputBuffers)) {
17180bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        Vector<MediaBuffer *> buffers;
17190bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        for (size_t i = 0; i < def.nBufferCountActual; ++i) {
17200bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            const BufferInfo &info = mPortBuffers[kPortIndexInput].itemAt(i);
17210bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
17220bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            MediaBuffer *mbuf = new MediaBuffer(info.mData, info.mSize);
17230bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            buffers.push(mbuf);
17240bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        }
17250bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
17260bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        status_t err = mSource->setBuffers(buffers);
17270bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
17280bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        if (err != OK) {
17290bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            for (size_t i = 0; i < def.nBufferCountActual; ++i) {
17300bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                buffers.editItemAt(i)->release();
17310bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            }
17320bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            buffers.clear();
17330bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
17340bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            CODEC_LOGE(
17350bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    "Codec requested to use secure input buffers but "
17360bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    "upstream source didn't support that.");
17370bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
17380bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            return err;
17390bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        }
17400bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    }
17410bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
1742693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return OK;
1743693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
1744693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
17457fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huberstatus_t OMXCodec::applyRotation() {
17467fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    sp<MetaData> meta = mSource->getFormat();
17477fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber
17487fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    int32_t rotationDegrees;
17497fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    if (!meta->findInt32(kKeyRotation, &rotationDegrees)) {
17507fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        rotationDegrees = 0;
17517fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    }
17527fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber
17537fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    uint32_t transform;
17547fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    switch (rotationDegrees) {
17557fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        case 0: transform = 0; break;
17567fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        case 90: transform = HAL_TRANSFORM_ROT_90; break;
17577fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        case 180: transform = HAL_TRANSFORM_ROT_180; break;
17587fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        case 270: transform = HAL_TRANSFORM_ROT_270; break;
17597fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        default: transform = 0; break;
17607fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    }
17617fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber
17627fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    status_t err = OK;
17637fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber
17647fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    if (transform) {
17657fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        err = native_window_set_buffers_transform(
17667fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber                mNativeWindow.get(), transform);
17677fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    }
17687fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber
17697fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    return err;
17707fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber}
17717fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber
17726a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennisstatus_t OMXCodec::allocateOutputBuffersFromNativeWindow() {
17736a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // Get the number of buffers needed.
17746a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    OMX_PARAM_PORTDEFINITIONTYPE def;
17756a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    InitOMXParams(&def);
17766a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    def.nPortIndex = kPortIndexOutput;
17776a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
17786a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    status_t err = mOMX->getParameter(
17796a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
17806a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != OK) {
17816a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        return err;
17826a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
17836a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
17849bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    err = native_window_set_scaling_mode(mNativeWindow.get(),
17859bc7af17974f448291a44912566ec7472a0d798bMathias Agopian            NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW);
17869bc7af17974f448291a44912566ec7472a0d798bMathias Agopian
17879bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    if (err != OK) {
17889bc7af17974f448291a44912566ec7472a0d798bMathias Agopian        return err;
17899bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    }
17909bc7af17974f448291a44912566ec7472a0d798bMathias Agopian
17916a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    err = native_window_set_buffers_geometry(
17926a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            mNativeWindow.get(),
17936a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            def.format.video.nFrameWidth,
17946a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            def.format.video.nFrameHeight,
17950821a824a718a28fa5144309bf09ea40411c8ae0Jamie Gennis            def.format.video.eColorFormat);
17966a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
17976a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != 0) {
17986a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        LOGE("native_window_set_buffers_geometry failed: %s (%d)",
17996a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                strerror(-err), -err);
18006a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        return err;
18016a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
18026a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
18037fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    err = applyRotation();
18047fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    if (err != OK) {
18057fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber        return err;
18067fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber    }
18077fac331b39ca49ce49a67e425dcc031a3cb9e97fAndreas Huber
18086a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // Set up the native window.
18093c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    OMX_U32 usage = 0;
18103c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    err = mOMX->getGraphicBufferUsage(mNode, kPortIndexOutput, &usage);
18113c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    if (err != 0) {
18123c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        LOGW("querying usage flags from OMX IL component failed: %d", err);
18133c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        // XXX: Currently this error is logged, but not fatal.
18143c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        usage = 0;
18153c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    }
18160bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if (mFlags & kEnableGrallocUsageProtected) {
18172eb62955eb84b97695e8a7e56e14310cbb86412bGlenn Kasten        usage |= GRALLOC_USAGE_PROTECTED;
18182eb62955eb84b97695e8a7e56e14310cbb86412bGlenn Kasten    }
18193c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis
1820d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis    // Make sure to check whether either Stagefright or the video decoder
1821d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis    // requested protected buffers.
1822d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis    if (usage & GRALLOC_USAGE_PROTECTED) {
1823d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        // Verify that the ANativeWindow sends images directly to
1824d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        // SurfaceFlinger.
1825d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        int queuesToNativeWindow = 0;
1826d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        err = mNativeWindow->query(
1827d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis                mNativeWindow.get(), NATIVE_WINDOW_QUEUES_TO_WINDOW_COMPOSER,
1828d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis                &queuesToNativeWindow);
1829d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        if (err != 0) {
1830d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis            LOGE("error authenticating native window: %d", err);
1831d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis            return err;
1832d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        }
1833d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        if (queuesToNativeWindow != 1) {
1834d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis            LOGE("native window could not be authenticated");
1835d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis            return PERMISSION_DENIED;
1836d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis        }
1837d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis    }
1838d4d43b2ad0f715211feb48ce2f3e2158883583c0Jamie Gennis
18390055e9e1c67e4614135da2cc0866caf3e7bfa3d6Andreas Huber    LOGV("native_window_set_usage usage=0x%lx", usage);
18406a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    err = native_window_set_usage(
18413c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis            mNativeWindow.get(), usage | GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_EXTERNAL_DISP);
18426a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != 0) {
18436a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        LOGE("native_window_set_usage failed: %s (%d)", strerror(-err), -err);
18446a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        return err;
18456a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
18466a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
1847258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    int minUndequeuedBufs = 0;
1848258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    err = mNativeWindow->query(mNativeWindow.get(),
1849258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS, &minUndequeuedBufs);
1850258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    if (err != 0) {
1851258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        LOGE("NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS query failed: %s (%d)",
1852258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                strerror(-err), -err);
1853258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        return err;
1854258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
1855258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
1856258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // XXX: Is this the right logic to use?  It's not clear to me what the OMX
1857258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // buffer counts refer to - how do they account for the renderer holding on
1858258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // to buffers?
1859258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    if (def.nBufferCountActual < def.nBufferCountMin + minUndequeuedBufs) {
1860258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        OMX_U32 newBufferCount = def.nBufferCountMin + minUndequeuedBufs;
1861258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        def.nBufferCountActual = newBufferCount;
1862258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        err = mOMX->setParameter(
1863258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
1864258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        if (err != OK) {
1865258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            CODEC_LOGE("setting nBufferCountActual to %lu failed: %d",
1866258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                    newBufferCount, err);
1867258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            return err;
1868258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        }
1869258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
1870258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
18716a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    err = native_window_set_buffer_count(
18726a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            mNativeWindow.get(), def.nBufferCountActual);
18736a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != 0) {
18746a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        LOGE("native_window_set_buffer_count failed: %s (%d)", strerror(-err),
18756a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                -err);
18766a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        return err;
18776a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
18786a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
18796a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    CODEC_LOGI("allocating %lu buffers from a native window of size %lu on "
18806a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            "output port", def.nBufferCountActual, def.nBufferSize);
18816a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
18826a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // Dequeue buffers and send them to OMX
188374006804065941841883c4b46ee785070164023fJamie Gennis    for (OMX_U32 i = 0; i < def.nBufferCountActual; i++) {
18848ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev        ANativeWindowBuffer* buf;
18856a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf);
18866a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        if (err != 0) {
18876a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            LOGE("dequeueBuffer failed: %s (%d)", strerror(-err), -err);
18886a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            break;
18896a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        }
18906a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
18916a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(buf, false));
189274006804065941841883c4b46ee785070164023fJamie Gennis        BufferInfo info;
189374006804065941841883c4b46ee785070164023fJamie Gennis        info.mData = NULL;
189474006804065941841883c4b46ee785070164023fJamie Gennis        info.mSize = def.nBufferSize;
189574006804065941841883c4b46ee785070164023fJamie Gennis        info.mStatus = OWNED_BY_US;
189674006804065941841883c4b46ee785070164023fJamie Gennis        info.mMem = NULL;
189774006804065941841883c4b46ee785070164023fJamie Gennis        info.mMediaBuffer = new MediaBuffer(graphicBuffer);
189874006804065941841883c4b46ee785070164023fJamie Gennis        info.mMediaBuffer->setObserver(this);
189974006804065941841883c4b46ee785070164023fJamie Gennis        mPortBuffers[kPortIndexOutput].push(info);
190074006804065941841883c4b46ee785070164023fJamie Gennis
19016a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        IOMX::buffer_id bufferId;
19026a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        err = mOMX->useGraphicBuffer(mNode, kPortIndexOutput, graphicBuffer,
19036a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                &bufferId);
19046a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        if (err != 0) {
190574006804065941841883c4b46ee785070164023fJamie Gennis            CODEC_LOGE("registering GraphicBuffer with OMX IL component "
190674006804065941841883c4b46ee785070164023fJamie Gennis                    "failed: %d", err);
19076a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            break;
19086a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        }
19096a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
191074006804065941841883c4b46ee785070164023fJamie Gennis        mPortBuffers[kPortIndexOutput].editItemAt(i).mBuffer = bufferId;
191174006804065941841883c4b46ee785070164023fJamie Gennis
19126a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        CODEC_LOGV("registered graphic buffer with ID %p (pointer = %p)",
19136a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                bufferId, graphicBuffer.get());
19146a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
19156a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19166a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    OMX_U32 cancelStart;
19176a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    OMX_U32 cancelEnd;
19186a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != 0) {
19196a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        // If an error occurred while dequeuing we need to cancel any buffers
19206a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        // that were dequeued.
19216a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        cancelStart = 0;
192274006804065941841883c4b46ee785070164023fJamie Gennis        cancelEnd = mPortBuffers[kPortIndexOutput].size();
19236a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    } else {
19246a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        // Return the last two buffers to the native window.
1925258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        cancelStart = def.nBufferCountActual - minUndequeuedBufs;
19266a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        cancelEnd = def.nBufferCountActual;
19276a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
19286a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19296a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    for (OMX_U32 i = cancelStart; i < cancelEnd; i++) {
19306a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        BufferInfo *info = &mPortBuffers[kPortIndexOutput].editItemAt(i);
19316a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        cancelBufferToNativeWindow(info);
19326a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
19336a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19346a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    return err;
19356a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis}
19366a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19376a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennisstatus_t OMXCodec::cancelBufferToNativeWindow(BufferInfo *info) {
193892bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    CHECK_EQ((int)info->mStatus, (int)OWNED_BY_US);
19396a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    CODEC_LOGV("Calling cancelBuffer on buffer %p", info->mBuffer);
19406a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    int err = mNativeWindow->cancelBuffer(
19416a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        mNativeWindow.get(), info->mMediaBuffer->graphicBuffer().get());
19426a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != 0) {
19436a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      CODEC_LOGE("cancelBuffer failed w/ error 0x%08x", err);
19446a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19456a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      setState(ERROR);
19466a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      return err;
19476a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
194892bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    info->mStatus = OWNED_BY_NATIVE_WINDOW;
19496a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    return OK;
19506a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis}
19516a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19526a9da9fc558263548ebfbae2cbf177eb7454a41bJamie GennisOMXCodec::BufferInfo* OMXCodec::dequeueBufferFromNativeWindow() {
19536a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // Dequeue the next buffer from the native window.
19548ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev    ANativeWindowBuffer* buf;
19556a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    int err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf);
19566a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != 0) {
19576a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      CODEC_LOGE("dequeueBuffer failed w/ error 0x%08x", err);
19586a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19596a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      setState(ERROR);
19606a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      return 0;
19616a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
19626a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19636a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // Determine which buffer we just dequeued.
19646a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexOutput];
19656a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    BufferInfo *bufInfo = 0;
19666a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    for (size_t i = 0; i < buffers->size(); i++) {
19676a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      sp<GraphicBuffer> graphicBuffer = buffers->itemAt(i).
19686a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis          mMediaBuffer->graphicBuffer();
19696a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      if (graphicBuffer->handle == buf->handle) {
19706a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        bufInfo = &buffers->editItemAt(i);
19716a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        break;
19726a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis      }
19736a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
19746a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19756a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (bufInfo == 0) {
19766a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        CODEC_LOGE("dequeued unrecognized buffer: %p", buf);
19776a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19786a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        setState(ERROR);
19796a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        return 0;
19806a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
19816a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19826a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // The native window no longer owns the buffer.
198392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    CHECK_EQ((int)bufInfo->mStatus, (int)OWNED_BY_NATIVE_WINDOW);
198492bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    bufInfo->mStatus = OWNED_BY_US;
19856a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
19866a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    return bufInfo;
19876a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis}
19886a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
1989965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dongint64_t OMXCodec::retrieveDecodingTimeUs(bool isCodecSpecific) {
1990965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    CHECK(mIsEncoder);
199103168ed2f76e12e5dd41740697ea8206759924c1James Dong
199203168ed2f76e12e5dd41740697ea8206759924c1James Dong    if (mDecodingTimeList.empty()) {
1993e8eb2f4f0f8eff0313a27892ab8514223553d943James Dong        CHECK(mSignalledEOS || mNoMoreOutputData);
199403168ed2f76e12e5dd41740697ea8206759924c1James Dong        // No corresponding input frame available.
199503168ed2f76e12e5dd41740697ea8206759924c1James Dong        // This could happen when EOS is reached.
199603168ed2f76e12e5dd41740697ea8206759924c1James Dong        return 0;
199703168ed2f76e12e5dd41740697ea8206759924c1James Dong    }
199803168ed2f76e12e5dd41740697ea8206759924c1James Dong
1999965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    List<int64_t>::iterator it = mDecodingTimeList.begin();
2000965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    int64_t timeUs = *it;
2001965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong
2002965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    // If the output buffer is codec specific configuration,
2003965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    // do not remove the decoding time from the list.
2004965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    if (!isCodecSpecific) {
2005965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong        mDecodingTimeList.erase(it);
2006965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    }
2007965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong    return timeUs;
2008965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong}
2009965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong
2010693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::on_message(const omx_message &msg) {
20110c03d5c7c2fa4d17f7f5159e3fddd2adf6bfc923Andreas Huber    if (mState == ERROR) {
20120c03d5c7c2fa4d17f7f5159e3fddd2adf6bfc923Andreas Huber        LOGW("Dropping OMX message - we're in ERROR state.");
20130c03d5c7c2fa4d17f7f5159e3fddd2adf6bfc923Andreas Huber        return;
20140c03d5c7c2fa4d17f7f5159e3fddd2adf6bfc923Andreas Huber    }
20150c03d5c7c2fa4d17f7f5159e3fddd2adf6bfc923Andreas Huber
2016693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    switch (msg.type) {
2017693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case omx_message::EVENT:
2018693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2019693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            onEvent(
2020693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                 msg.u.event_data.event, msg.u.event_data.data1,
2021693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                 msg.u.event_data.data2);
2022693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2023693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2024693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2025693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2026693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case omx_message::EMPTY_BUFFER_DONE:
2027693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2028693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            IOMX::buffer_id buffer = msg.u.extended_buffer_data.buffer;
2029693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
20307a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("EMPTY_BUFFER_DONE(buffer: %p)", buffer);
2031693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2032693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexInput];
2033693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            size_t i = 0;
2034693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            while (i < buffers->size() && (*buffers)[i].mBuffer != buffer) {
2035693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                ++i;
2036693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2037693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2038693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(i < buffers->size());
203992bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            if ((*buffers)[i].mStatus != OWNED_BY_COMPONENT) {
2040693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                LOGW("We already own input buffer %p, yet received "
2041693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                     "an EMPTY_BUFFER_DONE.", buffer);
2042693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2043693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
20448480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            BufferInfo* info = &buffers->editItemAt(i);
204592bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            info->mStatus = OWNED_BY_US;
20468480835b4bc1350646376aa7f3ae33742a7adeb1James Dong
20478480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            // Buffer could not be released until empty buffer done is called.
20488480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            if (info->mMediaBuffer != NULL) {
2049f23c4f92c3b0202435cf87db2642156fabc46f02James Dong                if (mIsEncoder &&
2050f23c4f92c3b0202435cf87db2642156fabc46f02James Dong                    (mQuirks & kAvoidMemcopyInputRecordingFrames)) {
2051f23c4f92c3b0202435cf87db2642156fabc46f02James Dong                    // If zero-copy mode is enabled this will send the
2052f23c4f92c3b0202435cf87db2642156fabc46f02James Dong                    // input buffer back to the upstream source.
2053f23c4f92c3b0202435cf87db2642156fabc46f02James Dong                    restorePatchedDataPointer(info);
2054f23c4f92c3b0202435cf87db2642156fabc46f02James Dong                }
2055f23c4f92c3b0202435cf87db2642156fabc46f02James Dong
20568480835b4bc1350646376aa7f3ae33742a7adeb1James Dong                info->mMediaBuffer->release();
20578480835b4bc1350646376aa7f3ae33742a7adeb1James Dong                info->mMediaBuffer = NULL;
20588480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            }
2059693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2060693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (mPortStatus[kPortIndexInput] == DISABLING) {
20617a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                CODEC_LOGV("Port is disabled, freeing buffer %p", buffer);
2062693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2063ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis                status_t err = freeBuffer(kPortIndexInput, i);
2064f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
2065ec9dd59902c8beea4ba6a842f3a843d46150d949Andreas Huber            } else if (mState != ERROR
2066ec9dd59902c8beea4ba6a842f3a843d46150d949Andreas Huber                    && mPortStatus[kPortIndexInput] != SHUTTING_DOWN) {
2067f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ((int)mPortStatus[kPortIndexInput], (int)ENABLED);
20680bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
20690bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                if (mFlags & kUseSecureInputBuffers) {
20700bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    drainAnyInputBuffer();
20710bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                } else {
20720bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    drainInputBuffer(&buffers->editItemAt(i));
20730bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                }
2074693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2075693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2076693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2077693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2078693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case omx_message::FILL_BUFFER_DONE:
2079693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2080693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            IOMX::buffer_id buffer = msg.u.extended_buffer_data.buffer;
2081693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_U32 flags = msg.u.extended_buffer_data.flags;
2082693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2083134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber            CODEC_LOGV("FILL_BUFFER_DONE(buffer: %p, size: %ld, flags: 0x%08lx, timestamp: %lld us (%.2f secs))",
2084693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                 buffer,
2085693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                 msg.u.extended_buffer_data.range_length,
2086134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber                 flags,
2087693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                 msg.u.extended_buffer_data.timestamp,
2088693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                 msg.u.extended_buffer_data.timestamp / 1E6);
2089693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2090693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexOutput];
2091693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            size_t i = 0;
2092693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            while (i < buffers->size() && (*buffers)[i].mBuffer != buffer) {
2093693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                ++i;
2094693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2095693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2096693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(i < buffers->size());
2097693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            BufferInfo *info = &buffers->editItemAt(i);
2098693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
209992bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            if (info->mStatus != OWNED_BY_COMPONENT) {
2100693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                LOGW("We already own output buffer %p, yet received "
2101693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                     "a FILL_BUFFER_DONE.", buffer);
2102693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2103693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
210492bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            info->mStatus = OWNED_BY_US;
2105693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2106693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (mPortStatus[kPortIndexOutput] == DISABLING) {
21077a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                CODEC_LOGV("Port is disabled, freeing buffer %p", buffer);
2108693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2109ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis                status_t err = freeBuffer(kPortIndexOutput, i);
2110f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
2111693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2112134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber#if 0
2113355edcea2f15c0f619c1e1d0f4fa433b0d38098fAndreas Huber            } else if (mPortStatus[kPortIndexOutput] == ENABLED
2114355edcea2f15c0f619c1e1d0f4fa433b0d38098fAndreas Huber                       && (flags & OMX_BUFFERFLAG_EOS)) {
21157a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                CODEC_LOGV("No more output data.");
2116693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mNoMoreOutputData = true;
2117693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mBufferFilled.signal();
2118134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber#endif
2119693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            } else if (mPortStatus[kPortIndexOutput] != SHUTTING_DOWN) {
2120f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ((int)mPortStatus[kPortIndexOutput], (int)ENABLED);
2121bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
2122213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                if (info->mMediaBuffer == NULL) {
2123213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    CHECK(mOMXLivesLocally);
2124213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    CHECK(mQuirks & kRequiresAllocateBufferOnOutputPorts);
2125213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    CHECK(mQuirks & kDefersOutputBufferAllocation);
2126213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber
2127213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    // The qcom video decoders on Nexus don't actually allocate
2128213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    // output buffer memory on a call to OMX_AllocateBuffer
2129213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    // the "pBuffer" member of the OMX_BUFFERHEADERTYPE
2130213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    // structure is only filled in later.
2131213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber
2132213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    info->mMediaBuffer = new MediaBuffer(
2133213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                            msg.u.extended_buffer_data.data_ptr,
2134213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                            info->mSize);
2135213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                    info->mMediaBuffer->setObserver(this);
2136213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber                }
2137213addfaf4b359c69da4e9b4490c511d116845bbAndreas Huber
2138693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                MediaBuffer *buffer = info->mMediaBuffer;
21396a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                bool isGraphicBuffer = buffer->graphicBuffer() != NULL;
2140693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
21416a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                if (!isGraphicBuffer
21426a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                    && msg.u.extended_buffer_data.range_offset
2143f8ca90452ff3e252f20de38f1c3eee524c808c3eAndreas Huber                        + msg.u.extended_buffer_data.range_length
2144f8ca90452ff3e252f20de38f1c3eee524c808c3eAndreas Huber                            > buffer->size()) {
2145f8ca90452ff3e252f20de38f1c3eee524c808c3eAndreas Huber                    CODEC_LOGE(
2146f8ca90452ff3e252f20de38f1c3eee524c808c3eAndreas Huber                            "Codec lied about its buffer size requirements, "
2147f8ca90452ff3e252f20de38f1c3eee524c808c3eAndreas Huber                            "sending a buffer larger than the originally "
2148f8ca90452ff3e252f20de38f1c3eee524c808c3eAndreas Huber                            "advertised size in FILL_BUFFER_DONE!");
2149f8ca90452ff3e252f20de38f1c3eee524c808c3eAndreas Huber                }
2150693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                buffer->set_range(
2151693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        msg.u.extended_buffer_data.range_offset,
2152693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        msg.u.extended_buffer_data.range_length);
2153693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2154693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                buffer->meta_data()->clear();
2155693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
215648c948b1137e7bbdb161b51908657ab72ac5e2daAndreas Huber                buffer->meta_data()->setInt64(
215748c948b1137e7bbdb161b51908657ab72ac5e2daAndreas Huber                        kKeyTime, msg.u.extended_buffer_data.timestamp);
2158693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2159693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                if (msg.u.extended_buffer_data.flags & OMX_BUFFERFLAG_SYNCFRAME) {
2160693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    buffer->meta_data()->setInt32(kKeyIsSyncFrame, true);
2161693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                }
2162965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong                bool isCodecSpecific = false;
216330ab66297501757d745b9ae10da61adcd891f497Andreas Huber                if (msg.u.extended_buffer_data.flags & OMX_BUFFERFLAG_CODECCONFIG) {
216430ab66297501757d745b9ae10da61adcd891f497Andreas Huber                    buffer->meta_data()->setInt32(kKeyIsCodecConfig, true);
2165965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong                    isCodecSpecific = true;
216630ab66297501757d745b9ae10da61adcd891f497Andreas Huber                }
2167693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
21686a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                if (isGraphicBuffer || mQuirks & kOutputBuffersAreUnreadable) {
21692b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber                    buffer->meta_data()->setInt32(kKeyIsUnreadable, true);
21702b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber                }
21712b82e9652ba049e754c2cc74e381282f231d5fbfAndreas Huber
2172693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                buffer->meta_data()->setPointer(
2173693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        kKeyPlatformPrivate,
2174693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        msg.u.extended_buffer_data.platform_private);
2175693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2176693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                buffer->meta_data()->setPointer(
2177693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        kKeyBufferID,
2178693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        msg.u.extended_buffer_data.buffer);
2179693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2180134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber                if (msg.u.extended_buffer_data.flags & OMX_BUFFERFLAG_EOS) {
2181134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber                    CODEC_LOGV("No more output data.");
2182134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber                    mNoMoreOutputData = true;
2183134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber                }
2184abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber
218503168ed2f76e12e5dd41740697ea8206759924c1James Dong                if (mIsEncoder) {
218603168ed2f76e12e5dd41740697ea8206759924c1James Dong                    int64_t decodingTimeUs = retrieveDecodingTimeUs(isCodecSpecific);
218703168ed2f76e12e5dd41740697ea8206759924c1James Dong                    buffer->meta_data()->setInt64(kKeyDecodingTime, decodingTimeUs);
218803168ed2f76e12e5dd41740697ea8206759924c1James Dong                }
218903168ed2f76e12e5dd41740697ea8206759924c1James Dong
2190abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                if (mTargetTimeUs >= 0) {
2191abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                    CHECK(msg.u.extended_buffer_data.timestamp <= mTargetTimeUs);
2192abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber
2193abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                    if (msg.u.extended_buffer_data.timestamp < mTargetTimeUs) {
2194abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                        CODEC_LOGV(
2195abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                                "skipping output buffer at timestamp %lld us",
2196abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                                msg.u.extended_buffer_data.timestamp);
2197abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber
2198abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                        fillOutputBuffer(info);
2199abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                        break;
2200abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                    }
2201abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber
2202abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                    CODEC_LOGV(
2203abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                            "returning output buffer at target timestamp "
2204abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                            "%lld us",
2205abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                            msg.u.extended_buffer_data.timestamp);
2206abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber
2207abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                    mTargetTimeUs = -1;
2208abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                }
2209abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber
2210abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                mFilledBuffers.push_back(i);
2211abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                mBufferFilled.signal();
2212985f838934510983d8a887461e98dca60a6e858fJames Dong                if (mIsEncoder) {
2213985f838934510983d8a887461e98dca60a6e858fJames Dong                    sched_yield();
2214985f838934510983d8a887461e98dca60a6e858fJames Dong                }
2215693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2216693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2217693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2218693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2219693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2220693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        default:
2221693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2222693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(!"should not be here.");
2223693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2224693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2225693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2226693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2227693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
222812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber// Has the format changed in any way that the client would have to be aware of?
222912658b720b176b7d934444256d07ba3c595d9b44Andreas Huberstatic bool formatHasNotablyChanged(
223012658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        const sp<MetaData> &from, const sp<MetaData> &to) {
223112658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    if (from.get() == NULL && to.get() == NULL) {
223212658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        return false;
223312658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    }
223412658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
2235c6323f9d72e0f08dd1e813550a56c8d2f0fd76a2Andreas Huber    if ((from.get() == NULL && to.get() != NULL)
2236c6323f9d72e0f08dd1e813550a56c8d2f0fd76a2Andreas Huber        || (from.get() != NULL && to.get() == NULL)) {
223712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        return true;
223812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    }
223912658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
224012658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    const char *mime_from, *mime_to;
224112658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    CHECK(from->findCString(kKeyMIMEType, &mime_from));
224212658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    CHECK(to->findCString(kKeyMIMEType, &mime_to));
224312658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
224412658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    if (strcasecmp(mime_from, mime_to)) {
224512658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        return true;
224612658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    }
224712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
224812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    if (!strcasecmp(mime_from, MEDIA_MIMETYPE_VIDEO_RAW)) {
224912658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        int32_t colorFormat_from, colorFormat_to;
225012658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(from->findInt32(kKeyColorFormat, &colorFormat_from));
225112658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(to->findInt32(kKeyColorFormat, &colorFormat_to));
225212658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
225312658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        if (colorFormat_from != colorFormat_to) {
225412658b720b176b7d934444256d07ba3c595d9b44Andreas Huber            return true;
225512658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        }
225612658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
225712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        int32_t width_from, width_to;
225812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(from->findInt32(kKeyWidth, &width_from));
225912658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(to->findInt32(kKeyWidth, &width_to));
226012658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
226112658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        if (width_from != width_to) {
226212658b720b176b7d934444256d07ba3c595d9b44Andreas Huber            return true;
226312658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        }
226412658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
226512658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        int32_t height_from, height_to;
226612658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(from->findInt32(kKeyHeight, &height_from));
226712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(to->findInt32(kKeyHeight, &height_to));
226812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
226912658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        if (height_from != height_to) {
227012658b720b176b7d934444256d07ba3c595d9b44Andreas Huber            return true;
227112658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        }
2272f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2273f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        int32_t left_from, top_from, right_from, bottom_from;
2274f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK(from->findRect(
2275f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    kKeyCropRect,
2276f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    &left_from, &top_from, &right_from, &bottom_from));
2277f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2278f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        int32_t left_to, top_to, right_to, bottom_to;
2279f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK(to->findRect(
2280f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    kKeyCropRect,
2281f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    &left_to, &top_to, &right_to, &bottom_to));
2282f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2283f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        if (left_to != left_from || top_to != top_from
2284f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                || right_to != right_from || bottom_to != bottom_from) {
2285f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            return true;
2286f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        }
228712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    } else if (!strcasecmp(mime_from, MEDIA_MIMETYPE_AUDIO_RAW)) {
228812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        int32_t numChannels_from, numChannels_to;
228912658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(from->findInt32(kKeyChannelCount, &numChannels_from));
229012658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(to->findInt32(kKeyChannelCount, &numChannels_to));
229112658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
229212658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        if (numChannels_from != numChannels_to) {
229312658b720b176b7d934444256d07ba3c595d9b44Andreas Huber            return true;
229412658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        }
229512658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
229612658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        int32_t sampleRate_from, sampleRate_to;
229712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(from->findInt32(kKeySampleRate, &sampleRate_from));
229812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        CHECK(to->findInt32(kKeySampleRate, &sampleRate_to));
229912658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
230012658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        if (sampleRate_from != sampleRate_to) {
230112658b720b176b7d934444256d07ba3c595d9b44Andreas Huber            return true;
230212658b720b176b7d934444256d07ba3c595d9b44Andreas Huber        }
230312658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    }
230412658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
230512658b720b176b7d934444256d07ba3c595d9b44Andreas Huber    return false;
230612658b720b176b7d934444256d07ba3c595d9b44Andreas Huber}
230712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
2308f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Hubervoid OMXCodec::onEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2309f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    switch (event) {
2310f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        case OMX_EventCmdComplete:
2311f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        {
2312f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            onCmdComplete((OMX_COMMANDTYPE)data1, data2);
2313f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            break;
2314f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        }
2315f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2316f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        case OMX_EventError:
2317f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        {
2318f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CODEC_LOGE("ERROR(0x%08lx, %ld)", data1, data2);
2319f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2320f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            setState(ERROR);
2321f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            break;
2322f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        }
2323f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2324f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        case OMX_EventPortSettingsChanged:
2325f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        {
2326f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CODEC_LOGV("OMX_EventPortSettingsChanged(port=%ld, data2=0x%08lx)",
2327f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                       data1, data2);
2328f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2329f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            if (data2 == 0 || data2 == OMX_IndexParamPortDefinition) {
23308edb8e82fa886564ee8e72178a1969e2437dd525James Dong                // There is no need to check whether mFilledBuffers is empty or not
23318edb8e82fa886564ee8e72178a1969e2437dd525James Dong                // when the OMX_EventPortSettingsChanged is not meant for reallocating
23328edb8e82fa886564ee8e72178a1969e2437dd525James Dong                // the output buffers.
23338edb8e82fa886564ee8e72178a1969e2437dd525James Dong                if (data1 == kPortIndexOutput) {
23348edb8e82fa886564ee8e72178a1969e2437dd525James Dong                    CHECK(mFilledBuffers.empty());
23358edb8e82fa886564ee8e72178a1969e2437dd525James Dong                }
2336f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                onPortSettingsChanged(data1);
23379cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong            } else if (data1 == kPortIndexOutput &&
23389cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        (data2 == OMX_IndexConfigCommonOutputCrop ||
23399cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                         data2 == OMX_IndexConfigCommonScale)) {
2340f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2341f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                sp<MetaData> oldOutputFormat = mOutputFormat;
2342f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                initOutputFormat(mSource->getFormat());
2343f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
23449cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                if (data2 == OMX_IndexConfigCommonOutputCrop &&
23459cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                    formatHasNotablyChanged(oldOutputFormat, mOutputFormat)) {
2346f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    mOutputPortSettingsHaveChanged = true;
2347f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2348f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    if (mNativeWindow != NULL) {
2349f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                        int32_t left, top, right, bottom;
2350f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                        CHECK(mOutputFormat->findRect(
2351f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                                    kKeyCropRect,
2352f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                                    &left, &top, &right, &bottom));
2353f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2354f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                        android_native_rect_t crop;
2355f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                        crop.left = left;
2356f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                        crop.top = top;
235789c120e7adbe09c6283591789594c5e591aa5032Andreas Huber                        crop.right = right + 1;
235889c120e7adbe09c6283591789594c5e591aa5032Andreas Huber                        crop.bottom = bottom + 1;
2359f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
23606592195d76b12a9823f82d3352245d0a6d459647Andreas Huber                        // We'll ignore any errors here, if the surface is
23616592195d76b12a9823f82d3352245d0a6d459647Andreas Huber                        // already invalid, we'll know soon enough.
23626592195d76b12a9823f82d3352245d0a6d459647Andreas Huber                        native_window_set_crop(mNativeWindow.get(), &crop);
2363f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    }
23649cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                } else if (data2 == OMX_IndexConfigCommonScale) {
23659cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                    OMX_CONFIG_SCALEFACTORTYPE scale;
23669cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                    InitOMXParams(&scale);
23679cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                    scale.nPortIndex = kPortIndexOutput;
23689cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong
23699cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                    // Change display dimension only when necessary.
23709cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                    if (OK == mOMX->getConfig(
23719cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                        mNode,
23729cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                        OMX_IndexConfigCommonScale,
23739cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                        &scale, sizeof(scale))) {
23749cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        int32_t left, top, right, bottom;
23759cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        CHECK(mOutputFormat->findRect(kKeyCropRect,
23769cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                                      &left, &top,
23779cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                                      &right, &bottom));
23789cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong
23799cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        // The scale is in 16.16 format.
23809cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        // scale 1.0 = 0x010000. When there is no
23819cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        // need to change the display, skip it.
23829cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        LOGV("Get OMX_IndexConfigScale: 0x%lx/0x%lx",
23839cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                scale.xWidth, scale.xHeight);
23849cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong
23859cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        if (scale.xWidth != 0x010000) {
23869cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                            mOutputFormat->setInt32(kKeyDisplayWidth,
23879cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                    ((right - left +  1) * scale.xWidth)  >> 16);
23889cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                            mOutputPortSettingsHaveChanged = true;
23899cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        }
23909cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong
23919cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        if (scale.xHeight != 0x010000) {
23929cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                            mOutputFormat->setInt32(kKeyDisplayHeight,
23939cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                                    ((bottom  - top + 1) * scale.xHeight) >> 16);
23949cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                            mOutputPortSettingsHaveChanged = true;
23959cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                        }
23969cbb1a6f99003a9cd8765e11b9d7380e67839f82James Dong                    }
2397f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                }
2398f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            }
2399f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            break;
2400f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        }
2401f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2402f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber#if 0
2403f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        case OMX_EventBufferFlag:
2404f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        {
2405f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CODEC_LOGV("EVENT_BUFFER_FLAG(%ld)", data1);
2406f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2407f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            if (data1 == kPortIndexOutput) {
2408f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                mNoMoreOutputData = true;
2409f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            }
2410f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            break;
2411f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        }
2412f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber#endif
2413f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2414f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        default:
2415f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        {
2416f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CODEC_LOGV("EVENT(%d, %ld, %ld)", event, data1, data2);
2417f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            break;
2418f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        }
2419f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    }
2420f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber}
2421f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
2422693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::onCmdComplete(OMX_COMMANDTYPE cmd, OMX_U32 data) {
2423693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    switch (cmd) {
2424693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_CommandStateSet:
2425693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2426693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            onStateChange((OMX_STATETYPE)data);
2427693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2428693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2429693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2430693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_CommandPortDisable:
2431693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2432693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_U32 portIndex = data;
24337a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("PORT_DISABLED(%ld)", portIndex);
2434693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2435693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(mState == EXECUTING || mState == RECONFIGURING);
2436f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((int)mPortStatus[portIndex], (int)DISABLING);
2437f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ(mPortBuffers[portIndex].size(), 0u);
2438693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2439693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mPortStatus[portIndex] = DISABLED;
2440693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2441693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (mState == RECONFIGURING) {
2442f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(portIndex, (OMX_U32)kPortIndexOutput);
2443693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
244412658b720b176b7d934444256d07ba3c595d9b44Andreas Huber                sp<MetaData> oldOutputFormat = mOutputFormat;
24457f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber                initOutputFormat(mSource->getFormat());
244612658b720b176b7d934444256d07ba3c595d9b44Andreas Huber
244712658b720b176b7d934444256d07ba3c595d9b44Andreas Huber                // Don't notify clients if the output port settings change
244812658b720b176b7d934444256d07ba3c595d9b44Andreas Huber                // wasn't of importance to them, i.e. it may be that just the
244912658b720b176b7d934444256d07ba3c595d9b44Andreas Huber                // number of buffers has changed and nothing else.
245097857479de48fda4c33bb415b2fbb15193f768e8James Dong                bool formatChanged = formatHasNotablyChanged(oldOutputFormat, mOutputFormat);
245197857479de48fda4c33bb415b2fbb15193f768e8James Dong                if (!mOutputPortSettingsHaveChanged) {
245297857479de48fda4c33bb415b2fbb15193f768e8James Dong                    mOutputPortSettingsHaveChanged = formatChanged;
245397857479de48fda4c33bb415b2fbb15193f768e8James Dong                }
24547f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber
2455693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                enablePortAsync(portIndex);
2456693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2457693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                status_t err = allocateBuffersOnPort(portIndex);
2458018761c5cc4aefd0495eaa4666f2372a59a7980aAndreas Huber
2459018761c5cc4aefd0495eaa4666f2372a59a7980aAndreas Huber                if (err != OK) {
2460018761c5cc4aefd0495eaa4666f2372a59a7980aAndreas Huber                    CODEC_LOGE("allocateBuffersOnPort failed (err = %d)", err);
2461018761c5cc4aefd0495eaa4666f2372a59a7980aAndreas Huber                    setState(ERROR);
2462018761c5cc4aefd0495eaa4666f2372a59a7980aAndreas Huber                }
2463693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2464693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2465693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2466693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2467693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_CommandPortEnable:
2468693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2469693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_U32 portIndex = data;
24707a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("PORT_ENABLED(%ld)", portIndex);
2471693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2472693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(mState == EXECUTING || mState == RECONFIGURING);
2473f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((int)mPortStatus[portIndex], (int)ENABLING);
2474693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2475693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mPortStatus[portIndex] = ENABLED;
2476693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2477693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (mState == RECONFIGURING) {
2478f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(portIndex, (OMX_U32)kPortIndexOutput);
2479693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2480693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                setState(EXECUTING);
2481693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2482693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                fillOutputBuffers();
2483693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2484693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2485693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2486693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2487693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_CommandFlush:
2488693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2489693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_U32 portIndex = data;
2490693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
24917a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("FLUSH_DONE(%ld)", portIndex);
2492693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2493f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((int)mPortStatus[portIndex], (int)SHUTTING_DOWN);
2494693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mPortStatus[portIndex] = ENABLED;
2495693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2496693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK_EQ(countBuffersWeOwn(mPortBuffers[portIndex]),
2497693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                     mPortBuffers[portIndex].size());
2498693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2499693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (mState == RECONFIGURING) {
2500f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(portIndex, (OMX_U32)kPortIndexOutput);
2501693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2502693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                disablePortAsync(portIndex);
25038297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber            } else if (mState == EXECUTING_TO_IDLE) {
25048297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                if (mPortStatus[kPortIndexInput] == ENABLED
25058297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    && mPortStatus[kPortIndexOutput] == ENABLED) {
25067a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                    CODEC_LOGV("Finished flushing both ports, now completing "
25078297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                         "transition from EXECUTING to IDLE.");
25088297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber
25098297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    mPortStatus[kPortIndexInput] = SHUTTING_DOWN;
25108297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    mPortStatus[kPortIndexOutput] = SHUTTING_DOWN;
25118297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber
25128297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    status_t err =
2513318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                        mOMX->sendCommand(mNode, OMX_CommandStateSet, OMX_StateIdle);
2514f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    CHECK_EQ(err, (status_t)OK);
25158297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                }
2516693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            } else {
2517693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                // We're flushing both ports in preparation for seeking.
2518693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2519693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                if (mPortStatus[kPortIndexInput] == ENABLED
2520693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    && mPortStatus[kPortIndexOutput] == ENABLED) {
25217a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                    CODEC_LOGV("Finished flushing both ports, now continuing from"
2522693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                         " seek-time.");
2523693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2524d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber                    // We implicitly resume pulling on our upstream source.
2525d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber                    mPaused = false;
2526d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber
2527693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    drainInputBuffers();
2528693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    fillOutputBuffers();
2529693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                }
25309c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber
25319c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber                if (mOutputPortSettingsChangedPending) {
25329c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber                    CODEC_LOGV(
25339c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber                            "Honoring deferred output port settings change.");
25349c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber
25359c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber                    mOutputPortSettingsChangedPending = false;
25369c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber                    onPortSettingsChanged(kPortIndexOutput);
25379c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber                }
2538693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2539693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2540693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2541693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2542693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2543693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        default:
2544693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
25457a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("CMD_COMPLETE(%d, %ld)", cmd, data);
2546693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2547693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2548693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2549693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2550693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2551693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::onStateChange(OMX_STATETYPE newState) {
2552570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber    CODEC_LOGV("onStateChange %d", newState);
2553570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber
2554693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    switch (newState) {
2555693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_StateIdle:
2556693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
25577a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("Now Idle.");
2558693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (mState == LOADED_TO_IDLE) {
2559318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                status_t err = mOMX->sendCommand(
2560693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        mNode, OMX_CommandStateSet, OMX_StateExecuting);
2561693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2562f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
2563693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2564693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                setState(IDLE_TO_EXECUTING);
2565693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            } else {
2566f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ((int)mState, (int)EXECUTING_TO_IDLE);
2567693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2568693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                CHECK_EQ(
2569693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    countBuffersWeOwn(mPortBuffers[kPortIndexInput]),
2570693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    mPortBuffers[kPortIndexInput].size());
2571693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2572693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                CHECK_EQ(
2573693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    countBuffersWeOwn(mPortBuffers[kPortIndexOutput]),
2574693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    mPortBuffers[kPortIndexOutput].size());
2575693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2576318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                status_t err = mOMX->sendCommand(
2577693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        mNode, OMX_CommandStateSet, OMX_StateLoaded);
2578693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2579f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
2580693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2581693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                err = freeBuffersOnPort(kPortIndexInput);
2582f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
2583693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2584693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                err = freeBuffersOnPort(kPortIndexOutput);
2585f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
2586693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2587693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mPortStatus[kPortIndexInput] = ENABLED;
2588693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mPortStatus[kPortIndexOutput] = ENABLED;
2589693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2590693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                setState(IDLE_TO_LOADED);
2591693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
2592693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2593693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2594693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2595693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_StateExecuting:
2596693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2597f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((int)mState, (int)IDLE_TO_EXECUTING);
2598693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
25997a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("Now Executing.");
2600693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
26019c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber            mOutputPortSettingsChangedPending = false;
26029c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber
2603693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            setState(EXECUTING);
2604693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2605284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber            // Buffers will be submitted to the component in the first
2606284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber            // call to OMXCodec::read as mInitialBufferSubmit is true at
2607284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber            // this point. This ensures that this on_message call returns,
2608284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber            // releases the lock and ::init can notice the state change and
2609284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber            // itself return.
2610693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2611693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2612693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2613693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_StateLoaded:
2614693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2615f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((int)mState, (int)IDLE_TO_LOADED);
2616693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
26177a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber            CODEC_LOGV("Now Loaded.");
2618693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2619693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            setState(LOADED);
2620693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2621693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2622693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2623570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        case OMX_StateInvalid:
2624570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        {
2625570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            setState(ERROR);
2626570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            break;
2627570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        }
2628570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber
2629693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        default:
2630693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
2631693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(!"should not be here.");
2632693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
2633693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2634693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2635693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2636693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2637693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber// static
2638693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubersize_t OMXCodec::countBuffersWeOwn(const Vector<BufferInfo> &buffers) {
2639693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    size_t n = 0;
2640693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (size_t i = 0; i < buffers.size(); ++i) {
264192bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        if (buffers[i].mStatus != OWNED_BY_COMPONENT) {
2642693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            ++n;
2643693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2644693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2645693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2646693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return n;
2647693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2648693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2649693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatus_t OMXCodec::freeBuffersOnPort(
2650693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        OMX_U32 portIndex, bool onlyThoseWeOwn) {
2651693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Vector<BufferInfo> *buffers = &mPortBuffers[portIndex];
2652693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2653693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t stickyErr = OK;
2654693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2655693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (size_t i = buffers->size(); i-- > 0;) {
2656693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        BufferInfo *info = &buffers->editItemAt(i);
2657693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
265892bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        if (onlyThoseWeOwn && info->mStatus == OWNED_BY_COMPONENT) {
2659693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            continue;
2660693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2661693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
266292bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        CHECK(info->mStatus == OWNED_BY_US
266392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber                || info->mStatus == OWNED_BY_NATIVE_WINDOW);
2664693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
266595301196d59fb88ef96342fd307a2be374b5079bAndreas Huber        CODEC_LOGV("freeing buffer %p on port %ld", info->mBuffer, portIndex);
266695301196d59fb88ef96342fd307a2be374b5079bAndreas Huber
2667ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        status_t err = freeBuffer(portIndex, i);
2668693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2669693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (err != OK) {
2670693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            stickyErr = err;
2671693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2672693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2673ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    }
2674693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2675ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    CHECK(onlyThoseWeOwn || buffers->isEmpty());
2676693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2677ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    return stickyErr;
2678ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis}
2679ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis
2680ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennisstatus_t OMXCodec::freeBuffer(OMX_U32 portIndex, size_t bufIndex) {
2681ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    Vector<BufferInfo> *buffers = &mPortBuffers[portIndex];
2682ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis
2683ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    BufferInfo *info = &buffers->editItemAt(bufIndex);
2684ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis
2685ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    status_t err = mOMX->freeBuffer(mNode, portIndex, info->mBuffer);
2686ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis
2687ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    if (err == OK && info->mMediaBuffer != NULL) {
2688f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(portIndex, (OMX_U32)kPortIndexOutput);
2689ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        info->mMediaBuffer->setObserver(NULL);
26906a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
2691ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        // Make sure nobody but us owns this buffer at this point.
2692ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        CHECK_EQ(info->mMediaBuffer->refcount(), 0);
2693ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis
2694ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        // Cancel the buffer if it belongs to an ANativeWindow.
2695ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        sp<GraphicBuffer> graphicBuffer = info->mMediaBuffer->graphicBuffer();
269692bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        if (info->mStatus == OWNED_BY_US && graphicBuffer != 0) {
2697ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis            err = cancelBufferToNativeWindow(info);
2698693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2699693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2700ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        info->mMediaBuffer->release();
2701f23c4f92c3b0202435cf87db2642156fabc46f02James Dong        info->mMediaBuffer = NULL;
2702693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2703693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2704ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    if (err == OK) {
2705ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis        buffers->removeAt(bufIndex);
2706ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    }
2707693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2708ed45fe0730a7e7fc9944741428f5a484350acc8aJamie Gennis    return err;
2709693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2710693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2711693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::onPortSettingsChanged(OMX_U32 portIndex) {
27127a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    CODEC_LOGV("PORT_SETTINGS_CHANGED(%ld)", portIndex);
2713693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2714f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)mState, (int)EXECUTING);
2715f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(portIndex, (OMX_U32)kPortIndexOutput);
27169c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber    CHECK(!mOutputPortSettingsChangedPending);
27179c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber
27189c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber    if (mPortStatus[kPortIndexOutput] != ENABLED) {
27199c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber        CODEC_LOGV("Deferring output port settings change.");
27209c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber        mOutputPortSettingsChangedPending = true;
27219c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber        return;
27229c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber    }
27239c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber
2724693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    setState(RECONFIGURING);
2725693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2726693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (mQuirks & kNeedsFlushBeforeDisable) {
27271beb760d920561679862ded945a04e370368c7f7Andreas Huber        if (!flushPortAsync(portIndex)) {
27281beb760d920561679862ded945a04e370368c7f7Andreas Huber            onCmdComplete(OMX_CommandFlush, portIndex);
27291beb760d920561679862ded945a04e370368c7f7Andreas Huber        }
2730693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
2731693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        disablePortAsync(portIndex);
2732693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2733693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2734693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
27351beb760d920561679862ded945a04e370368c7f7Andreas Huberbool OMXCodec::flushPortAsync(OMX_U32 portIndex) {
27368297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber    CHECK(mState == EXECUTING || mState == RECONFIGURING
27378297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber            || mState == EXECUTING_TO_IDLE);
2738693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
27397a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    CODEC_LOGV("flushPortAsync(%ld): we own %d out of %d buffers already.",
27401beb760d920561679862ded945a04e370368c7f7Andreas Huber         portIndex, countBuffersWeOwn(mPortBuffers[portIndex]),
27411beb760d920561679862ded945a04e370368c7f7Andreas Huber         mPortBuffers[portIndex].size());
27421beb760d920561679862ded945a04e370368c7f7Andreas Huber
2743f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)mPortStatus[portIndex], (int)ENABLED);
2744693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mPortStatus[portIndex] = SHUTTING_DOWN;
2745693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
27461beb760d920561679862ded945a04e370368c7f7Andreas Huber    if ((mQuirks & kRequiresFlushCompleteEmulation)
27471beb760d920561679862ded945a04e370368c7f7Andreas Huber        && countBuffersWeOwn(mPortBuffers[portIndex])
27481beb760d920561679862ded945a04e370368c7f7Andreas Huber                == mPortBuffers[portIndex].size()) {
27491beb760d920561679862ded945a04e370368c7f7Andreas Huber        // No flush is necessary and this component fails to send a
27501beb760d920561679862ded945a04e370368c7f7Andreas Huber        // flush-complete event in this case.
27511beb760d920561679862ded945a04e370368c7f7Andreas Huber
27521beb760d920561679862ded945a04e370368c7f7Andreas Huber        return false;
27531beb760d920561679862ded945a04e370368c7f7Andreas Huber    }
27541beb760d920561679862ded945a04e370368c7f7Andreas Huber
2755693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err =
2756318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        mOMX->sendCommand(mNode, OMX_CommandFlush, portIndex);
2757f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
27581beb760d920561679862ded945a04e370368c7f7Andreas Huber
27591beb760d920561679862ded945a04e370368c7f7Andreas Huber    return true;
2760693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2761693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2762693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::disablePortAsync(OMX_U32 portIndex) {
2763693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CHECK(mState == EXECUTING || mState == RECONFIGURING);
2764693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2765f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)mPortStatus[portIndex], (int)ENABLED);
2766693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mPortStatus[portIndex] = DISABLING;
2767693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
276827fdb181daebec3dbe477080adad94f81ed667adAndreas Huber    CODEC_LOGV("sending OMX_CommandPortDisable(%ld)", portIndex);
2769693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err =
2770318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        mOMX->sendCommand(mNode, OMX_CommandPortDisable, portIndex);
2771f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
2772693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2773693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    freeBuffersOnPort(portIndex, true);
2774693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2775693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2776693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::enablePortAsync(OMX_U32 portIndex) {
2777693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CHECK(mState == EXECUTING || mState == RECONFIGURING);
2778693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2779f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)mPortStatus[portIndex], (int)DISABLED);
2780693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mPortStatus[portIndex] = ENABLING;
2781693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
27826a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    CODEC_LOGV("sending OMX_CommandPortEnable(%ld)", portIndex);
2783693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err =
2784318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        mOMX->sendCommand(mNode, OMX_CommandPortEnable, portIndex);
2785f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
2786693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2787693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2788693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::fillOutputBuffers() {
2789f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)mState, (int)EXECUTING);
2790693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2791b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber    // This is a workaround for some decoders not properly reporting
2792b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber    // end-of-output-stream. If we own all input buffers and also own
2793b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber    // all output buffers and we already signalled end-of-input-stream,
2794b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber    // the end-of-output-stream is implied.
2795b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber    if (mSignalledEOS
2796b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber            && countBuffersWeOwn(mPortBuffers[kPortIndexInput])
2797b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber                == mPortBuffers[kPortIndexInput].size()
2798b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber            && countBuffersWeOwn(mPortBuffers[kPortIndexOutput])
2799b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber                == mPortBuffers[kPortIndexOutput].size()) {
2800b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber        mNoMoreOutputData = true;
2801b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber        mBufferFilled.signal();
2802b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber
2803b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber        return;
2804b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber    }
2805b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber
2806693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexOutput];
2807693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (size_t i = 0; i < buffers->size(); ++i) {
28086a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        BufferInfo *info = &buffers->editItemAt(i);
280992bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        if (info->mStatus == OWNED_BY_US) {
28106a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            fillOutputBuffer(&buffers->editItemAt(i));
28116a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        }
2812693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2813693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2814693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2815693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::drainInputBuffers() {
2816450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber    CHECK(mState == EXECUTING || mState == RECONFIGURING);
2817693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
28180bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if (mFlags & kUseSecureInputBuffers) {
28190bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexInput];
28200bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        for (size_t i = 0; i < buffers->size(); ++i) {
28210bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            if (!drainAnyInputBuffer()
28220bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    || (mFlags & kOnlySubmitOneInputBufferAtOneTime)) {
28230bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                break;
28240bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            }
28250bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        }
28260bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    } else {
28270bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexInput];
28280bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        for (size_t i = 0; i < buffers->size(); ++i) {
28290bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            BufferInfo *info = &buffers->editItemAt(i);
28307757f5010a771fb8824b6fdf9788f588a1577e3fJames Dong
28310bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            if (info->mStatus != OWNED_BY_US) {
28320bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                continue;
28330bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            }
28340bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28350bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            if (!drainInputBuffer(info)) {
28360bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                break;
28370bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            }
28380bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28390bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            if (mFlags & kOnlySubmitOneInputBufferAtOneTime) {
28400bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                break;
28410bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            }
28427757f5010a771fb8824b6fdf9788f588a1577e3fJames Dong        }
28430bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    }
28440bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber}
28457757f5010a771fb8824b6fdf9788f588a1577e3fJames Dong
28460bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huberbool OMXCodec::drainAnyInputBuffer() {
28470bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    return drainInputBuffer((BufferInfo *)NULL);
28480bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber}
28490bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28500bb6b4c5da2451ee0973372b0a3858b15c742689Andreas HuberOMXCodec::BufferInfo *OMXCodec::findInputBufferByDataPointer(void *ptr) {
28510bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    Vector<BufferInfo> *infos = &mPortBuffers[kPortIndexInput];
28520bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    for (size_t i = 0; i < infos->size(); ++i) {
28530bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        BufferInfo *info = &infos->editItemAt(i);
28540bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28550bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        if (info->mData == ptr) {
28560bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            CODEC_LOGV(
28570bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    "input buffer data ptr = %p, buffer_id = %p",
28580bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    ptr,
28590bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                    info->mBuffer);
28600bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28610bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            return info;
28627757f5010a771fb8824b6fdf9788f588a1577e3fJames Dong        }
28630bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    }
28647757f5010a771fb8824b6fdf9788f588a1577e3fJames Dong
28650bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    TRESPASS();
28660bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber}
28670bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28680bb6b4c5da2451ee0973372b0a3858b15c742689Andreas HuberOMXCodec::BufferInfo *OMXCodec::findEmptyInputBuffer() {
28690bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    Vector<BufferInfo> *infos = &mPortBuffers[kPortIndexInput];
28700bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    for (size_t i = 0; i < infos->size(); ++i) {
28710bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        BufferInfo *info = &infos->editItemAt(i);
28720bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28730bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        if (info->mStatus == OWNED_BY_US) {
28740bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            return info;
287592bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        }
2876693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
28770bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
28780bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    TRESPASS();
2879693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
2880693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
288192bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huberbool OMXCodec::drainInputBuffer(BufferInfo *info) {
28820bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if (info != NULL) {
28830bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        CHECK_EQ((int)info->mStatus, (int)OWNED_BY_US);
28840bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    }
2885693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2886693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (mSignalledEOS) {
288792bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        return false;
2888693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2889693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2890693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (mCodecSpecificDataIndex < mCodecSpecificData.size()) {
28910bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        CHECK(!(mFlags & kUseSecureInputBuffers));
28920bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
2893693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        const CodecSpecificData *specific =
2894693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mCodecSpecificData[mCodecSpecificDataIndex];
2895693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2896693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        size_t size = specific->mSize;
2897693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
289818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mMIME)
2899dbc03445db2bbf83b64f0c0a5dc62e61408864d7Andreas Huber                && !(mQuirks & kWantsNALFragments)) {
2900693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            static const uint8_t kNALStartCode[4] =
2901693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                    { 0x00, 0x00, 0x00, 0x01 };
2902693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2903570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            CHECK(info->mSize >= specific->mSize + 4);
2904693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2905693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            size += 4;
2906693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2907570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            memcpy(info->mData, kNALStartCode, 4);
2908570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            memcpy((uint8_t *)info->mData + 4,
2909693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                   specific->mData, specific->mSize);
2910693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        } else {
2911570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            CHECK(info->mSize >= specific->mSize);
2912570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber            memcpy(info->mData, specific->mData, specific->mSize);
2913693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
2914693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2915134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber        mNoMoreOutputData = false;
2916134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber
2917b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber        CODEC_LOGV("calling emptyBuffer with codec specific data");
2918b03fd8c97695d381e202f6a64989b51c7024c04aAndreas Huber
2919318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        status_t err = mOMX->emptyBuffer(
2920693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mNode, info->mBuffer, 0, size,
2921693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                OMX_BUFFERFLAG_ENDOFFRAME | OMX_BUFFERFLAG_CODECCONFIG,
2922693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                0);
2923f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(err, (status_t)OK);
2924693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
292592bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        info->mStatus = OWNED_BY_COMPONENT;
2926693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2927693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        ++mCodecSpecificDataIndex;
292892bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        return true;
2929693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
2930693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
2931d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber    if (mPaused) {
293292bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        return false;
2933d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber    }
2934d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber
2935693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err;
2936134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber
29377f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    bool signalEOS = false;
29387f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    int64_t timestampUs = 0;
2939693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
29407f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    size_t offset = 0;
29417f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    int32_t n = 0;
294292bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
29430c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi
29447f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    for (;;) {
29457f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        MediaBuffer *srcBuffer;
29467f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        if (mSeekTimeUs >= 0) {
29477f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            if (mLeftOverBuffer) {
29487f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                mLeftOverBuffer->release();
29497f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                mLeftOverBuffer = NULL;
29507f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            }
295179e23b41fad961008bfde6e26b3c6f86878ca69dJames Dong
295279e23b41fad961008bfde6e26b3c6f86878ca69dJames Dong            MediaSource::ReadOptions options;
2953abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber            options.setSeekTo(mSeekTimeUs, mSeekMode);
2954693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
29557f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            mSeekTimeUs = -1;
2956abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber            mSeekMode = ReadOptions::SEEK_CLOSEST_SYNC;
29577f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            mBufferFilled.signal();
2958693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
29597f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            err = mSource->read(&srcBuffer, &options);
2960abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber
2961abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber            if (err == OK) {
2962abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                int64_t targetTimeUs;
2963abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                if (srcBuffer->meta_data()->findInt64(
2964abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                            kKeyTargetTime, &targetTimeUs)
2965abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                        && targetTimeUs >= 0) {
29669c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber                    CODEC_LOGV("targetTimeUs = %lld us", targetTimeUs);
2967abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                    mTargetTimeUs = targetTimeUs;
2968abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                } else {
2969abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                    mTargetTimeUs = -1;
2970abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber                }
2971abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber            }
29727f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        } else if (mLeftOverBuffer) {
29737f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            srcBuffer = mLeftOverBuffer;
29747f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            mLeftOverBuffer = NULL;
29757f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
29767f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            err = OK;
29777f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        } else {
297879e23b41fad961008bfde6e26b3c6f86878ca69dJames Dong            err = mSource->read(&srcBuffer);
29797f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        }
2980134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber
29817f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        if (err != OK) {
29827f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            signalEOS = true;
29837f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            mFinalStatus = err;
29847f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            mSignalledEOS = true;
2985763b772bb6bca6a4aaef85cec840fd7061630d61Andreas Huber            mBufferFilled.signal();
29867f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            break;
29877f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        }
2988693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
29890bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        if (mFlags & kUseSecureInputBuffers) {
29900bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            info = findInputBufferByDataPointer(srcBuffer->data());
29910bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            CHECK(info != NULL);
29920bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        }
29930bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
29947f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        size_t remainingBytes = info->mSize - offset;
2995c017cfcc24dd63c7efde5747a6ed2d4053788abcAndreas Huber
29967f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        if (srcBuffer->range_length() > remainingBytes) {
29977f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            if (offset == 0) {
29987f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                CODEC_LOGE(
29997f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                     "Codec's input buffers are too small to accomodate "
30007f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                     "buffer read from source (info->mSize = %d, srcLength = %d)",
30017f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                     info->mSize, srcBuffer->range_length());
3002c017cfcc24dd63c7efde5747a6ed2d4053788abcAndreas Huber
30037f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                srcBuffer->release();
30047f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                srcBuffer = NULL;
30057f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
30067f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber                setState(ERROR);
300792bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber                return false;
30087f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            }
30097f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
30107f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            mLeftOverBuffer = srcBuffer;
30117f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            break;
3012693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
30137f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
30148480835b4bc1350646376aa7f3ae33742a7adeb1James Dong        bool releaseBuffer = true;
3015d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong        if (mIsEncoder && (mQuirks & kAvoidMemcopyInputRecordingFrames)) {
3016d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong            CHECK(mOMXLivesLocally && offset == 0);
301792bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
301892bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            OMX_BUFFERHEADERTYPE *header =
301992bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber                (OMX_BUFFERHEADERTYPE *)info->mBuffer;
302092bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
3021f23c4f92c3b0202435cf87db2642156fabc46f02James Dong            CHECK(header->pBuffer == info->mData);
302292bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
302392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            header->pBuffer =
302492bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber                (OMX_U8 *)srcBuffer->data() + srcBuffer->range_offset();
302592bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
30268480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            releaseBuffer = false;
30278480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            info->mMediaBuffer = srcBuffer;
3028d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong        } else {
30290bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            if (mFlags & kStoreMetaDataInVideoBuffers) {
30308480835b4bc1350646376aa7f3ae33742a7adeb1James Dong                releaseBuffer = false;
30318480835b4bc1350646376aa7f3ae33742a7adeb1James Dong                info->mMediaBuffer = srcBuffer;
30328480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            }
30330bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
30340bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            if (mFlags & kUseSecureInputBuffers) {
30350bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                // Data in "info" is already provided at this time.
30360bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
30370bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                releaseBuffer = false;
30380bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
30390bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                CHECK(info->mMediaBuffer == NULL);
30400bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                info->mMediaBuffer = srcBuffer;
30410bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            } else {
30420c5c7d2b119d2350c186ae9902919bcf28c3e277Pannag Sanketi                CHECK(srcBuffer->data() != NULL) ;
30430bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                memcpy((uint8_t *)info->mData + offset,
30440bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                        (const uint8_t *)srcBuffer->data()
30450bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                            + srcBuffer->range_offset(),
30460bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber                        srcBuffer->range_length());
30470bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber            }
3048d07139e2e817a9b3ae9c87ba4e1e8d65d3e549daJames Dong        }
3049693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3050d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber        int64_t lastBufferTimeUs;
3051d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber        CHECK(srcBuffer->meta_data()->findInt64(kKeyTime, &lastBufferTimeUs));
3052abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber        CHECK(lastBufferTimeUs >= 0);
3053965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong        if (mIsEncoder) {
3054965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong            mDecodingTimeList.push_back(lastBufferTimeUs);
3055965e4239ca1cf6c824c1f8ce23116f9ba8cf6ebdJames Dong        }
3056d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber
30577f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        if (offset == 0) {
3058d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber            timestampUs = lastBufferTimeUs;
3059693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3060693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
30617f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        offset += srcBuffer->range_length();
30627f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
3063bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (!strcasecmp(MEDIA_MIMETYPE_AUDIO_VORBIS, mMIME)) {
3064bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            CHECK(!(mQuirks & kSupportsMultipleFramesPerInputBuffer));
3065bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            CHECK_GE(info->mSize, offset + sizeof(int32_t));
3066bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3067bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            int32_t numPageSamples;
3068bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (!srcBuffer->meta_data()->findInt32(
3069bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                        kKeyValidSamples, &numPageSamples)) {
3070bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                numPageSamples = -1;
3071bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
3072bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3073bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            memcpy((uint8_t *)info->mData + offset,
3074bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                   &numPageSamples,
3075bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                   sizeof(numPageSamples));
3076bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3077bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            offset += sizeof(numPageSamples);
3078bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
3079bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
30808480835b4bc1350646376aa7f3ae33742a7adeb1James Dong        if (releaseBuffer) {
30818480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            srcBuffer->release();
30828480835b4bc1350646376aa7f3ae33742a7adeb1James Dong            srcBuffer = NULL;
30838480835b4bc1350646376aa7f3ae33742a7adeb1James Dong        }
30847f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
30857f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        ++n;
30867f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
30877f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        if (!(mQuirks & kSupportsMultipleFramesPerInputBuffer)) {
30887f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            break;
30897f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        }
3090d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber
3091d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber        int64_t coalescedDurationUs = lastBufferTimeUs - timestampUs;
3092d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber
3093d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber        if (coalescedDurationUs > 250000ll) {
3094d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber            // Don't coalesce more than 250ms worth of encoded data at once.
3095d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber            break;
3096d2c6894b56a538aa807e20d3ef421807cd55c009Andreas Huber        }
3097693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
309836efa039efaae4526791336cb688032d22b34becAndreas Huber
30997f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    if (n > 1) {
31007f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        LOGV("coalesced %d frames into one input buffer", n);
31017f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    }
31027f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
31037f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    OMX_U32 flags = OMX_BUFFERFLAG_ENDOFFRAME;
31047f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
31057f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    if (signalEOS) {
31067f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        flags |= OMX_BUFFERFLAG_EOS;
31077f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    } else {
31087f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        mNoMoreOutputData = false;
31097f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    }
31107f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
31117f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    CODEC_LOGV("Calling emptyBuffer on buffer %p (length %d), "
31127f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber               "timestamp %lld us (%.2f secs)",
31137f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber               info->mBuffer, offset,
31147f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber               timestampUs, timestampUs / 1E6);
31157f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
31160bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    if (info == NULL) {
31170bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        CHECK(mFlags & kUseSecureInputBuffers);
31180bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        CHECK(signalEOS);
31190bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
31200bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        // This is fishy, there's still a MediaBuffer corresponding to this
31210bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        // info available to the source at this point even though we're going
31220bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        // to use it to signal EOS to the codec.
31230bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber        info = findEmptyInputBuffer();
31240bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber    }
31250bb6b4c5da2451ee0973372b0a3858b15c742689Andreas Huber
3126318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->emptyBuffer(
31277f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber            mNode, info->mBuffer, 0, offset,
312848c948b1137e7bbdb161b51908657ab72ac5e2daAndreas Huber            flags, timestampUs);
312936efa039efaae4526791336cb688032d22b34becAndreas Huber
313036efa039efaae4526791336cb688032d22b34becAndreas Huber    if (err != OK) {
313136efa039efaae4526791336cb688032d22b34becAndreas Huber        setState(ERROR);
313292bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber        return false;
313336efa039efaae4526791336cb688032d22b34becAndreas Huber    }
313436efa039efaae4526791336cb688032d22b34becAndreas Huber
313592bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    info->mStatus = OWNED_BY_COMPONENT;
313630ab66297501757d745b9ae10da61adcd891f497Andreas Huber
313730ab66297501757d745b9ae10da61adcd891f497Andreas Huber    // This component does not ever signal the EOS flag on output buffers,
313830ab66297501757d745b9ae10da61adcd891f497Andreas Huber    // Thanks for nothing.
313930ab66297501757d745b9ae10da61adcd891f497Andreas Huber    if (mSignalledEOS && !strcmp(mComponentName, "OMX.TI.Video.encoder")) {
314030ab66297501757d745b9ae10da61adcd891f497Andreas Huber        mNoMoreOutputData = true;
314130ab66297501757d745b9ae10da61adcd891f497Andreas Huber        mBufferFilled.signal();
314230ab66297501757d745b9ae10da61adcd891f497Andreas Huber    }
314392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
314492bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    return true;
3145693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3146693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3147693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::fillOutputBuffer(BufferInfo *info) {
314892bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    CHECK_EQ((int)info->mStatus, (int)OWNED_BY_US);
3149693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
31501beb760d920561679862ded945a04e370368c7f7Andreas Huber    if (mNoMoreOutputData) {
31517a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        CODEC_LOGV("There is no more output data available, not "
31521beb760d920561679862ded945a04e370368c7f7Andreas Huber             "calling fillOutputBuffer");
31531beb760d920561679862ded945a04e370368c7f7Andreas Huber        return;
31541beb760d920561679862ded945a04e370368c7f7Andreas Huber    }
31551beb760d920561679862ded945a04e370368c7f7Andreas Huber
3156f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    if (info->mMediaBuffer != NULL) {
3157f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        sp<GraphicBuffer> graphicBuffer = info->mMediaBuffer->graphicBuffer();
3158f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        if (graphicBuffer != 0) {
3159f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            // When using a native buffer we need to lock the buffer before
3160f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            // giving it to OMX.
3161f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CODEC_LOGV("Calling lockBuffer on %p", info->mBuffer);
3162f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            int err = mNativeWindow->lockBuffer(mNativeWindow.get(),
3163f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    graphicBuffer.get());
3164f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            if (err != 0) {
3165f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CODEC_LOGE("lockBuffer failed w/ error 0x%08x", err);
31666a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
3167f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                setState(ERROR);
3168f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                return;
3169f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            }
31706a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        }
31716a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
31726a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
31736a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    CODEC_LOGV("Calling fillBuffer on buffer %p", info->mBuffer);
3174318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->fillBuffer(mNode, info->mBuffer);
31757110385b31c7bc65c8a923a09689ebb0c4a1e07aAndreas Huber
31767110385b31c7bc65c8a923a09689ebb0c4a1e07aAndreas Huber    if (err != OK) {
31777110385b31c7bc65c8a923a09689ebb0c4a1e07aAndreas Huber        CODEC_LOGE("fillBuffer failed w/ error 0x%08x", err);
31787110385b31c7bc65c8a923a09689ebb0c4a1e07aAndreas Huber
31797110385b31c7bc65c8a923a09689ebb0c4a1e07aAndreas Huber        setState(ERROR);
31807110385b31c7bc65c8a923a09689ebb0c4a1e07aAndreas Huber        return;
31817110385b31c7bc65c8a923a09689ebb0c4a1e07aAndreas Huber    }
3182693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
318392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    info->mStatus = OWNED_BY_COMPONENT;
3184693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3185693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
318692bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huberbool OMXCodec::drainInputBuffer(IOMX::buffer_id buffer) {
3187693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexInput];
3188693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (size_t i = 0; i < buffers->size(); ++i) {
3189693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if ((*buffers)[i].mBuffer == buffer) {
319092bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            return drainInputBuffer(&buffers->editItemAt(i));
3191693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3192693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3193693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3194693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CHECK(!"should not be here.");
319592bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
319692bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    return false;
3197693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3198693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3199693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::fillOutputBuffer(IOMX::buffer_id buffer) {
3200693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexOutput];
3201693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (size_t i = 0; i < buffers->size(); ++i) {
3202693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if ((*buffers)[i].mBuffer == buffer) {
3203693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            fillOutputBuffer(&buffers->editItemAt(i));
3204693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            return;
3205693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3206693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3207693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3208693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CHECK(!"should not be here.");
3209693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3210693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3211693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::setState(State newState) {
3212693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mState = newState;
3213693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mAsyncCompletion.signal();
3214693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3215693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    // This may cause some spurious wakeups but is necessary to
3216693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    // unblock the reader if we enter ERROR state.
3217693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mBufferFilled.signal();
3218693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3219693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3220a57a9a491272aa884494b2ec7854960827a73742James Dongstatus_t OMXCodec::waitForBufferFilled_l() {
322164b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong
322264b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong    if (mIsEncoder) {
322364b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong        // For timelapse video recording, the timelapse video recording may
322464b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong        // not send an input frame for a _long_ time. Do not use timeout
322564b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong        // for video encoding.
322664b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong        return mBufferFilled.wait(mLock);
322764b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong    }
32288edb8e82fa886564ee8e72178a1969e2437dd525James Dong    status_t err = mBufferFilled.waitRelative(mLock, kBufferFilledEventTimeOutNs);
3229a57a9a491272aa884494b2ec7854960827a73742James Dong    if (err != OK) {
323064b944106c697933a453a1ffd8bcddb137fdbc4cJames Dong        CODEC_LOGE("Timed out waiting for output buffers: %d/%d",
3231a57a9a491272aa884494b2ec7854960827a73742James Dong            countBuffersWeOwn(mPortBuffers[kPortIndexInput]),
3232a57a9a491272aa884494b2ec7854960827a73742James Dong            countBuffersWeOwn(mPortBuffers[kPortIndexOutput]));
3233a57a9a491272aa884494b2ec7854960827a73742James Dong    }
3234a57a9a491272aa884494b2ec7854960827a73742James Dong    return err;
3235a57a9a491272aa884494b2ec7854960827a73742James Dong}
3236a57a9a491272aa884494b2ec7854960827a73742James Dong
3237e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Hubervoid OMXCodec::setRawAudioFormat(
3238e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        OMX_U32 portIndex, int32_t sampleRate, int32_t numChannels) {
3239050b28a593350047845a45a14cc5026221ac1620James Dong
3240050b28a593350047845a45a14cc5026221ac1620James Dong    // port definition
3241050b28a593350047845a45a14cc5026221ac1620James Dong    OMX_PARAM_PORTDEFINITIONTYPE def;
3242050b28a593350047845a45a14cc5026221ac1620James Dong    InitOMXParams(&def);
3243050b28a593350047845a45a14cc5026221ac1620James Dong    def.nPortIndex = portIndex;
3244050b28a593350047845a45a14cc5026221ac1620James Dong    status_t err = mOMX->getParameter(
3245050b28a593350047845a45a14cc5026221ac1620James Dong            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3246f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3247050b28a593350047845a45a14cc5026221ac1620James Dong    def.format.audio.eEncoding = OMX_AUDIO_CodingPCM;
3248050b28a593350047845a45a14cc5026221ac1620James Dong    CHECK_EQ(mOMX->setParameter(mNode, OMX_IndexParamPortDefinition,
3249f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            &def, sizeof(def)), (status_t)OK);
3250050b28a593350047845a45a14cc5026221ac1620James Dong
3251050b28a593350047845a45a14cc5026221ac1620James Dong    // pcm param
3252e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    OMX_AUDIO_PARAM_PCMMODETYPE pcmParams;
32537a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&pcmParams);
3254e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    pcmParams.nPortIndex = portIndex;
3255e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3256050b28a593350047845a45a14cc5026221ac1620James Dong    err = mOMX->getParameter(
3257e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            mNode, OMX_IndexParamAudioPcm, &pcmParams, sizeof(pcmParams));
3258e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3259f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3260e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3261e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    pcmParams.nChannels = numChannels;
3262e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    pcmParams.eNumData = OMX_NumericalDataSigned;
3263e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    pcmParams.bInterleaved = OMX_TRUE;
3264e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    pcmParams.nBitPerSample = 16;
3265e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    pcmParams.nSamplingRate = sampleRate;
3266e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    pcmParams.ePCMMode = OMX_AUDIO_PCMModeLinear;
3267e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3268e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    if (numChannels == 1) {
3269e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        pcmParams.eChannelMapping[0] = OMX_AUDIO_ChannelCF;
3270e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    } else {
3271e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        CHECK_EQ(numChannels, 2);
3272e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3273e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        pcmParams.eChannelMapping[0] = OMX_AUDIO_ChannelLF;
3274e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        pcmParams.eChannelMapping[1] = OMX_AUDIO_ChannelRF;
3275e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    }
3276e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3277318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->setParameter(
3278e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            mNode, OMX_IndexParamAudioPcm, &pcmParams, sizeof(pcmParams));
3279e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3280f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3281e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber}
3282e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber
3283956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dongstatic OMX_AUDIO_AMRBANDMODETYPE pickModeFromBitRate(bool isAMRWB, int32_t bps) {
3284956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    if (isAMRWB) {
3285956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        if (bps <= 6600) {
3286956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB0;
3287956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 8850) {
3288956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB1;
3289956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 12650) {
3290956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB2;
3291956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 14250) {
3292956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB3;
3293956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 15850) {
3294956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB4;
3295956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 18250) {
3296956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB5;
3297956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 19850) {
3298956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB6;
3299956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 23050) {
3300956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeWB7;
3301956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        }
3302956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong
3303956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        // 23850 bps
3304956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        return OMX_AUDIO_AMRBandModeWB8;
3305956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    } else {  // AMRNB
3306956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        if (bps <= 4750) {
3307956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeNB0;
3308956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 5150) {
3309956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeNB1;
3310956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 5900) {
3311956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeNB2;
3312956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 6700) {
3313956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeNB3;
3314956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 7400) {
3315956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeNB4;
3316956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 7950) {
3317956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeNB5;
3318956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        } else if (bps <= 10200) {
3319956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong            return OMX_AUDIO_AMRBandModeNB6;
3320956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        }
3321956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong
3322956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        // 12200 bps
3323956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        return OMX_AUDIO_AMRBandModeNB7;
3324956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    }
3325956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong}
3326956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong
3327956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dongvoid OMXCodec::setAMRFormat(bool isWAMR, int32_t bitRate) {
3328c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber    OMX_U32 portIndex = mIsEncoder ? kPortIndexOutput : kPortIndexInput;
3329693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3330c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber    OMX_AUDIO_PARAM_AMRTYPE def;
3331c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber    InitOMXParams(&def);
3332c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber    def.nPortIndex = portIndex;
3333456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber
3334c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber    status_t err =
3335c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber        mOMX->getParameter(mNode, OMX_IndexParamAudioAmr, &def, sizeof(def));
3336456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber
3337f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3338456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber
3339c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber    def.eAMRFrameFormat = OMX_AUDIO_AMRFrameFormatFSF;
3340050b28a593350047845a45a14cc5026221ac1620James Dong
3341956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong    def.eAMRBandMode = pickModeFromBitRate(isWAMR, bitRate);
3342c297fccffc4ab1cb3b9f5c6a5b0802be057f3e0fAndreas Huber    err = mOMX->setParameter(mNode, OMX_IndexParamAudioAmr, &def, sizeof(def));
3343f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3344456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber
3345456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber    ////////////////////////
3346456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber
3347456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber    if (mIsEncoder) {
3348456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber        sp<MetaData> format = mSource->getFormat();
3349456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber        int32_t sampleRate;
3350456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber        int32_t numChannels;
3351456db75516efc889e1ee4e5e16021e77c03b0941Andreas Huber        CHECK(format->findInt32(kKeySampleRate, &sampleRate));
3352693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        CHECK(format->findInt32(kKeyChannelCount, &numChannels));
3353693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3354e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        setRawAudioFormat(kPortIndexInput, sampleRate, numChannels);
3355e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    }
3356e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber}
3357693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3358956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dongvoid OMXCodec::setAACFormat(int32_t numChannels, int32_t sampleRate, int32_t bitRate) {
3359050b28a593350047845a45a14cc5026221ac1620James Dong    CHECK(numChannels == 1 || numChannels == 2);
3360e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    if (mIsEncoder) {
3361050b28a593350047845a45a14cc5026221ac1620James Dong        //////////////// input port ////////////////////
3362e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        setRawAudioFormat(kPortIndexInput, sampleRate, numChannels);
3363050b28a593350047845a45a14cc5026221ac1620James Dong
3364050b28a593350047845a45a14cc5026221ac1620James Dong        //////////////// output port ////////////////////
3365050b28a593350047845a45a14cc5026221ac1620James Dong        // format
3366050b28a593350047845a45a14cc5026221ac1620James Dong        OMX_AUDIO_PARAM_PORTFORMATTYPE format;
3367050b28a593350047845a45a14cc5026221ac1620James Dong        format.nPortIndex = kPortIndexOutput;
3368050b28a593350047845a45a14cc5026221ac1620James Dong        format.nIndex = 0;
3369050b28a593350047845a45a14cc5026221ac1620James Dong        status_t err = OMX_ErrorNone;
3370050b28a593350047845a45a14cc5026221ac1620James Dong        while (OMX_ErrorNone == err) {
3371050b28a593350047845a45a14cc5026221ac1620James Dong            CHECK_EQ(mOMX->getParameter(mNode, OMX_IndexParamAudioPortFormat,
3372f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                    &format, sizeof(format)), (status_t)OK);
3373050b28a593350047845a45a14cc5026221ac1620James Dong            if (format.eEncoding == OMX_AUDIO_CodingAAC) {
3374050b28a593350047845a45a14cc5026221ac1620James Dong                break;
3375050b28a593350047845a45a14cc5026221ac1620James Dong            }
3376050b28a593350047845a45a14cc5026221ac1620James Dong            format.nIndex++;
3377050b28a593350047845a45a14cc5026221ac1620James Dong        }
3378f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ((status_t)OK, err);
3379050b28a593350047845a45a14cc5026221ac1620James Dong        CHECK_EQ(mOMX->setParameter(mNode, OMX_IndexParamAudioPortFormat,
3380f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                &format, sizeof(format)), (status_t)OK);
3381050b28a593350047845a45a14cc5026221ac1620James Dong
3382050b28a593350047845a45a14cc5026221ac1620James Dong        // port definition
3383050b28a593350047845a45a14cc5026221ac1620James Dong        OMX_PARAM_PORTDEFINITIONTYPE def;
3384050b28a593350047845a45a14cc5026221ac1620James Dong        InitOMXParams(&def);
3385050b28a593350047845a45a14cc5026221ac1620James Dong        def.nPortIndex = kPortIndexOutput;
3386050b28a593350047845a45a14cc5026221ac1620James Dong        CHECK_EQ(mOMX->getParameter(mNode, OMX_IndexParamPortDefinition,
3387f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                &def, sizeof(def)), (status_t)OK);
3388050b28a593350047845a45a14cc5026221ac1620James Dong        def.format.audio.bFlagErrorConcealment = OMX_TRUE;
3389050b28a593350047845a45a14cc5026221ac1620James Dong        def.format.audio.eEncoding = OMX_AUDIO_CodingAAC;
3390050b28a593350047845a45a14cc5026221ac1620James Dong        CHECK_EQ(mOMX->setParameter(mNode, OMX_IndexParamPortDefinition,
3391f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                &def, sizeof(def)), (status_t)OK);
3392050b28a593350047845a45a14cc5026221ac1620James Dong
3393050b28a593350047845a45a14cc5026221ac1620James Dong        // profile
3394050b28a593350047845a45a14cc5026221ac1620James Dong        OMX_AUDIO_PARAM_AACPROFILETYPE profile;
3395050b28a593350047845a45a14cc5026221ac1620James Dong        InitOMXParams(&profile);
3396050b28a593350047845a45a14cc5026221ac1620James Dong        profile.nPortIndex = kPortIndexOutput;
3397050b28a593350047845a45a14cc5026221ac1620James Dong        CHECK_EQ(mOMX->getParameter(mNode, OMX_IndexParamAudioAac,
3398f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                &profile, sizeof(profile)), (status_t)OK);
3399050b28a593350047845a45a14cc5026221ac1620James Dong        profile.nChannels = numChannels;
3400050b28a593350047845a45a14cc5026221ac1620James Dong        profile.eChannelMode = (numChannels == 1?
3401050b28a593350047845a45a14cc5026221ac1620James Dong                OMX_AUDIO_ChannelModeMono: OMX_AUDIO_ChannelModeStereo);
3402050b28a593350047845a45a14cc5026221ac1620James Dong        profile.nSampleRate = sampleRate;
3403956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong        profile.nBitRate = bitRate;
3404050b28a593350047845a45a14cc5026221ac1620James Dong        profile.nAudioBandWidth = 0;
3405050b28a593350047845a45a14cc5026221ac1620James Dong        profile.nFrameLength = 0;
3406050b28a593350047845a45a14cc5026221ac1620James Dong        profile.nAACtools = OMX_AUDIO_AACToolAll;
3407050b28a593350047845a45a14cc5026221ac1620James Dong        profile.nAACERtools = OMX_AUDIO_AACERNone;
3408050b28a593350047845a45a14cc5026221ac1620James Dong        profile.eAACProfile = OMX_AUDIO_AACObjectLC;
3409050b28a593350047845a45a14cc5026221ac1620James Dong        profile.eAACStreamFormat = OMX_AUDIO_AACStreamFormatMP4FF;
3410050b28a593350047845a45a14cc5026221ac1620James Dong        CHECK_EQ(mOMX->setParameter(mNode, OMX_IndexParamAudioAac,
3411f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                &profile, sizeof(profile)), (status_t)OK);
3412050b28a593350047845a45a14cc5026221ac1620James Dong
3413e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber    } else {
3414e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        OMX_AUDIO_PARAM_AACPROFILETYPE profile;
34157a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        InitOMXParams(&profile);
3416e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        profile.nPortIndex = kPortIndexInput;
3417693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3418318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        status_t err = mOMX->getParameter(
3419e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
3420f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(err, (status_t)OK);
3421693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3422e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        profile.nChannels = numChannels;
3423e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        profile.nSampleRate = sampleRate;
3424e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber        profile.eAACStreamFormat = OMX_AUDIO_AACStreamFormatMP4ADTS;
3425693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3426318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        err = mOMX->setParameter(
3427e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
3428f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(err, (status_t)OK);
3429693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3430693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3431693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3432bbba88cb1bdc34705d1477208990a06904c022e7Andreas Hubervoid OMXCodec::setG711Format(int32_t numChannels) {
3433bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    CHECK(!mIsEncoder);
3434bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    setRawAudioFormat(kPortIndexInput, 8000, numChannels);
3435bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
3436bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3437693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::setImageOutputFormat(
3438693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        OMX_COLOR_FORMATTYPE format, OMX_U32 width, OMX_U32 height) {
34397a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    CODEC_LOGV("setImageOutputFormat(%ld, %ld)", width, height);
3440693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3441693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#if 0
3442693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_INDEXTYPE index;
3443693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err = mOMX->get_extension_index(
3444693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, "OMX.TI.JPEG.decode.Config.OutputColorFormat", &index);
3445f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3446693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3447693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    err = mOMX->set_config(mNode, index, &format, sizeof(format));
3448f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3449693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber#endif
3450693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3451693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
34527a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
3453693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = kPortIndexOutput;
3454693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3455318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->getParameter(
3456693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3457f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3458693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3459f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainImage);
3460693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3461693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_IMAGE_PORTDEFINITIONTYPE *imageDef = &def.format.image;
3462bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
3463f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)imageDef->eCompressionFormat, (int)OMX_IMAGE_CodingUnused);
3464693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    imageDef->eColorFormat = format;
3465693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    imageDef->nFrameWidth = width;
3466693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    imageDef->nFrameHeight = height;
3467693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3468693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    switch (format) {
3469693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_COLOR_FormatYUV420PackedPlanar:
3470693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_COLOR_FormatYUV411Planar:
3471693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
3472693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            def.nBufferSize = (width * height * 3) / 2;
3473693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
3474693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3475693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3476693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_COLOR_FormatCbYCrY:
3477693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
3478693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            def.nBufferSize = width * height * 2;
3479693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
3480693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3481693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3482693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_COLOR_Format32bitARGB8888:
3483693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
3484693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            def.nBufferSize = width * height * 4;
3485693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
3486693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3487693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3488195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber        case OMX_COLOR_Format16bitARGB4444:
3489195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber        case OMX_COLOR_Format16bitARGB1555:
3490195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber        case OMX_COLOR_Format16bitRGB565:
3491195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber        case OMX_COLOR_Format16bitBGR565:
3492195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber        {
3493195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber            def.nBufferSize = width * height * 2;
3494195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber            break;
3495195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber        }
3496195ea1e0d4484f1deb2309e239f6a5cc9a7e5ba7Andreas Huber
3497693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        default:
3498693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(!"Should not be here. Unknown color format.");
3499693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
3500693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3501693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
35027530e9c708275c273c134c36c68179f511c1940eAndreas Huber    def.nBufferCountActual = def.nBufferCountMin;
35037530e9c708275c273c134c36c68179f511c1940eAndreas Huber
3504318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->setParameter(
3505693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3506f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
35077530e9c708275c273c134c36c68179f511c1940eAndreas Huber}
3508693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
35097530e9c708275c273c134c36c68179f511c1940eAndreas Hubervoid OMXCodec::setJPEGInputFormat(
35107530e9c708275c273c134c36c68179f511c1940eAndreas Huber        OMX_U32 width, OMX_U32 height, OMX_U32 compressedSize) {
35117530e9c708275c273c134c36c68179f511c1940eAndreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
35127a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
3513693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = kPortIndexInput;
3514693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3515318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->getParameter(
3516693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3517f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3518693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3519f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainImage);
35207530e9c708275c273c134c36c68179f511c1940eAndreas Huber    OMX_IMAGE_PORTDEFINITIONTYPE *imageDef = &def.format.image;
35217530e9c708275c273c134c36c68179f511c1940eAndreas Huber
3522f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ((int)imageDef->eCompressionFormat, (int)OMX_IMAGE_CodingJPEG);
3523693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    imageDef->nFrameWidth = width;
3524693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    imageDef->nFrameHeight = height;
3525693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
35267530e9c708275c273c134c36c68179f511c1940eAndreas Huber    def.nBufferSize = compressedSize;
3527693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nBufferCountActual = def.nBufferCountMin;
3528693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3529318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    err = mOMX->setParameter(
3530693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3531f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
3532693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3533693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3534693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::addCodecSpecificData(const void *data, size_t size) {
3535693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CodecSpecificData *specific =
3536693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        (CodecSpecificData *)malloc(sizeof(CodecSpecificData) + size - 1);
3537693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3538693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    specific->mSize = size;
3539693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    memcpy(specific->mData, data, size);
3540693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3541693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mCodecSpecificData.push(specific);
3542693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3543693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3544693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::clearCodecSpecificData() {
3545693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (size_t i = 0; i < mCodecSpecificData.size(); ++i) {
3546693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        free(mCodecSpecificData.editItemAt(i));
3547693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3548693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mCodecSpecificData.clear();
3549693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mCodecSpecificDataIndex = 0;
3550693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3551693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3552f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dongstatus_t OMXCodec::start(MetaData *meta) {
3553284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber    Mutex::Autolock autoLock(mLock);
3554284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber
3555693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (mState != LOADED) {
3556693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return UNKNOWN_ERROR;
3557693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3558bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber
3559693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    sp<MetaData> params = new MetaData;
3560dbc03445db2bbf83b64f0c0a5dc62e61408864d7Andreas Huber    if (mQuirks & kWantsNALFragments) {
3561dbc03445db2bbf83b64f0c0a5dc62e61408864d7Andreas Huber        params->setInt32(kKeyWantsNALFragments, true);
3562693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3563f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong    if (meta) {
3564f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong        int64_t startTimeUs = 0;
3565f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong        int64_t timeUs;
3566f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong        if (meta->findInt64(kKeyTime, &timeUs)) {
3567f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong            startTimeUs = timeUs;
3568f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong        }
3569f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong        params->setInt64(kKeyTime, startTimeUs);
3570f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong    }
3571693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    status_t err = mSource->start(params.get());
3572693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3573693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (err != OK) {
3574693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return err;
3575693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3576693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3577693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mCodecSpecificDataIndex = 0;
3578284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber    mInitialBufferSubmit = true;
3579693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mSignalledEOS = false;
3580693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mNoMoreOutputData = false;
35817f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber    mOutputPortSettingsHaveChanged = false;
3582693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mSeekTimeUs = -1;
3583abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber    mSeekMode = ReadOptions::SEEK_CLOSEST_SYNC;
3584abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber    mTargetTimeUs = -1;
3585693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mFilledBuffers.clear();
3586d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber    mPaused = false;
3587693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3588693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return init();
3589693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3590693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3591693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatus_t OMXCodec::stop() {
3592a3f4d7f096e5c91dc6af085761b1459866c043d9James Dong    CODEC_LOGV("stop mState=%d", mState);
3593693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3594693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Mutex::Autolock autoLock(mLock);
3595693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3596693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    while (isIntermediateState(mState)) {
3597693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        mAsyncCompletion.wait(mLock);
3598693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3599693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3600693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    switch (mState) {
3601693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case LOADED:
3602693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case ERROR:
3603693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
3604693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3605693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case EXECUTING:
3606693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
3607693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            setState(EXECUTING_TO_IDLE);
3608693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
36098297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber            if (mQuirks & kRequiresFlushBeforeShutdown) {
36107a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                CODEC_LOGV("This component requires a flush before transitioning "
36118297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                     "from EXECUTING to IDLE...");
36128297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber
36138297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                bool emulateInputFlushCompletion =
36148297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    !flushPortAsync(kPortIndexInput);
36158297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber
36168297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                bool emulateOutputFlushCompletion =
36178297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    !flushPortAsync(kPortIndexOutput);
3618693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
36198297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                if (emulateInputFlushCompletion) {
36208297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    onCmdComplete(OMX_CommandFlush, kPortIndexInput);
36218297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                }
36228297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber
36238297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                if (emulateOutputFlushCompletion) {
36248297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                    onCmdComplete(OMX_CommandFlush, kPortIndexOutput);
36258297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                }
36268297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber            } else {
36278297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                mPortStatus[kPortIndexInput] = SHUTTING_DOWN;
36288297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                mPortStatus[kPortIndexOutput] = SHUTTING_DOWN;
36298297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber
36308297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber                status_t err =
3631318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                    mOMX->sendCommand(mNode, OMX_CommandStateSet, OMX_StateIdle);
3632f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
36338297cedd12f689167d74de5ddacb514e9f6896eaAndreas Huber            }
3634693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3635693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            while (mState != LOADED && mState != ERROR) {
3636693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                mAsyncCompletion.wait(mLock);
3637693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
3638693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3639693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
3640693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3641693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3642693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        default:
3643693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
3644693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(!"should not be here.");
3645693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
3646693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3647693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3648693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
36497f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    if (mLeftOverBuffer) {
36507f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        mLeftOverBuffer->release();
36517f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber        mLeftOverBuffer = NULL;
36527f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber    }
36537f498b90a4300ef9badf14d202b0a67c26e20931Andreas Huber
3654693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mSource->stop();
3655693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
365643e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber    CODEC_LOGI("stopped in state %d", mState);
3657ec9dd59902c8beea4ba6a842f3a843d46150d949Andreas Huber
3658693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return OK;
3659693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3660693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3661693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubersp<MetaData> OMXCodec::getFormat() {
36627f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber    Mutex::Autolock autoLock(mLock);
36637f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber
3664693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return mOutputFormat;
3665693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3666693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3667693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatus_t OMXCodec::read(
3668693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        MediaBuffer **buffer, const ReadOptions *options) {
3669a57a9a491272aa884494b2ec7854960827a73742James Dong    status_t err = OK;
3670693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    *buffer = NULL;
3671693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3672693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Mutex::Autolock autoLock(mLock);
3673693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3674450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber    if (mState != EXECUTING && mState != RECONFIGURING) {
3675450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber        return UNKNOWN_ERROR;
3676450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber    }
3677450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber
36787e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber    bool seeking = false;
36797e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber    int64_t seekTimeUs;
3680abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber    ReadOptions::SeekMode seekMode;
3681abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber    if (options && options->getSeekTo(&seekTimeUs, &seekMode)) {
36827e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber        seeking = true;
36837e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber    }
36847e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber
3685284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber    if (mInitialBufferSubmit) {
3686284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber        mInitialBufferSubmit = false;
3687284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber
36887e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber        if (seeking) {
36897e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber            CHECK(seekTimeUs >= 0);
36907e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber            mSeekTimeUs = seekTimeUs;
3691abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber            mSeekMode = seekMode;
36927e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber
36937e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber            // There's no reason to trigger the code below, there's
36947e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber            // nothing to flush yet.
36957e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber            seeking = false;
3696d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber            mPaused = false;
36977e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber        }
36987e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber
3699284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber        drainInputBuffers();
3700284f513a4ce2c5aa0e250c07c873731aedb0be26Andreas Huber
3701450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber        if (mState == EXECUTING) {
3702450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber            // Otherwise mState == RECONFIGURING and this code will trigger
3703450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber            // after the output port is reenabled.
3704450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber            fillOutputBuffers();
3705450bf4b0d25c933431b790d911a5fcae750fe38dAndreas Huber        }
3706693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3707693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
37087e04dcf8d6784dd56f53aa90bf34431ab4f0710cAndreas Huber    if (seeking) {
37099c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber        while (mState == RECONFIGURING) {
3710a57a9a491272aa884494b2ec7854960827a73742James Dong            if ((err = waitForBufferFilled_l()) != OK) {
3711a57a9a491272aa884494b2ec7854960827a73742James Dong                return err;
3712a57a9a491272aa884494b2ec7854960827a73742James Dong            }
37139c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber        }
37149c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber
37159c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber        if (mState != EXECUTING) {
37169c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber            return UNKNOWN_ERROR;
37179c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber        }
37189c0096378820e5a61db26e52a7e6df50ba9c872dAndreas Huber
37197a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber        CODEC_LOGV("seeking to %lld us (%.2f secs)", seekTimeUs, seekTimeUs / 1E6);
3720693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3721693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        mSignalledEOS = false;
3722693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3723693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        CHECK(seekTimeUs >= 0);
3724693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        mSeekTimeUs = seekTimeUs;
3725abd1f4f870925d6776dbe4b930b759a1ab6595caAndreas Huber        mSeekMode = seekMode;
3726693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3727693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        mFilledBuffers.clear();
3728693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3729f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ((int)mState, (int)EXECUTING);
3730693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
37311beb760d920561679862ded945a04e370368c7f7Andreas Huber        bool emulateInputFlushCompletion = !flushPortAsync(kPortIndexInput);
37321beb760d920561679862ded945a04e370368c7f7Andreas Huber        bool emulateOutputFlushCompletion = !flushPortAsync(kPortIndexOutput);
37331beb760d920561679862ded945a04e370368c7f7Andreas Huber
37341beb760d920561679862ded945a04e370368c7f7Andreas Huber        if (emulateInputFlushCompletion) {
37351beb760d920561679862ded945a04e370368c7f7Andreas Huber            onCmdComplete(OMX_CommandFlush, kPortIndexInput);
37361beb760d920561679862ded945a04e370368c7f7Andreas Huber        }
37371beb760d920561679862ded945a04e370368c7f7Andreas Huber
37381beb760d920561679862ded945a04e370368c7f7Andreas Huber        if (emulateOutputFlushCompletion) {
37391beb760d920561679862ded945a04e370368c7f7Andreas Huber            onCmdComplete(OMX_CommandFlush, kPortIndexOutput);
37401beb760d920561679862ded945a04e370368c7f7Andreas Huber        }
3741134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber
3742134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber        while (mSeekTimeUs >= 0) {
3743a57a9a491272aa884494b2ec7854960827a73742James Dong            if ((err = waitForBufferFilled_l()) != OK) {
3744a57a9a491272aa884494b2ec7854960827a73742James Dong                return err;
3745a57a9a491272aa884494b2ec7854960827a73742James Dong            }
3746134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber        }
3747693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3748693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3749693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    while (mState != ERROR && !mNoMoreOutputData && mFilledBuffers.empty()) {
3750a57a9a491272aa884494b2ec7854960827a73742James Dong        if ((err = waitForBufferFilled_l()) != OK) {
3751a57a9a491272aa884494b2ec7854960827a73742James Dong            return err;
375241152efd144ccf70c380d5c9a32105c02a039f43James Dong        }
3753693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3754693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3755693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (mState == ERROR) {
3756693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return UNKNOWN_ERROR;
3757693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3758693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3759693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (mFilledBuffers.empty()) {
37605295c0c55d41a2906ea7f65a3f22e6278cb17d4bAndreas Huber        return mSignalledEOS ? mFinalStatus : ERROR_END_OF_STREAM;
3761693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3762693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
37637f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber    if (mOutputPortSettingsHaveChanged) {
37647f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber        mOutputPortSettingsHaveChanged = false;
37657f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber
37667f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber        return INFO_FORMAT_CHANGED;
37677f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber    }
37687f281f87670c63775ea1ae8b24af31822d5ad416Andreas Huber
3769693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    size_t index = *mFilledBuffers.begin();
3770693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mFilledBuffers.erase(mFilledBuffers.begin());
3771693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3772693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    BufferInfo *info = &mPortBuffers[kPortIndexOutput].editItemAt(index);
377392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    CHECK_EQ((int)info->mStatus, (int)OWNED_BY_US);
377492bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber    info->mStatus = OWNED_BY_CLIENT;
377592bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
3776693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    info->mMediaBuffer->add_ref();
3777693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    *buffer = info->mMediaBuffer;
3778693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3779693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    return OK;
3780693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3781693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3782693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::signalBufferReturned(MediaBuffer *buffer) {
3783693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Mutex::Autolock autoLock(mLock);
3784693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3785693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    Vector<BufferInfo> *buffers = &mPortBuffers[kPortIndexOutput];
3786693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    for (size_t i = 0; i < buffers->size(); ++i) {
3787693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        BufferInfo *info = &buffers->editItemAt(i);
3788693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3789693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        if (info->mMediaBuffer == buffer) {
3790f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((int)mPortStatus[kPortIndexOutput], (int)ENABLED);
379192bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            CHECK_EQ((int)info->mStatus, (int)OWNED_BY_CLIENT);
379292bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
379392bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber            info->mStatus = OWNED_BY_US;
379492bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
37956a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            if (buffer->graphicBuffer() == 0) {
37966a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                fillOutputBuffer(info);
37976a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            } else {
37986a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                sp<MetaData> metaData = info->mMediaBuffer->meta_data();
37996a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                int32_t rendered = 0;
38006a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                if (!metaData->findInt32(kKeyRendered, &rendered)) {
38016a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                    rendered = 0;
38026a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                }
38036a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                if (!rendered) {
38046a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                    status_t err = cancelBufferToNativeWindow(info);
38056a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                    if (err < 0) {
38066a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                        return;
38076a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                    }
38086a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                }
38096a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
381092bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber                info->mStatus = OWNED_BY_NATIVE_WINDOW;
381192bf2f96c53d24adc1ace362439e82ca2cf6b856Andreas Huber
38126a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                // Dequeue the next buffer from the native window.
38136a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                BufferInfo *nextBufInfo = dequeueBufferFromNativeWindow();
38146a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                if (nextBufInfo == 0) {
38156a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                    return;
38166a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                }
38176a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
38186a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                // Give the buffer to the OMX node to fill.
38196a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis                fillOutputBuffer(nextBufInfo);
38206a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis            }
3821693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            return;
3822693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
3823693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3824693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3825693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CHECK(!"should not be here.");
3826693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3827693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3828693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const char *imageCompressionFormatString(OMX_IMAGE_CODINGTYPE type) {
3829693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    static const char *kNames[] = {
3830693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingUnused",
3831693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingAutoDetect",
3832693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingJPEG",
3833693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingJPEG2K",
3834693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingEXIF",
3835693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingTIFF",
3836693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingGIF",
3837693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingPNG",
3838693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingLZW",
3839693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_IMAGE_CodingBMP",
3840693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    };
3841693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3842693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    size_t numNames = sizeof(kNames) / sizeof(kNames[0]);
3843693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3844693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (type < 0 || (size_t)type >= numNames) {
3845693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return "UNKNOWN";
3846693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
3847693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return kNames[type];
3848693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3849693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3850693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3851693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const char *colorFormatString(OMX_COLOR_FORMATTYPE type) {
3852693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    static const char *kNames[] = {
3853693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatUnused",
3854693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatMonochrome",
3855693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format8bitRGB332",
3856693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format12bitRGB444",
3857693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format16bitARGB4444",
3858693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format16bitARGB1555",
3859693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format16bitRGB565",
3860693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format16bitBGR565",
3861693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format18bitRGB666",
3862693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format18bitARGB1665",
3863bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber        "OMX_COLOR_Format19bitARGB1666",
3864693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format24bitRGB888",
3865693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format24bitBGR888",
3866693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format24bitARGB1887",
3867693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format25bitARGB1888",
3868693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format32bitBGRA8888",
3869693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format32bitARGB8888",
3870693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV411Planar",
3871693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV411PackedPlanar",
3872693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV420Planar",
3873693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV420PackedPlanar",
3874693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV420SemiPlanar",
3875693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV422Planar",
3876693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV422PackedPlanar",
3877693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV422SemiPlanar",
3878693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYCbYCr",
3879693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYCrYCb",
3880693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatCbYCrY",
3881693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatCrYCbY",
3882693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV444Interleaved",
3883693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatRawBayer8bit",
3884693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatRawBayer10bit",
3885693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatRawBayer8bitcompressed",
3886bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber        "OMX_COLOR_FormatL2",
3887bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber        "OMX_COLOR_FormatL4",
3888bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber        "OMX_COLOR_FormatL8",
3889bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber        "OMX_COLOR_FormatL16",
3890bde3caae211e215e4bbfef1a267f8d680efa4764Andreas Huber        "OMX_COLOR_FormatL24",
3891693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatL32",
3892693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV420PackedSemiPlanar",
3893693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_FormatYUV422PackedSemiPlanar",
3894693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format18BitBGR666",
3895693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format24BitARGB6666",
3896693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_COLOR_Format24BitABGR6666",
3897693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    };
3898693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3899693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    size_t numNames = sizeof(kNames) / sizeof(kNames[0]);
3900693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3901cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan    if (type == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar) {
3902cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan        return "OMX_TI_COLOR_FormatYUV420PackedSemiPlanar";
3903cb62bc3fe54222cf05824e6f98fefafee552049aAnu Sundararajan    } else if (type == OMX_QCOM_COLOR_FormatYVU420SemiPlanar) {
3904693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return "OMX_QCOM_COLOR_FormatYVU420SemiPlanar";
3905693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else if (type < 0 || (size_t)type >= numNames) {
3906693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return "UNKNOWN";
3907693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
3908693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return kNames[type];
3909693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3910693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3911693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3912693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const char *videoCompressionFormatString(OMX_VIDEO_CODINGTYPE type) {
3913693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    static const char *kNames[] = {
3914693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingUnused",
3915693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingAutoDetect",
3916693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingMPEG2",
3917693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingH263",
3918693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingMPEG4",
3919693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingWMV",
3920693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingRV",
3921693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingAVC",
3922693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_VIDEO_CodingMJPEG",
3923693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    };
3924693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3925693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    size_t numNames = sizeof(kNames) / sizeof(kNames[0]);
3926693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3927693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (type < 0 || (size_t)type >= numNames) {
3928693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return "UNKNOWN";
3929693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
3930693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return kNames[type];
3931693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3932693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3933693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3934693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const char *audioCodingTypeString(OMX_AUDIO_CODINGTYPE type) {
3935693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    static const char *kNames[] = {
3936693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingUnused",
3937693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingAutoDetect",
3938693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingPCM",
3939693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingADPCM",
3940693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingAMR",
3941693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingGSMFR",
3942693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingGSMEFR",
3943693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingGSMHR",
3944693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingPDCFR",
3945693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingPDCEFR",
3946693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingPDCHR",
3947693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingTDMAFR",
3948693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingTDMAEFR",
3949693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingQCELP8",
3950693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingQCELP13",
3951693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingEVRC",
3952693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingSMV",
3953693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingG711",
3954693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingG723",
3955693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingG726",
3956693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingG729",
3957693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingAAC",
3958693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingMP3",
3959693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingSBC",
3960693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingVORBIS",
3961693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingWMA",
3962693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingRA",
3963693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_CodingMIDI",
3964693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    };
3965693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3966693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    size_t numNames = sizeof(kNames) / sizeof(kNames[0]);
3967693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3968693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (type < 0 || (size_t)type >= numNames) {
3969693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return "UNKNOWN";
3970693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
3971693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return kNames[type];
3972693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3973693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3974693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3975693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberstatic const char *audioPCMModeString(OMX_AUDIO_PCMMODETYPE type) {
3976693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    static const char *kNames[] = {
3977693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_PCMModeLinear",
3978693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_PCMModeALaw",
3979693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        "OMX_AUDIO_PCMModeMULaw",
3980693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    };
3981693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3982693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    size_t numNames = sizeof(kNames) / sizeof(kNames[0]);
3983693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
3984693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    if (type < 0 || (size_t)type >= numNames) {
3985693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return "UNKNOWN";
3986693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    } else {
3987693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        return kNames[type];
3988693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
3989693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
3990693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
39912f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huberstatic const char *amrBandModeString(OMX_AUDIO_AMRBANDMODETYPE type) {
39922f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    static const char *kNames[] = {
39932f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeUnused",
39942f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB0",
39952f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB1",
39962f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB2",
39972f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB3",
39982f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB4",
39992f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB5",
40002f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB6",
40012f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeNB7",
40022f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB0",
40032f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB1",
40042f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB2",
40052f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB3",
40062f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB4",
40072f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB5",
40082f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB6",
40092f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB7",
40102f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRBandModeWB8",
40112f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    };
40122f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
40132f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    size_t numNames = sizeof(kNames) / sizeof(kNames[0]);
40142f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
40152f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    if (type < 0 || (size_t)type >= numNames) {
40162f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        return "UNKNOWN";
40172f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    } else {
40182f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        return kNames[type];
40192f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    }
40202f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber}
40212f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
40222f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huberstatic const char *amrFrameFormatString(OMX_AUDIO_AMRFRAMEFORMATTYPE type) {
40232f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    static const char *kNames[] = {
40242f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRFrameFormatConformance",
40252f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRFrameFormatIF1",
40262f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRFrameFormatIF2",
40272f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRFrameFormatFSF",
40282f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRFrameFormatRTPPayload",
40292f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        "OMX_AUDIO_AMRFrameFormatITU",
40302f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    };
40312f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
40322f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    size_t numNames = sizeof(kNames) / sizeof(kNames[0]);
40332f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
40342f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    if (type < 0 || (size_t)type >= numNames) {
40352f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        return "UNKNOWN";
40362f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    } else {
40372f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber        return kNames[type];
40382f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber    }
40392f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber}
4040693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4041693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::dumpPortStatus(OMX_U32 portIndex) {
4042693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
40437a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
4044693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = portIndex;
4045693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4046318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->getParameter(
4047693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
4048f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
4049693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4050693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    printf("%s Port = {\n", portIndex == kPortIndexInput ? "Input" : "Output");
4051693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4052693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    CHECK((portIndex == kPortIndexInput && def.eDir == OMX_DirInput)
4053693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber          || (portIndex == kPortIndexOutput && def.eDir == OMX_DirOutput));
4054693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4055693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    printf("  nBufferCountActual = %ld\n", def.nBufferCountActual);
4056693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    printf("  nBufferCountMin = %ld\n", def.nBufferCountMin);
4057693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    printf("  nBufferSize = %ld\n", def.nBufferSize);
4058693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4059693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    switch (def.eDomain) {
4060693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_PortDomainImage:
4061693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4062693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            const OMX_IMAGE_PORTDEFINITIONTYPE *imageDef = &def.format.image;
4063693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4064693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("\n");
4065693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  // Image\n");
4066693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  nFrameWidth = %ld\n", imageDef->nFrameWidth);
4067693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  nFrameHeight = %ld\n", imageDef->nFrameHeight);
4068693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  nStride = %ld\n", imageDef->nStride);
4069693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4070693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  eCompressionFormat = %s\n",
4071693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                   imageCompressionFormatString(imageDef->eCompressionFormat));
4072693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4073693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  eColorFormat = %s\n",
4074693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                   colorFormatString(imageDef->eColorFormat));
4075693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4076693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4077693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4078693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4079693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_PortDomainVideo:
4080693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4081693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_VIDEO_PORTDEFINITIONTYPE *videoDef = &def.format.video;
4082693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4083693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("\n");
4084693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  // Video\n");
4085693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  nFrameWidth = %ld\n", videoDef->nFrameWidth);
4086693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  nFrameHeight = %ld\n", videoDef->nFrameHeight);
4087693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  nStride = %ld\n", videoDef->nStride);
4088693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4089693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  eCompressionFormat = %s\n",
4090693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                   videoCompressionFormatString(videoDef->eCompressionFormat));
4091693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4092693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  eColorFormat = %s\n",
4093693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                   colorFormatString(videoDef->eColorFormat));
4094693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4095693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4096693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4097693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4098693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_PortDomainAudio:
4099693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4100693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_AUDIO_PORTDEFINITIONTYPE *audioDef = &def.format.audio;
4101693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4102693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("\n");
4103693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  // Audio\n");
4104693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  eEncoding = %s\n",
4105693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                   audioCodingTypeString(audioDef->eEncoding));
4106693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4107693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (audioDef->eEncoding == OMX_AUDIO_CodingPCM) {
4108693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                OMX_AUDIO_PARAM_PCMMODETYPE params;
41097a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                InitOMXParams(&params);
4110693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                params.nPortIndex = portIndex;
4111693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4112318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                err = mOMX->getParameter(
4113693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        mNode, OMX_IndexParamAudioPcm, &params, sizeof(params));
4114f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
4115693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4116693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                printf("  nSamplingRate = %ld\n", params.nSamplingRate);
4117693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                printf("  nChannels = %ld\n", params.nChannels);
4118693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                printf("  bInterleaved = %d\n", params.bInterleaved);
4119693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                printf("  nBitPerSample = %ld\n", params.nBitPerSample);
4120693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4121693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                printf("  eNumData = %s\n",
4122693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                       params.eNumData == OMX_NumericalDataSigned
4123693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                        ? "signed" : "unsigned");
4124693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4125693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                printf("  ePCMMode = %s\n", audioPCMModeString(params.ePCMMode));
41262f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber            } else if (audioDef->eEncoding == OMX_AUDIO_CodingAMR) {
41272f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                OMX_AUDIO_PARAM_AMRTYPE amr;
41282f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                InitOMXParams(&amr);
41292f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                amr.nPortIndex = portIndex;
41302f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
4131318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                err = mOMX->getParameter(
41322f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                        mNode, OMX_IndexParamAudioAmr, &amr, sizeof(amr));
4133f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
41342f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
41352f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                printf("  nChannels = %ld\n", amr.nChannels);
41362f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                printf("  eAMRBandMode = %s\n",
41372f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                        amrBandModeString(amr.eAMRBandMode));
41382f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                printf("  eAMRFrameFormat = %s\n",
41392f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                        amrFrameFormatString(amr.eAMRFrameFormat));
4140693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
4141693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4142693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4143693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4144693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4145693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        default:
4146693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4147693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            printf("  // Unknown\n");
4148693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4149693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4150693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
4151693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4152693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    printf("}\n");
4153693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
4154693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
41556a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennisstatus_t OMXCodec::initNativeWindow() {
41566a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // Enable use of a GraphicBuffer as the output for this node.  This must
41576a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // happen before getting the IndexParamPortDefinition parameter because it
41586a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    // will affect the pixel format that the node reports.
41596a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    status_t err = mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_TRUE);
41606a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    if (err != 0) {
41616a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis        return err;
41626a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    }
41636a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
41646a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis    return OK;
41656a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis}
41666a9da9fc558263548ebfbae2cbf177eb7454a41bJamie Gennis
4167693d271e62a3726689ff68f4505ba49228eb94b2Andreas Hubervoid OMXCodec::initOutputFormat(const sp<MetaData> &inputFormat) {
4168693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mOutputFormat = new MetaData;
4169693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    mOutputFormat->setCString(kKeyDecoderComponent, mComponentName);
41708f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong    if (mIsEncoder) {
41718f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong        int32_t timeScale;
41728f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong        if (inputFormat->findInt32(kKeyTimeScale, &timeScale)) {
41738f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong            mOutputFormat->setInt32(kKeyTimeScale, timeScale);
41748f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong        }
41758f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong    }
4176693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4177693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
41787a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber    InitOMXParams(&def);
4179693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    def.nPortIndex = kPortIndexOutput;
4180693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4181318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    status_t err = mOMX->getParameter(
4182693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
4183f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber    CHECK_EQ(err, (status_t)OK);
4184693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4185693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    switch (def.eDomain) {
4186693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_PortDomainImage:
4187693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4188693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_IMAGE_PORTDEFINITIONTYPE *imageDef = &def.format.image;
4189f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            CHECK_EQ((int)imageDef->eCompressionFormat,
4190f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                     (int)OMX_IMAGE_CodingUnused);
4191693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
419218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            mOutputFormat->setCString(kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_RAW);
4193693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mOutputFormat->setInt32(kKeyColorFormat, imageDef->eColorFormat);
4194693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mOutputFormat->setInt32(kKeyWidth, imageDef->nFrameWidth);
4195693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mOutputFormat->setInt32(kKeyHeight, imageDef->nFrameHeight);
4196693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4197693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4198693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4199693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_PortDomainAudio:
4200693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4201693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_AUDIO_PORTDEFINITIONTYPE *audio_def = &def.format.audio;
4202693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4203e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            if (audio_def->eEncoding == OMX_AUDIO_CodingPCM) {
4204e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                OMX_AUDIO_PARAM_PCMMODETYPE params;
42057a6b9e2eca7d20457ace3538c689640e5bfda4f3Andreas Huber                InitOMXParams(&params);
4206e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                params.nPortIndex = kPortIndexOutput;
4207693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4208318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                err = mOMX->getParameter(
4209e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                        mNode, OMX_IndexParamAudioPcm, &params, sizeof(params));
4210f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
4211693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4212f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ((int)params.eNumData, (int)OMX_NumericalDataSigned);
4213f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(params.nBitPerSample, 16u);
4214f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ((int)params.ePCMMode, (int)OMX_AUDIO_PCMModeLinear);
4215693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4216e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                int32_t numChannels, sampleRate;
4217e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                inputFormat->findInt32(kKeyChannelCount, &numChannels);
4218e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                inputFormat->findInt32(kKeySampleRate, &sampleRate);
4219693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4220e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                if ((OMX_U32)numChannels != params.nChannels) {
4221e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                    LOGW("Codec outputs a different number of channels than "
422278d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                         "the input stream contains (contains %d channels, "
422378d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                         "codec outputs %ld channels).",
422478d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                         numChannels, params.nChannels);
4225e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                }
422637940eefcba982836b579fe1ffec6cada72b0974Andreas Huber
4227b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber                if (sampleRate != (int32_t)params.nSamplingRate) {
4228bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                    LOGW("Codec outputs at different sampling rate than "
4229bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                         "what the input stream contains (contains data at "
4230b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber                         "%d Hz, codec outputs %lu Hz)",
4231bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                         sampleRate, params.nSamplingRate);
4232bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                }
4233bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
423418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                mOutputFormat->setCString(
423518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                        kKeyMIMEType, MEDIA_MIMETYPE_AUDIO_RAW);
423637940eefcba982836b579fe1ffec6cada72b0974Andreas Huber
4237e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                // Use the codec-advertised number of channels, as some
4238e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                // codecs appear to output stereo even if the input data is
423978d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                // mono. If we know the codec lies about this information,
424078d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                // use the actual number of channels instead.
424178d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                mOutputFormat->setInt32(
424278d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                        kKeyChannelCount,
424378d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                        (mQuirks & kDecoderLiesAboutNumberOfChannels)
424478d529eb330ab6c04b5c694403f5a7e7de4b702fAndreas Huber                            ? numChannels : params.nChannels);
424537940eefcba982836b579fe1ffec6cada72b0974Andreas Huber
4246bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                mOutputFormat->setInt32(kKeySampleRate, params.nSamplingRate);
4247e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            } else if (audio_def->eEncoding == OMX_AUDIO_CodingAMR) {
42482f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                OMX_AUDIO_PARAM_AMRTYPE amr;
42492f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                InitOMXParams(&amr);
42502f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                amr.nPortIndex = kPortIndexOutput;
42512f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
4252318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                err = mOMX->getParameter(
42532f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                        mNode, OMX_IndexParamAudioAmr, &amr, sizeof(amr));
4254f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(err, (status_t)OK);
42552f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
4256f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber                CHECK_EQ(amr.nChannels, 1u);
42572f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                mOutputFormat->setInt32(kKeyChannelCount, 1);
42582f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber
42592f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                if (amr.eAMRBandMode >= OMX_AUDIO_AMRBandModeNB0
42602f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                    && amr.eAMRBandMode <= OMX_AUDIO_AMRBandModeNB7) {
426118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                    mOutputFormat->setCString(
426218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                            kKeyMIMEType, MEDIA_MIMETYPE_AUDIO_AMR_NB);
42632f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                    mOutputFormat->setInt32(kKeySampleRate, 8000);
42642f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                } else if (amr.eAMRBandMode >= OMX_AUDIO_AMRBandModeWB0
42652f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                            && amr.eAMRBandMode <= OMX_AUDIO_AMRBandModeWB8) {
426618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                    mOutputFormat->setCString(
426718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                            kKeyMIMEType, MEDIA_MIMETYPE_AUDIO_AMR_WB);
42682f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                    mOutputFormat->setInt32(kKeySampleRate, 16000);
42692f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                } else {
42702f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                    CHECK(!"Unknown AMR band mode.");
42712f7daa1d7e01bfc2cd7546edeeaea0c3c2ee728cAndreas Huber                }
4272e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            } else if (audio_def->eEncoding == OMX_AUDIO_CodingAAC) {
427318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                mOutputFormat->setCString(
427418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                        kKeyMIMEType, MEDIA_MIMETYPE_AUDIO_AAC);
4275956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong                int32_t numChannels, sampleRate, bitRate;
4276050b28a593350047845a45a14cc5026221ac1620James Dong                inputFormat->findInt32(kKeyChannelCount, &numChannels);
4277050b28a593350047845a45a14cc5026221ac1620James Dong                inputFormat->findInt32(kKeySampleRate, &sampleRate);
4278956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong                inputFormat->findInt32(kKeyBitRate, &bitRate);
4279050b28a593350047845a45a14cc5026221ac1620James Dong                mOutputFormat->setInt32(kKeyChannelCount, numChannels);
4280050b28a593350047845a45a14cc5026221ac1620James Dong                mOutputFormat->setInt32(kKeySampleRate, sampleRate);
4281956c553ab0ce72f8074ad0fda2ffd66a0305700cJames Dong                mOutputFormat->setInt32(kKeyBitRate, bitRate);
4282e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            } else {
4283e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber                CHECK(!"Should not be here. Unknown audio encoding.");
4284e5adbeee3401915ff8e1a983396ce3554436b11cAndreas Huber            }
4285693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4286693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4287693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4288693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        case OMX_PortDomainVideo:
4289693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4290693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            OMX_VIDEO_PORTDEFINITIONTYPE *video_def = &def.format.video;
4291693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4292693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            if (video_def->eCompressionFormat == OMX_VIDEO_CodingUnused) {
429318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                mOutputFormat->setCString(
429418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                        kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_RAW);
4295693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            } else if (video_def->eCompressionFormat == OMX_VIDEO_CodingMPEG4) {
429618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                mOutputFormat->setCString(
429718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                        kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_MPEG4);
4298693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            } else if (video_def->eCompressionFormat == OMX_VIDEO_CodingH263) {
429918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                mOutputFormat->setCString(
430018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                        kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_H263);
4301693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            } else if (video_def->eCompressionFormat == OMX_VIDEO_CodingAVC) {
430218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                mOutputFormat->setCString(
430318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                        kKeyMIMEType, MEDIA_MIMETYPE_VIDEO_AVC);
4304693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            } else {
4305693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber                CHECK(!"Unknown compression format.");
4306693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            }
4307693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4308c810b75fd04eef8af509bb42aa8837566a374b32James Dong            mOutputFormat->setInt32(kKeyWidth, video_def->nFrameWidth);
4309c810b75fd04eef8af509bb42aa8837566a374b32James Dong            mOutputFormat->setInt32(kKeyHeight, video_def->nFrameHeight);
4310693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            mOutputFormat->setInt32(kKeyColorFormat, video_def->eColorFormat);
4311f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber
4312495f154b602a0103b24b98226370349100a26adfJames Dong            if (!mIsEncoder) {
4313495f154b602a0103b24b98226370349100a26adfJames Dong                OMX_CONFIG_RECTTYPE rect;
4314c29dd6f4ccd7ea7e41921d48d03d26e159aeaae4James Dong                InitOMXParams(&rect);
4315c29dd6f4ccd7ea7e41921d48d03d26e159aeaae4James Dong                rect.nPortIndex = kPortIndexOutput;
4316495f154b602a0103b24b98226370349100a26adfJames Dong                status_t err =
4317495f154b602a0103b24b98226370349100a26adfJames Dong                        mOMX->getConfig(
4318495f154b602a0103b24b98226370349100a26adfJames Dong                            mNode, OMX_IndexConfigCommonOutputCrop,
4319495f154b602a0103b24b98226370349100a26adfJames Dong                            &rect, sizeof(rect));
4320495f154b602a0103b24b98226370349100a26adfJames Dong
4321884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber                CODEC_LOGI(
4322884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber                        "video dimensions are %ld x %ld",
4323884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber                        video_def->nFrameWidth, video_def->nFrameHeight);
4324884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber
4325495f154b602a0103b24b98226370349100a26adfJames Dong                if (err == OK) {
4326495f154b602a0103b24b98226370349100a26adfJames Dong                    CHECK_GE(rect.nLeft, 0);
4327495f154b602a0103b24b98226370349100a26adfJames Dong                    CHECK_GE(rect.nTop, 0);
4328495f154b602a0103b24b98226370349100a26adfJames Dong                    CHECK_GE(rect.nWidth, 0u);
4329495f154b602a0103b24b98226370349100a26adfJames Dong                    CHECK_GE(rect.nHeight, 0u);
4330495f154b602a0103b24b98226370349100a26adfJames Dong                    CHECK_LE(rect.nLeft + rect.nWidth - 1, video_def->nFrameWidth);
4331495f154b602a0103b24b98226370349100a26adfJames Dong                    CHECK_LE(rect.nTop + rect.nHeight - 1, video_def->nFrameHeight);
4332495f154b602a0103b24b98226370349100a26adfJames Dong
4333495f154b602a0103b24b98226370349100a26adfJames Dong                    mOutputFormat->setRect(
4334495f154b602a0103b24b98226370349100a26adfJames Dong                            kKeyCropRect,
4335495f154b602a0103b24b98226370349100a26adfJames Dong                            rect.nLeft,
4336495f154b602a0103b24b98226370349100a26adfJames Dong                            rect.nTop,
4337495f154b602a0103b24b98226370349100a26adfJames Dong                            rect.nLeft + rect.nWidth - 1,
4338495f154b602a0103b24b98226370349100a26adfJames Dong                            rect.nTop + rect.nHeight - 1);
4339884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber
4340884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber                    CODEC_LOGI(
4341884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber                            "Crop rect is %ld x %ld @ (%ld, %ld)",
4342884946a23cfe804491ef6067bd9361371d7848fdAndreas Huber                            rect.nWidth, rect.nHeight, rect.nLeft, rect.nTop);
4343495f154b602a0103b24b98226370349100a26adfJames Dong                } else {
4344495f154b602a0103b24b98226370349100a26adfJames Dong                    mOutputFormat->setRect(
4345495f154b602a0103b24b98226370349100a26adfJames Dong                            kKeyCropRect,
4346495f154b602a0103b24b98226370349100a26adfJames Dong                            0, 0,
4347495f154b602a0103b24b98226370349100a26adfJames Dong                            video_def->nFrameWidth - 1,
4348495f154b602a0103b24b98226370349100a26adfJames Dong                            video_def->nFrameHeight - 1);
4349495f154b602a0103b24b98226370349100a26adfJames Dong                }
4350f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber            }
4351693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4352693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4353693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4354693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        default:
4355693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        {
4356693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            CHECK(!"should not be here, neither audio nor video.");
4357693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber            break;
4358693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber        }
4359693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    }
4360b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber
4361b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber    // If the input format contains rotation information, flag the output
4362b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber    // format accordingly.
4363b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber
4364b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber    int32_t rotationDegrees;
4365b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber    if (mSource->getFormat()->findInt32(kKeyRotation, &rotationDegrees)) {
4366b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber        mOutputFormat->setInt32(kKeyRotation, rotationDegrees);
4367b5746d9058c3e81a195f82f5345e1ffe0a26c0b2Andreas Huber    }
4368693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}
4369693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
4370d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huberstatus_t OMXCodec::pause() {
4371d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber    Mutex::Autolock autoLock(mLock);
4372d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber
4373d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber    mPaused = true;
4374d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber
4375d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber    return OK;
4376d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber}
4377d35bd5fb4e09c2cd8608497c279cbb2ef9c3a029Andreas Huber
437818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber////////////////////////////////////////////////////////////////////////////////
437918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
438018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huberstatus_t QueryCodecs(
438118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        const sp<IOMX> &omx,
4382df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi        const char *mime, bool queryDecoders, bool hwCodecOnly,
438318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        Vector<CodecCapabilities> *results) {
4384df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi    Vector<String8> matchingCodecs;
438518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    results->clear();
438618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
4387df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi    OMXCodec::findMatchingCodecs(mime,
4388df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi            !queryDecoders /*createEncoder*/,
4389df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi            NULL /*matchComponentName*/,
4390df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi            hwCodecOnly ? OMXCodec::kHardwareCodecsOnly : 0 /*flags*/,
4391df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi            &matchingCodecs);
439218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
4393df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi    for (size_t c = 0; c < matchingCodecs.size(); c++) {
4394df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi        const char *componentName = matchingCodecs.itemAt(c).string();
439518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
439608a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber        if (strncmp(componentName, "OMX.", 4)) {
439708a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber            // Not an OpenMax component but a software codec.
439808a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber
439908a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber            results->push();
440008a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber            CodecCapabilities *caps = &results->editItemAt(results->size() - 1);
440108a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber            caps->mComponentName = componentName;
440208a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber            continue;
440308a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber        }
440408a88e195d2b3697f2f967e9216491e8c5bd3c9eAndreas Huber
4405318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        sp<OMXCodecObserver> observer = new OMXCodecObserver;
440618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        IOMX::node_id node;
4407318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        status_t err = omx->allocateNode(componentName, observer, &node);
440818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
440918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        if (err != OK) {
441018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            continue;
441118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        }
441218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
44135e3cf6cf36feb3b73b8f8c0a8134c46f5bc85094James Dong        OMXCodec::setComponentRole(omx, node, !queryDecoders, mime);
441418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
441518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        results->push();
441618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        CodecCapabilities *caps = &results->editItemAt(results->size() - 1);
441718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        caps->mComponentName = componentName;
441818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
441918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        OMX_VIDEO_PARAM_PROFILELEVELTYPE param;
442018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        InitOMXParams(&param);
442118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
442218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        param.nPortIndex = queryDecoders ? 0 : 1;
442318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
442418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        for (param.nProfileIndex = 0;; ++param.nProfileIndex) {
4425318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            err = omx->getParameter(
442618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                    node, OMX_IndexParamVideoProfileLevelQuerySupported,
442718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                    &param, sizeof(param));
442818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
442918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            if (err != OK) {
443018291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber                break;
443118291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            }
443218291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
443318291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            CodecProfileLevel profileLevel;
443418291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            profileLevel.mProfile = param.eProfile;
443518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            profileLevel.mLevel = param.eLevel;
443618291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
443718291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber            caps->mProfileLevels.push(profileLevel);
443818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber        }
443918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
4440b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong        // Color format query
4441b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong        OMX_VIDEO_PARAM_PORTFORMATTYPE portFormat;
4442b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong        InitOMXParams(&portFormat);
4443b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong        portFormat.nPortIndex = queryDecoders ? 1 : 0;
4444b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong        for (portFormat.nIndex = 0;; ++portFormat.nIndex)  {
4445b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong            err = omx->getParameter(
4446b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong                    node, OMX_IndexParamVideoPortFormat,
4447b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong                    &portFormat, sizeof(portFormat));
4448b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong            if (err != OK) {
4449b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong                break;
4450b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong            }
4451b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong            caps->mColorFormats.push(portFormat.eColorFormat);
4452b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong        }
4453b50a8033f1c7c2b58913212825f9200f1a9e5652James Dong
4454f5ab57c2d5e02af7483c94eddb177e4f5c9e9892Andreas Huber        CHECK_EQ(omx->freeNode(node), (status_t)OK);
445518291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber    }
4456df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi
4457df57a0d6d3b0609e69ff79afa63fe44dfa91f6f5Jean-Michel Trivi    return OK;
445818291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber}
445918291bc20e55e8f3fd5feb786771a8ed32c19c59Andreas Huber
4460d35924d9928f29dcee6f5666b5bbd084640c7b34Jean-Michel Trivistatus_t QueryCodecs(
4461d35924d9928f29dcee6f5666b5bbd084640c7b34Jean-Michel Trivi        const sp<IOMX> &omx,
4462d35924d9928f29dcee6f5666b5bbd084640c7b34Jean-Michel Trivi        const char *mimeType, bool queryDecoders,
4463d35924d9928f29dcee6f5666b5bbd084640c7b34Jean-Michel Trivi        Vector<CodecCapabilities> *results) {
4464d35924d9928f29dcee6f5666b5bbd084640c7b34Jean-Michel Trivi    return QueryCodecs(omx, mimeType, queryDecoders, false /*hwCodecOnly*/, results);
4465d35924d9928f29dcee6f5666b5bbd084640c7b34Jean-Michel Trivi}
4466d35924d9928f29dcee6f5666b5bbd084640c7b34Jean-Michel Trivi
4467f23c4f92c3b0202435cf87db2642156fabc46f02James Dongvoid OMXCodec::restorePatchedDataPointer(BufferInfo *info) {
4468f23c4f92c3b0202435cf87db2642156fabc46f02James Dong    CHECK(mIsEncoder && (mQuirks & kAvoidMemcopyInputRecordingFrames));
4469f23c4f92c3b0202435cf87db2642156fabc46f02James Dong    CHECK(mOMXLivesLocally);
4470f23c4f92c3b0202435cf87db2642156fabc46f02James Dong
4471f23c4f92c3b0202435cf87db2642156fabc46f02James Dong    OMX_BUFFERHEADERTYPE *header = (OMX_BUFFERHEADERTYPE *)info->mBuffer;
4472f23c4f92c3b0202435cf87db2642156fabc46f02James Dong    header->pBuffer = (OMX_U8 *)info->mData;
4473f23c4f92c3b0202435cf87db2642156fabc46f02James Dong}
4474f23c4f92c3b0202435cf87db2642156fabc46f02James Dong
4475693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber}  // namespace android
4476