ACodec.cpp revision 604bb9ea6e9bec763ae231330066ecffa90a2786
127c174483a8ae9688d5d4897c19074f62c7f1701James Dong/*
227c174483a8ae9688d5d4897c19074f62c7f1701James Dong * Copyright (C) 2010 The Android Open Source Project
327c174483a8ae9688d5d4897c19074f62c7f1701James Dong *
427c174483a8ae9688d5d4897c19074f62c7f1701James Dong * Licensed under the Apache License, Version 2.0 (the "License");
527c174483a8ae9688d5d4897c19074f62c7f1701James Dong * you may not use this file except in compliance with the License.
627c174483a8ae9688d5d4897c19074f62c7f1701James Dong * You may obtain a copy of the License at
727c174483a8ae9688d5d4897c19074f62c7f1701James Dong *
827c174483a8ae9688d5d4897c19074f62c7f1701James Dong *      http://www.apache.org/licenses/LICENSE-2.0
927c174483a8ae9688d5d4897c19074f62c7f1701James Dong *
1027c174483a8ae9688d5d4897c19074f62c7f1701James Dong * Unless required by applicable law or agreed to in writing, software
1127c174483a8ae9688d5d4897c19074f62c7f1701James Dong * distributed under the License is distributed on an "AS IS" BASIS,
1227c174483a8ae9688d5d4897c19074f62c7f1701James Dong * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1327c174483a8ae9688d5d4897c19074f62c7f1701James Dong * See the License for the specific language governing permissions and
1427c174483a8ae9688d5d4897c19074f62c7f1701James Dong * limitations under the License.
1527c174483a8ae9688d5d4897c19074f62c7f1701James Dong */
1627c174483a8ae9688d5d4897c19074f62c7f1701James Dong
17f933441648ef6a71dee783d733aac17b9508b452Andreas Huber//#define LOG_NDEBUG 0
18f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#define LOG_TAG "ACodec"
19f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
20f87e30fe71752dc431d8e8d5682c38271c03265aAndy Hung#ifdef __LP64__
21f87e30fe71752dc431d8e8d5682c38271c03265aAndy Hung#define OMX_ANDROID_COMPILE_AS_32BIT_ON_64BIT_PLATFORMS
22f87e30fe71752dc431d8e8d5682c38271c03265aAndy Hung#endif
23f87e30fe71752dc431d8e8d5682c38271c03265aAndy Hung
24609b815a3131d22da38b2f452faa9f89daad4039Andy Hung#include <inttypes.h>
256fbc1afd36f514f8b8d36ce67f5e69776c690bd2Lajos Molnar#include <utils/Trace.h>
266fbc1afd36f514f8b8d36ce67f5e69776c690bd2Lajos Molnar
271de1e25cba872bd4c077c2e394f8ca9c70b65856Lajos Molnar#include <gui/Surface.h>
281de1e25cba872bd4c077c2e394f8ca9c70b65856Lajos Molnar
29f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/ACodec.h>
30f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
31f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <binder/MemoryDealer.h>
32f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
33f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/hexdump.h>
34f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/ABuffer.h>
35f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/ADebug.h>
36f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/AMessage.h>
37a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar#include <media/stagefright/foundation/AUtils.h>
38f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
397cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden#include <media/stagefright/BufferProducerWrapper.h>
4090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar#include <media/stagefright/MediaCodec.h>
41afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber#include <media/stagefright/MediaCodecList.h>
42f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/MediaDefs.h>
43f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/OMXClient.h>
441065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber#include <media/stagefright/OMXCodec.h>
45d291c222357303b9611cab89d0c3b047584ef377Chong Zhang#include <media/stagefright/PersistentSurface.h>
46b2d0b487efd40700199852c9a18b369b1651f15bLajos Molnar#include <media/stagefright/SurfaceUtils.h>
473a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber#include <media/hardware/HardwareAPI.h>
483a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber
4997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu#include <OMX_AudioExt.h>
504154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev#include <OMX_VideoExt.h>
51f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <OMX_Component.h>
5297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu#include <OMX_IndexExt.h>
53777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim#include <OMX_AsString.h>
54f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
55496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber#include "include/avc_utils.h"
56496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
57f933441648ef6a71dee783d733aac17b9508b452Andreas Hubernamespace android {
58f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
59251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// OMX errors are directly mapped into status_t range if
60251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// there is no corresponding MediaError status code.
61251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// Use the statusFromOMXError(int32_t omxError) function.
62251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung//
63251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// Currently this is a direct map.
64251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// See frameworks/native/include/media/openmax/OMX_Core.h
65251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung//
66251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// Vendor OMX errors     from 0x90000000 - 0x9000FFFF
67251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// Extension OMX errors  from 0x8F000000 - 0x90000000
68251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// Standard OMX errors   from 0x80001000 - 0x80001024 (0x80001024 current)
69251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung//
70251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung
71251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// returns true if err is a recognized OMX error code.
72251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// as OMX error is OMX_S32, this is an int32_t type
73251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hungstatic inline bool isOMXError(int32_t err) {
74251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    return (ERROR_CODEC_MIN <= err && err <= ERROR_CODEC_MAX);
75251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung}
76251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung
77251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// converts an OMX error to a status_t
78251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hungstatic inline status_t statusFromOMXError(int32_t omxError) {
79251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    switch (omxError) {
80251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    case OMX_ErrorInvalidComponentName:
81251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    case OMX_ErrorComponentNotFound:
82251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        return NAME_NOT_FOUND; // can trigger illegal argument error for provided names.
83251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    default:
84251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        return isOMXError(omxError) ? omxError : 0; // no translation required
85251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    }
86251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung}
87251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung
88251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung// checks and converts status_t to a non-side-effect status_t
89251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hungstatic inline status_t makeNoSideEffectStatus(status_t err) {
90251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    switch (err) {
91251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    // the following errors have side effects and may come
92251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    // from other code modules. Remap for safety reasons.
93251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    case INVALID_OPERATION:
94251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    case DEAD_OBJECT:
95251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        return UNKNOWN_ERROR;
96251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    default:
97251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        return err;
98251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    }
99251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung}
100251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung
101f933441648ef6a71dee783d733aac17b9508b452Andreas Hubertemplate<class T>
102f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatic void InitOMXParams(T *params) {
103f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nSize = sizeof(T);
104f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nVersionMajor = 1;
105f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nVersionMinor = 0;
106f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nRevision = 0;
107f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nStep = 0;
108f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
109f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
11026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnarstruct MessageList : public RefBase {
11126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    MessageList() {
11226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    }
113984a54322f7c70bc75e862d91bdd975814872affLajos Molnar    virtual ~MessageList() {
114984a54322f7c70bc75e862d91bdd975814872affLajos Molnar    }
11526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    std::list<sp<AMessage> > &getList() { return mList; }
11626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnarprivate:
11726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    std::list<sp<AMessage> > mList;
11826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
11926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    DISALLOW_EVIL_CONSTRUCTORS(MessageList);
12026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar};
12126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
122f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct CodecObserver : public BnOMXObserver {
123f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CodecObserver() {}
124f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
125f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void setNotificationMessage(const sp<AMessage> &msg) {
126f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mNotify = msg;
127f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
128f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
129f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // from IOMXObserver
13026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    virtual void onMessages(const std::list<omx_message> &messages) {
131984a54322f7c70bc75e862d91bdd975814872affLajos Molnar        if (messages.empty()) {
132984a54322f7c70bc75e862d91bdd975814872affLajos Molnar            return;
133984a54322f7c70bc75e862d91bdd975814872affLajos Molnar        }
134984a54322f7c70bc75e862d91bdd975814872affLajos Molnar
135984a54322f7c70bc75e862d91bdd975814872affLajos Molnar        sp<AMessage> notify = mNotify->dup();
13626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        bool first = true;
13726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        sp<MessageList> msgList = new MessageList();
13826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        for (std::list<omx_message>::const_iterator it = messages.cbegin();
13926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar              it != messages.cend(); ++it) {
14026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            const omx_message &omx_msg = *it;
14126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            if (first) {
14226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                notify->setInt32("node", omx_msg.node);
143984a54322f7c70bc75e862d91bdd975814872affLajos Molnar                first = false;
144f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
145f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
14626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            sp<AMessage> msg = new AMessage;
14726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            msg->setInt32("type", omx_msg.type);
14826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            switch (omx_msg.type) {
14926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                case omx_message::EVENT:
15026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                {
15126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32("event", omx_msg.u.event_data.event);
15226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32("data1", omx_msg.u.event_data.data1);
15326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32("data2", omx_msg.u.event_data.data2);
15426a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    break;
15526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                }
156f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
15726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                case omx_message::EMPTY_BUFFER_DONE:
15826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                {
15926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32("buffer", omx_msg.u.buffer_data.buffer);
16026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32("fence_fd", omx_msg.fenceFd);
16126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    break;
16226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                }
163f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
16426a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                case omx_message::FILL_BUFFER_DONE:
16526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                {
16626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32(
16726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            "buffer", omx_msg.u.extended_buffer_data.buffer);
16826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32(
16926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            "range_offset",
17026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            omx_msg.u.extended_buffer_data.range_offset);
17126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32(
17226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            "range_length",
17326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            omx_msg.u.extended_buffer_data.range_length);
17426a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32(
17526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            "flags",
17626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            omx_msg.u.extended_buffer_data.flags);
17726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt64(
17826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            "timestamp",
17926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            omx_msg.u.extended_buffer_data.timestamp);
18026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    msg->setInt32(
18126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                            "fence_fd", omx_msg.fenceFd);
18226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    break;
18326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                }
184f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
18590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                case omx_message::FRAME_RENDERED:
18690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                {
18790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                    msg->setInt64(
18890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                            "media_time_us", omx_msg.u.render_data.timestamp);
18990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                    msg->setInt64(
19090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                            "system_nano", omx_msg.u.render_data.nanoTime);
19190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                    break;
19290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                }
19390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
19426a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                default:
19526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    ALOGE("Unrecognized message type: %d", omx_msg.type);
19626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar                    break;
19726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            }
19826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            msgList->getList().push_back(msg);
19926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        }
20026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        notify->setObject("messages", msgList);
20126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        notify->post();
202f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
203f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
204f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
205f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual ~CodecObserver() {}
206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
207f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> mNotify;
209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
210f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(CodecObserver);
211f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
212f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
213f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
214f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
215f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::BaseState : public AState {
216f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState(ACodec *codec, const sp<AState> &parentState = NULL);
217f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
218f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
219f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    enum PortMode {
220f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        KEEP_BUFFERS,
221f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        RESUBMIT_BUFFERS,
222f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        FREE_BUFFERS,
223f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
224f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
225f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ACodec *mCodec;
226f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
227f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual PortMode getPortMode(OMX_U32 portIndex);
228f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
229f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
230f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
231f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
232f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onOutputBufferDrained(const sp<AMessage> &msg);
234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onInputBufferFilled(const sp<AMessage> &msg);
235f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
236f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void postFillThisBuffer(BufferInfo *info);
237f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
238f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
23926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    // Handles an OMX message. Returns true iff message was handled.
240f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool onOMXMessage(const sp<AMessage> &msg);
241f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
24226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    // Handles a list of messages. Returns true iff messages were handled.
24326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    bool onOMXMessageList(const sp<AMessage> &msg);
24426a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
24526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    // returns true iff this message is for this component and the component is alive
24626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    bool checkOMXMessage(const sp<AMessage> &msg);
24726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
24815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    bool onOMXEmptyBufferDone(IOMX::buffer_id bufferID, int fenceFd);
249f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
250f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool onOMXFillBufferDone(
251f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID,
252f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            size_t rangeOffset, size_t rangeLength,
253f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            OMX_U32 flags,
25415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            int64_t timeUs,
25515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            int fenceFd);
256f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
25790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    virtual bool onOMXFrameRendered(int64_t mediaTimeUs, nsecs_t systemNano);
25890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
259f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void getMoreInputDataIfPossible();
260f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
261f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(BaseState);
262f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
263f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
264f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
265f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
266ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huberstruct ACodec::DeathNotifier : public IBinder::DeathRecipient {
267ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    DeathNotifier(const sp<AMessage> &notify)
268ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        : mNotify(notify) {
269ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    }
270ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
271ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    virtual void binderDied(const wp<IBinder> &) {
272ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        mNotify->post();
273ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    }
274ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
275ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huberprotected:
276ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    virtual ~DeathNotifier() {}
277ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
278ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huberprivate:
279ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    sp<AMessage> mNotify;
280ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
281ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(DeathNotifier);
282ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber};
283ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
284f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::UninitializedState : public ACodec::BaseState {
285f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    UninitializedState(ACodec *codec);
286f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
287f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
288f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
289c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    virtual void stateEntered();
290f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
291f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
292f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void onSetup(const sp<AMessage> &msg);
293c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    bool onAllocateComponent(const sp<AMessage> &msg);
294f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
295ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    sp<DeathNotifier> mDeathNotifier;
296ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
297f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(UninitializedState);
298f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
299f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
300f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
301f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
302c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huberstruct ACodec::LoadedState : public ACodec::BaseState {
303c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    LoadedState(ACodec *codec);
304c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
305c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huberprotected:
306c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
307c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    virtual void stateEntered();
308c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
309c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huberprivate:
310c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    friend struct ACodec::UninitializedState;
311c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
312c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    bool onConfigureComponent(const sp<AMessage> &msg);
3137cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden    void onCreateInputSurface(const sp<AMessage> &msg);
3148f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang    void onSetInputSurface(const sp<AMessage> &msg);
315c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    void onStart();
316c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    void onShutdown(bool keepComponentAllocated);
317c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
318d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    status_t setupInputSurface();
319d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
320c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(LoadedState);
321c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber};
322c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
323c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber////////////////////////////////////////////////////////////////////////////////
324c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
325f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::LoadedToIdleState : public ACodec::BaseState {
326f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LoadedToIdleState(ACodec *codec);
327f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
328f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
329f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
330f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
331f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
333f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
334f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t allocateBuffers();
335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(LoadedToIdleState);
337f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
338f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
341f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::IdleToExecutingState : public ACodec::BaseState {
342f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IdleToExecutingState(ACodec *codec);
343f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
344f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
345f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
346f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
347f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
348f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
349f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
350f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(IdleToExecutingState);
351f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
352f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
353f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
354f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
355f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::ExecutingState : public ACodec::BaseState {
356f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ExecutingState(ACodec *codec);
357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
358054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    void submitRegularOutputBuffers();
359054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    void submitOutputMetaBuffers();
360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void submitOutputBuffers();
361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Submit output buffers to the decoder, submit input buffers to client
363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // to fill with data.
364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void resume();
365f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
366349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    // Returns true iff input and output buffers are in play.
367349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    bool active() const { return mActive; }
368349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
369f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual PortMode getPortMode(OMX_U32 portIndex);
371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
373f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
374f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
37590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    virtual bool onOMXFrameRendered(int64_t mediaTimeUs, nsecs_t systemNano);
376f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
377f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
378349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    bool mActive;
379349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(ExecutingState);
381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
384f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
385f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::OutputPortSettingsChangedState : public ACodec::BaseState {
386f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OutputPortSettingsChangedState(ACodec *codec);
387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
388f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
389f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual PortMode getPortMode(OMX_U32 portIndex);
390f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
391f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
392f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
393f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
39490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    virtual bool onOMXFrameRendered(int64_t mediaTimeUs, nsecs_t systemNano);
395f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
396f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
397f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(OutputPortSettingsChangedState);
398f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
399f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
400f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
401f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
402f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::ExecutingToIdleState : public ACodec::BaseState {
403f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ExecutingToIdleState(ACodec *codec);
404f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
405f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
406f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
407f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
408f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
409f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
410f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
411f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onOutputBufferDrained(const sp<AMessage> &msg);
412f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onInputBufferFilled(const sp<AMessage> &msg);
413f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
414f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
415f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void changeStateIfWeOwnAllBuffers();
416f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4175778822d86b0337407514b9372562b86edfa91cdAndreas Huber    bool mComponentNowIdle;
4185778822d86b0337407514b9372562b86edfa91cdAndreas Huber
419f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(ExecutingToIdleState);
420f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
421f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
422f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
423f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
424f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::IdleToLoadedState : public ACodec::BaseState {
425f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IdleToLoadedState(ACodec *codec);
426f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
427f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
428f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
429f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
430f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
431f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
432f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
433f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
434f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(IdleToLoadedState);
435f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
436f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
437f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
438f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
439f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::FlushingState : public ACodec::BaseState {
440f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    FlushingState(ACodec *codec);
441f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
442f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
443f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
444f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
445f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
446f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
447f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
448f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onOutputBufferDrained(const sp<AMessage> &msg);
449f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onInputBufferFilled(const sp<AMessage> &msg);
450f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
451f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
452f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool mFlushComplete[2];
453f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
454f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void changeStateIfWeOwnAllBuffers();
455f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
456f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(FlushingState);
457f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
458f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
459f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
460f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
46115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnarvoid ACodec::BufferInfo::setWriteFence(int fenceFd, const char *dbg) {
46215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (mFenceFd >= 0) {
46315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        ALOGW("OVERWRITE OF %s fence %d by write fence %d in %s",
46415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                mIsReadFence ? "read" : "write", mFenceFd, fenceFd, dbg);
46515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
46615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    mFenceFd = fenceFd;
46715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    mIsReadFence = false;
46815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar}
46915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
47015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnarvoid ACodec::BufferInfo::setReadFence(int fenceFd, const char *dbg) {
47115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (mFenceFd >= 0) {
47215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        ALOGW("OVERWRITE OF %s fence %d by read fence %d in %s",
47315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                mIsReadFence ? "read" : "write", mFenceFd, fenceFd, dbg);
47415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
47515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    mFenceFd = fenceFd;
47615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    mIsReadFence = true;
47715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar}
47815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
47915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnarvoid ACodec::BufferInfo::checkWriteFence(const char *dbg) {
48015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (mFenceFd >= 0 && mIsReadFence) {
48115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        ALOGD("REUSING read fence %d as write fence in %s", mFenceFd, dbg);
48215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
48315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar}
48415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
48515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnarvoid ACodec::BufferInfo::checkReadFence(const char *dbg) {
48615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (mFenceFd >= 0 && !mIsReadFence) {
48715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        ALOGD("REUSING write fence %d as read fence in %s", mFenceFd, dbg);
48815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
48915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar}
49015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
49115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar////////////////////////////////////////////////////////////////////////////////
49215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
493f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ACodec()
494afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber    : mQuirks(0),
495609b815a3131d22da38b2f452faa9f89daad4039Andy Hung      mNode(0),
4965778822d86b0337407514b9372562b86edfa91cdAndreas Huber      mSentFormat(false),
4978db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu      mIsVideo(false),
498c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber      mIsEncoder(false),
4999806555d3930be43e11106281dee354820ac1c88Andreas Huber      mShutdownInProgress(false),
50054b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar      mExplicitShutdown(false),
5019806555d3930be43e11106281dee354820ac1c88Andreas Huber      mEncoderDelay(0),
5029806555d3930be43e11106281dee354820ac1c88Andreas Huber      mEncoderPadding(0),
503e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang      mRotationDegrees(0),
5049806555d3930be43e11106281dee354820ac1c88Andreas Huber      mChannelMaskPresent(false),
505054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar      mChannelMask(0),
506054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar      mDequeueCounter(0),
507054219874873b41f1c815552987c10465c34ba2bLajos Molnar      mInputMetadataType(kMetadataBufferTypeInvalid),
508054219874873b41f1c815552987c10465c34ba2bLajos Molnar      mOutputMetadataType(kMetadataBufferTypeInvalid),
509011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar      mLegacyAdaptiveExperiment(false),
510054219874873b41f1c815552987c10465c34ba2bLajos Molnar      mMetadataBuffersToSubmit(0),
51194ee4b708acfa941581160b267afb79192b1d816Chong Zhang      mRepeatFrameDelayUs(-1ll),
5122c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang      mMaxPtsGapUs(-1ll),
51337b2b389139ed638831e49708c947863eef631efRonghua Wu      mMaxFps(-1),
5142c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang      mTimePerFrameUs(-1ll),
515609b815a3131d22da38b2f452faa9f89daad4039Andy Hung      mTimePerCaptureUs(-1ll),
516ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad      mCreateInputBuffersSuspended(false),
517ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad      mTunneled(false) {
518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mUninitializedState = new UninitializedState(this);
519c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    mLoadedState = new LoadedState(this);
520f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mLoadedToIdleState = new LoadedToIdleState(this);
521f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mIdleToExecutingState = new IdleToExecutingState(this);
522f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mExecutingState = new ExecutingState(this);
523f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
524f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mOutputPortSettingsChangedState =
525f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        new OutputPortSettingsChangedState(this);
526f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
527f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mExecutingToIdleState = new ExecutingToIdleState(this);
528f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mIdleToLoadedState = new IdleToLoadedState(this);
529f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mFlushingState = new FlushingState(this);
530f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
531f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mPortEOS[kPortIndexInput] = mPortEOS[kPortIndexOutput] = false;
532dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber    mInputEOSResult = OK;
533f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
534f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeState(mUninitializedState);
535f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
536f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
537f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::~ACodec() {
538f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
539f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
540f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::setNotificationMessage(const sp<AMessage> &msg) {
541f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mNotify = msg;
542f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
543f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
544f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::initiateSetup(const sp<AMessage> &msg) {
545f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->setWhat(kWhatSetup);
5461d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    msg->setTarget(this);
547f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->post();
548f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
549f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
550a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Hubervoid ACodec::signalSetParameters(const sp<AMessage> &params) {
5511d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    sp<AMessage> msg = new AMessage(kWhatSetParameters, this);
552a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber    msg->setMessage("params", params);
553a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber    msg->post();
554a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber}
555a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
5565778822d86b0337407514b9372562b86edfa91cdAndreas Hubervoid ACodec::initiateAllocateComponent(const sp<AMessage> &msg) {
5575778822d86b0337407514b9372562b86edfa91cdAndreas Huber    msg->setWhat(kWhatAllocateComponent);
5581d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    msg->setTarget(this);
5595778822d86b0337407514b9372562b86edfa91cdAndreas Huber    msg->post();
5605778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
5615778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5625778822d86b0337407514b9372562b86edfa91cdAndreas Hubervoid ACodec::initiateConfigureComponent(const sp<AMessage> &msg) {
5635778822d86b0337407514b9372562b86edfa91cdAndreas Huber    msg->setWhat(kWhatConfigureComponent);
5641d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    msg->setTarget(this);
5655778822d86b0337407514b9372562b86edfa91cdAndreas Huber    msg->post();
5665778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
5675778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5681dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnarstatus_t ACodec::setSurface(const sp<Surface> &surface) {
5691dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    sp<AMessage> msg = new AMessage(kWhatSetSurface, this);
5701dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    msg->setObject("surface", surface);
5711dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
5721dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    sp<AMessage> response;
5731dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    status_t err = msg->postAndAwaitResponse(&response);
5741dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
5751dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err == OK) {
5761dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        (void)response->findInt32("err", &err);
5771dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
5781dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    return err;
5791dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar}
5801dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
5817cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFaddenvoid ACodec::initiateCreateInputSurface() {
5821d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    (new AMessage(kWhatCreateInputSurface, this))->post();
5837cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden}
5847cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden
5858f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhangvoid ACodec::initiateSetInputSurface(
586d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        const sp<PersistentSurface> &surface) {
5878f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang    sp<AMessage> msg = new AMessage(kWhatSetInputSurface, this);
588d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    msg->setObject("input-surface", surface);
589d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    msg->post();
590d291c222357303b9611cab89d0c3b047584ef377Chong Zhang}
591d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5927cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFaddenvoid ACodec::signalEndOfInputStream() {
5931d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    (new AMessage(kWhatSignalEndOfInputStream, this))->post();
5947cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden}
5957cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden
5965778822d86b0337407514b9372562b86edfa91cdAndreas Hubervoid ACodec::initiateStart() {
5971d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    (new AMessage(kWhatStart, this))->post();
5985778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
5995778822d86b0337407514b9372562b86edfa91cdAndreas Huber
600f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::signalFlush() {
6017a3a2b2f9bb9421dcf83fbd47276e57917078aefJames Dong    ALOGV("[%s] signalFlush", mComponentName.c_str());
6021d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    (new AMessage(kWhatFlush, this))->post();
603f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
604f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
605f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::signalResume() {
6061d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    (new AMessage(kWhatResume, this))->post();
607f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
608f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
609c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Hubervoid ACodec::initiateShutdown(bool keepComponentAllocated) {
6101d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    sp<AMessage> msg = new AMessage(kWhatShutdown, this);
611c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    msg->setInt32("keepComponentAllocated", keepComponentAllocated);
612c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    msg->post();
61330358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar    if (!keepComponentAllocated) {
61430358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar        // ensure shutdown completes in 3 seconds
6151d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar        (new AMessage(kWhatReleaseCodecInstance, this))->post(3000000);
61630358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar    }
617f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
618f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
619496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Hubervoid ACodec::signalRequestIDRFrame() {
6201d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    (new AMessage(kWhatRequestIDRFrame, this))->post();
621496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber}
622496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
6234dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar// *** NOTE: THE FOLLOWING WORKAROUND WILL BE REMOVED ***
6244dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar// Some codecs may return input buffers before having them processed.
6254dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar// This causes a halt if we already signaled an EOS on the input
6264dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar// port.  For now keep submitting an output buffer if there was an
6274dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar// EOS on the input port, but not yet on the output port.
628054219874873b41f1c815552987c10465c34ba2bLajos Molnarvoid ACodec::signalSubmitOutputMetadataBufferIfEOS_workaround() {
6294dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar    if (mPortEOS[kPortIndexInput] && !mPortEOS[kPortIndexOutput] &&
630054219874873b41f1c815552987c10465c34ba2bLajos Molnar            mMetadataBuffersToSubmit > 0) {
631054219874873b41f1c815552987c10465c34ba2bLajos Molnar        (new AMessage(kWhatSubmitOutputMetadataBufferIfEOS, this))->post();
6324dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar    }
6334dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar}
6344dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar
6351dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnarstatus_t ACodec::handleSetSurface(const sp<Surface> &surface) {
6361dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // allow keeping unset surface
6371dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (surface == NULL) {
6381dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        if (mNativeWindow != NULL) {
6391dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            ALOGW("cannot unset a surface");
6401dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            return INVALID_OPERATION;
6411dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        }
6421dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return OK;
6431dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6441dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
6451dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // allow keeping unset surface
6461dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (mNativeWindow == NULL) {
6471dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        ALOGW("component was not configured with a surface");
6481dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return INVALID_OPERATION;
6491dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6501dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
6511dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    ANativeWindow *nativeWindow = surface.get();
6521dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // if we have not yet started the codec, we can simply set the native window
6531dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (mBuffers[kPortIndexInput].size() == 0) {
6541dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        mNativeWindow = surface;
6551dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return OK;
6561dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6571dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
6581dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // we do not support changing a tunneled surface after start
6591dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (mTunneled) {
6601dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        ALOGW("cannot change tunneled surface");
6611dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return INVALID_OPERATION;
6621dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6631dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
6641dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    status_t err = setupNativeWindowSizeFormatAndUsage(nativeWindow);
6651dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err != OK) {
6661dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return err;
6671dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6681dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
6691dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // get min undequeued count. We cannot switch to a surface that has a higher
6701dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // undequeued count than we allocated.
6711dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    int minUndequeuedBuffers = 0;
6721dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    err = nativeWindow->query(
6731dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            nativeWindow, NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS,
6741dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            &minUndequeuedBuffers);
6751dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err != 0) {
6761dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        ALOGE("NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS query failed: %s (%d)",
6771dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                strerror(-err), -err);
6781dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return err;
6791dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6801dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (minUndequeuedBuffers > (int)mNumUndequeuedBuffers) {
6811dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        ALOGE("new surface holds onto more buffers (%d) than planned for (%zu)",
6821dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                minUndequeuedBuffers, mNumUndequeuedBuffers);
6831dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return BAD_VALUE;
6841dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6851dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
6861dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // we cannot change the number of output buffers while OMX is running
6871dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // set up surface to the same count
6881dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    Vector<BufferInfo> &buffers = mBuffers[kPortIndexOutput];
6891dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    ALOGV("setting up surface for %zu buffers", buffers.size());
6901dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
6911dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    err = native_window_set_buffer_count(nativeWindow, buffers.size());
6921dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err != 0) {
6931dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        ALOGE("native_window_set_buffer_count failed: %s (%d)", strerror(-err),
6941dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                -err);
6951dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return err;
6961dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
6971dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
698dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar    // need to enable allocation when attaching
699dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar    surface->getIGraphicBufferProducer()->allowAllocation(true);
700dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar
7011dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // for meta data mode, we move dequeud buffers to the new surface.
7021dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // for non-meta mode, we must move all registered buffers
7031dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    for (size_t i = 0; i < buffers.size(); ++i) {
7041dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        const BufferInfo &info = buffers[i];
7051dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        // skip undequeued buffers for meta data mode
706054219874873b41f1c815552987c10465c34ba2bLajos Molnar        if (storingMetadataInDecodedBuffers()
707011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                && !mLegacyAdaptiveExperiment
7081dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                && info.mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
7091dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            ALOGV("skipping buffer %p", info.mGraphicBuffer->getNativeBuffer());
7101dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            continue;
7111dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        }
7121dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        ALOGV("attaching buffer %p", info.mGraphicBuffer->getNativeBuffer());
7131dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
7141dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        err = surface->attachBuffer(info.mGraphicBuffer->getNativeBuffer());
7151dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        if (err != OK) {
7161dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            ALOGE("failed to attach buffer %p to the new surface: %s (%d)",
7171dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                    info.mGraphicBuffer->getNativeBuffer(),
7181dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                    strerror(-err), -err);
7191dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            return err;
7201dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        }
7211dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
7221dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
7231dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    // cancel undequeued buffers to new surface
724054219874873b41f1c815552987c10465c34ba2bLajos Molnar    if (!storingMetadataInDecodedBuffers() || mLegacyAdaptiveExperiment) {
7251dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        for (size_t i = 0; i < buffers.size(); ++i) {
72615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            BufferInfo &info = buffers.editItemAt(i);
7271dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            if (info.mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
7281dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                ALOGV("canceling buffer %p", info.mGraphicBuffer->getNativeBuffer());
7291dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                err = nativeWindow->cancelBuffer(
73015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                        nativeWindow, info.mGraphicBuffer->getNativeBuffer(), info.mFenceFd);
73115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info.mFenceFd = -1;
7321dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                if (err != OK) {
7331dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                    ALOGE("failed to cancel buffer %p to the new surface: %s (%d)",
7341dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                            info.mGraphicBuffer->getNativeBuffer(),
7351dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                            strerror(-err), -err);
7361dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                    return err;
7371dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar                }
7381dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            }
7391dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        }
7401dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        // disallow further allocation
7411dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        (void)surface->getIGraphicBufferProducer()->allowAllocation(false);
7421dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
7431dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
744484979b1ab25aa41f503bd846323ab93b46d37e5Lajos Molnar    // push blank buffers to previous window if requested
745484979b1ab25aa41f503bd846323ab93b46d37e5Lajos Molnar    if (mFlags & kFlagPushBlankBuffersToNativeWindowOnShutdown) {
746484979b1ab25aa41f503bd846323ab93b46d37e5Lajos Molnar        pushBlankBuffersToNativeWindow(mNativeWindow.get());
747484979b1ab25aa41f503bd846323ab93b46d37e5Lajos Molnar    }
748484979b1ab25aa41f503bd846323ab93b46d37e5Lajos Molnar
7491dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    mNativeWindow = nativeWindow;
7501dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    return OK;
7511dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar}
7521dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
753f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::allocateBuffersOnPort(OMX_U32 portIndex) {
754f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(portIndex == kPortIndexInput || portIndex == kPortIndexOutput);
755f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(mDealer[portIndex] == NULL);
757f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(mBuffers[portIndex].isEmpty());
758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
7595778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err;
760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mNativeWindow != NULL && portIndex == kPortIndexOutput) {
761054219874873b41f1c815552987c10465c34ba2bLajos Molnar        if (storingMetadataInDecodedBuffers()) {
762054219874873b41f1c815552987c10465c34ba2bLajos Molnar            err = allocateOutputMetadataBuffers();
763054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        } else {
764054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar            err = allocateOutputBuffersFromNativeWindow();
765054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        }
7665778822d86b0337407514b9372562b86edfa91cdAndreas Huber    } else {
7675778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMX_PARAM_PORTDEFINITIONTYPE def;
7685778822d86b0337407514b9372562b86edfa91cdAndreas Huber        InitOMXParams(&def);
7695778822d86b0337407514b9372562b86edfa91cdAndreas Huber        def.nPortIndex = portIndex;
770f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
7715778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = mOMX->getParameter(
7725778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
773f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
7745778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err == OK) {
775054219874873b41f1c815552987c10465c34ba2bLajos Molnar            MetadataBufferType type =
776054219874873b41f1c815552987c10465c34ba2bLajos Molnar                portIndex == kPortIndexOutput ? mOutputMetadataType : mInputMetadataType;
777054219874873b41f1c815552987c10465c34ba2bLajos Molnar            int32_t bufSize = def.nBufferSize;
778054219874873b41f1c815552987c10465c34ba2bLajos Molnar            if (type == kMetadataBufferTypeGrallocSource) {
779054219874873b41f1c815552987c10465c34ba2bLajos Molnar                bufSize = sizeof(VideoGrallocMetadata);
780054219874873b41f1c815552987c10465c34ba2bLajos Molnar            } else if (type == kMetadataBufferTypeANWBuffer) {
781054219874873b41f1c815552987c10465c34ba2bLajos Molnar                bufSize = sizeof(VideoNativeMetadata);
782054219874873b41f1c815552987c10465c34ba2bLajos Molnar            }
783054219874873b41f1c815552987c10465c34ba2bLajos Molnar
784054219874873b41f1c815552987c10465c34ba2bLajos Molnar            // If using gralloc or native source input metadata buffers, allocate largest
785054219874873b41f1c815552987c10465c34ba2bLajos Molnar            // metadata size as we prefer to generate native source metadata, but component
7865fb8b2987ab96ad65dc4b046616607ece16d6fb3Lajos Molnar            // may require gralloc source. For camera source, allocate at least enough
7875fb8b2987ab96ad65dc4b046616607ece16d6fb3Lajos Molnar            // size for native metadata buffers.
788054219874873b41f1c815552987c10465c34ba2bLajos Molnar            int32_t allottedSize = bufSize;
7895fb8b2987ab96ad65dc4b046616607ece16d6fb3Lajos Molnar            if (portIndex == kPortIndexInput && type >= kMetadataBufferTypeGrallocSource) {
790054219874873b41f1c815552987c10465c34ba2bLajos Molnar                bufSize = max(sizeof(VideoGrallocMetadata), sizeof(VideoNativeMetadata));
7915fb8b2987ab96ad65dc4b046616607ece16d6fb3Lajos Molnar            } else if (portIndex == kPortIndexInput && type == kMetadataBufferTypeCameraSource) {
7925fb8b2987ab96ad65dc4b046616607ece16d6fb3Lajos Molnar                bufSize = max(bufSize, (int32_t)sizeof(VideoNativeMetadata));
793054219874873b41f1c815552987c10465c34ba2bLajos Molnar            }
794054219874873b41f1c815552987c10465c34ba2bLajos Molnar
795054219874873b41f1c815552987c10465c34ba2bLajos Molnar            ALOGV("[%s] Allocating %u buffers of size %d/%d (from %u using %s) on %s port",
7965778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    mComponentName.c_str(),
797054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    def.nBufferCountActual, bufSize, allottedSize, def.nBufferSize, asString(type),
7985778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    portIndex == kPortIndexInput ? "input" : "output");
799f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
800054219874873b41f1c815552987c10465c34ba2bLajos Molnar            size_t totalSize = def.nBufferCountActual * bufSize;
8015778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mDealer[portIndex] = new MemoryDealer(totalSize, "ACodec");
802f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
803054219874873b41f1c815552987c10465c34ba2bLajos Molnar            for (OMX_U32 i = 0; i < def.nBufferCountActual && err == OK; ++i) {
804054219874873b41f1c815552987c10465c34ba2bLajos Molnar                sp<IMemory> mem = mDealer[portIndex]->allocate(bufSize);
8055581770ee0dde70e2e9c50533be35e537a5800efChong Zhang                if (mem == NULL || mem->pointer() == NULL) {
806777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    return NO_MEMORY;
807777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                }
808f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
809ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                BufferInfo info;
810ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                info.mStatus = BufferInfo::OWNED_BY_US;
81115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info.mFenceFd = -1;
81290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                info.mRenderInfo = NULL;
813f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
814afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber                uint32_t requiresAllocateBufferBit =
815afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber                    (portIndex == kPortIndexInput)
816afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber                        ? OMXCodec::kRequiresAllocateBufferOnInputPorts
817afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber                        : OMXCodec::kRequiresAllocateBufferOnOutputPorts;
8181065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
819308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                if ((portIndex == kPortIndexInput && (mFlags & kFlagIsSecure))
820054219874873b41f1c815552987c10465c34ba2bLajos Molnar                        || (portIndex == kPortIndexOutput && usingMetadataOnEncoderOutput())) {
821ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                    mem.clear();
822ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
823ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                    void *ptr;
824ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                    err = mOMX->allocateBuffer(
825054219874873b41f1c815552987c10465c34ba2bLajos Molnar                            mNode, portIndex, bufSize, &info.mBufferID,
826ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                            &ptr);
827ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
828308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                    info.mData = new ABuffer(ptr, bufSize);
829ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                } else if (mQuirks & requiresAllocateBufferBit) {
8305778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    err = mOMX->allocateBufferWithBackup(
831054219874873b41f1c815552987c10465c34ba2bLajos Molnar                            mNode, portIndex, mem, &info.mBufferID, allottedSize);
8325778822d86b0337407514b9372562b86edfa91cdAndreas Huber                } else {
833054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    err = mOMX->useBuffer(mNode, portIndex, mem, &info.mBufferID, allottedSize);
834ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                }
835ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
836ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                if (mem != NULL) {
837054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    info.mData = new ABuffer(mem->pointer(), bufSize);
838054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    if (type == kMetadataBufferTypeANWBuffer) {
839054219874873b41f1c815552987c10465c34ba2bLajos Molnar                        ((VideoNativeMetadata *)mem->pointer())->nFenceFd = -1;
840054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    }
8415778822d86b0337407514b9372562b86edfa91cdAndreas Huber                }
8421065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
8435778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mBuffers[portIndex].push(info);
8441065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            }
8451065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        }
8465778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
847f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
8485778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
8495778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
8505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
851f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
8525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    sp<AMessage> notify = mNotify->dup();
853d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar    notify->setInt32("what", CodecBase::kWhatBuffersAllocated);
8545778822d86b0337407514b9372562b86edfa91cdAndreas Huber
8555778822d86b0337407514b9372562b86edfa91cdAndreas Huber    notify->setInt32("portIndex", portIndex);
856eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
857eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber    sp<PortDescription> desc = new PortDescription;
858eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
8595778822d86b0337407514b9372562b86edfa91cdAndreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
860eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber        const BufferInfo &info = mBuffers[portIndex][i];
8615778822d86b0337407514b9372562b86edfa91cdAndreas Huber
862eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber        desc->addBuffer(info.mBufferID, info.mData);
863f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
864f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
865eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber    notify->setObject("portDesc", desc);
8665778822d86b0337407514b9372562b86edfa91cdAndreas Huber    notify->post();
8675778822d86b0337407514b9372562b86edfa91cdAndreas Huber
868f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
869f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
870f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
8711dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnarstatus_t ACodec::setupNativeWindowSizeFormatAndUsage(ANativeWindow *nativeWindow /* nonnull */) {
8721dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    OMX_PARAM_PORTDEFINITIONTYPE def;
8731dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    InitOMXParams(&def);
8741dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    def.nPortIndex = kPortIndexOutput;
8751dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
8761dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    status_t err = mOMX->getParameter(
8771dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
8781dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
8791dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err != OK) {
8801dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return err;
8811dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
8821dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
8831dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    OMX_U32 usage = 0;
8841dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    err = mOMX->getGraphicBufferUsage(mNode, kPortIndexOutput, &usage);
8851dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err != 0) {
8861dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        ALOGW("querying usage flags from OMX IL component failed: %d", err);
8871dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        // XXX: Currently this error is logged, but not fatal.
8881dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        usage = 0;
8891dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
8901dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    int omxUsage = usage;
8911dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
8921dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (mFlags & kFlagIsGrallocUsageProtected) {
8931dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        usage |= GRALLOC_USAGE_PROTECTED;
8941dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
8951dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
896b2d0b487efd40700199852c9a18b369b1651f15bLajos Molnar    usage |= GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_EXTERNAL_DISP;
897b2d0b487efd40700199852c9a18b369b1651f15bLajos Molnar
8981dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    ALOGV("gralloc usage: %#x(OMX) => %#x(ACodec)", omxUsage, usage);
8991dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    return setNativeWindowSizeFormatAndUsage(
9001dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            nativeWindow,
9011dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            def.format.video.nFrameWidth,
9021dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            def.format.video.nFrameHeight,
9031dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            def.format.video.eColorFormat,
9041dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            mRotationDegrees,
9051dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            usage);
9061dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar}
9071dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
9081dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnarstatus_t ACodec::configureOutputBuffersFromNativeWindow(
9091dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        OMX_U32 *bufferCount, OMX_U32 *bufferSize,
9101dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        OMX_U32 *minUndequeuedBuffers) {
9111dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    OMX_PARAM_PORTDEFINITIONTYPE def;
9121dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    InitOMXParams(&def);
9131dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    def.nPortIndex = kPortIndexOutput;
9141dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
9151dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    status_t err = mOMX->getParameter(
9161dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
9171dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
9181dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err == OK) {
9191dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        err = setupNativeWindowSizeFormatAndUsage(mNativeWindow.get());
9201dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
9211dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    if (err != OK) {
9221dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        return err;
9231dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar    }
9241dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
925ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad    // Exits here for tunneled video playback codecs -- i.e. skips native window
926ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad    // buffer allocation step as this is managed by the tunneled OMX omponent
927ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad    // itself and explicitly sets def.nBufferCountActual to 0.
928ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad    if (mTunneled) {
929ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        ALOGV("Tunneled Playback: skipping native window buffer allocation.");
930ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        def.nBufferCountActual = 0;
931ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        err = mOMX->setParameter(
932ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
933ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad
934ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        *minUndequeuedBuffers = 0;
935ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        *bufferCount = 0;
936ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        *bufferSize = 0;
937ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        return err;
938ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad    }
939ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad
940054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    *minUndequeuedBuffers = 0;
941258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    err = mNativeWindow->query(
942258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            mNativeWindow.get(), NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS,
943054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar            (int *)minUndequeuedBuffers);
944258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
945258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    if (err != 0) {
94629357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS query failed: %s (%d)",
947258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                strerror(-err), -err);
948258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        return err;
949258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
950258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
951e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    // FIXME: assume that surface is controlled by app (native window
952e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    // returns the number for the case when surface is not controlled by app)
9531faa41704e0b976e546321effcb09a85767d51baLajos Molnar    // FIXME2: This means that minUndeqeueudBufs can be 1 larger than reported
9541faa41704e0b976e546321effcb09a85767d51baLajos Molnar    // For now, try to allocate 1 more buffer, but don't fail if unsuccessful
955e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar
956e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    // Use conservative allocation while also trying to reduce starvation
957e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    //
958e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    // 1. allocate at least nBufferCountMin + minUndequeuedBuffers - that is the
959e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    //    minimum needed for the consumer to be able to work
960e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    // 2. try to allocate two (2) additional buffers to reduce starvation from
961e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    //    the consumer
9621faa41704e0b976e546321effcb09a85767d51baLajos Molnar    //    plus an extra buffer to account for incorrect minUndequeuedBufs
9631faa41704e0b976e546321effcb09a85767d51baLajos Molnar    for (OMX_U32 extraBuffers = 2 + 1; /* condition inside loop */; extraBuffers--) {
964e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar        OMX_U32 newBufferCount =
965e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar            def.nBufferCountMin + *minUndequeuedBuffers + extraBuffers;
966258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        def.nBufferCountActual = newBufferCount;
967258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        err = mOMX->setParameter(
968258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
969258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
970e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar        if (err == OK) {
971e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar            *minUndequeuedBuffers += extraBuffers;
972e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar            break;
973e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar        }
974e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar
975609b815a3131d22da38b2f452faa9f89daad4039Andy Hung        ALOGW("[%s] setting nBufferCountActual to %u failed: %d",
976e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar                mComponentName.c_str(), newBufferCount, err);
977e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar        /* exit condition */
978e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar        if (extraBuffers == 0) {
979258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            return err;
980258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        }
981258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
982258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
983f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = native_window_set_buffer_count(
984f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNativeWindow.get(), def.nBufferCountActual);
985f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
986f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
98729357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("native_window_set_buffer_count failed: %s (%d)", strerror(-err),
988f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                -err);
989f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
990f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
991f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
992054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    *bufferCount = def.nBufferCountActual;
993054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    *bufferSize =  def.nBufferSize;
994054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    return err;
995054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar}
996054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
997054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnarstatus_t ACodec::allocateOutputBuffersFromNativeWindow() {
998054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    OMX_U32 bufferCount, bufferSize, minUndequeuedBuffers;
999054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    status_t err = configureOutputBuffersFromNativeWindow(
1000054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar            &bufferCount, &bufferSize, &minUndequeuedBuffers);
1001054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    if (err != 0)
1002054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        return err;
1003e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    mNumUndequeuedBuffers = minUndequeuedBuffers;
1004054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1005054219874873b41f1c815552987c10465c34ba2bLajos Molnar    if (!storingMetadataInDecodedBuffers()) {
10063298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia        static_cast<Surface*>(mNativeWindow.get())
10073298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia                ->getIGraphicBufferProducer()->allowAllocation(true);
10083298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia    }
10093298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia
1010609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    ALOGV("[%s] Allocating %u buffers from a native window of size %u on "
1011f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         "output port",
1012054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar         mComponentName.c_str(), bufferCount, bufferSize);
1013f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1014f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Dequeue buffers and send them to OMX
1015054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    for (OMX_U32 i = 0; i < bufferCount; i++) {
10168ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev        ANativeWindowBuffer *buf;
101715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        int fenceFd;
101815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf, &fenceFd);
1019f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != 0) {
102029357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("dequeueBuffer failed: %s (%d)", strerror(-err), -err);
1021f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1022f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1023f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1024f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(buf, false));
102574006804065941841883c4b46ee785070164023fJamie Gennis        BufferInfo info;
102674006804065941841883c4b46ee785070164023fJamie Gennis        info.mStatus = BufferInfo::OWNED_BY_US;
102715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        info.mFenceFd = fenceFd;
102815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        info.mIsReadFence = false;
102990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        info.mRenderInfo = NULL;
1030054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        info.mData = new ABuffer(NULL /* data */, bufferSize /* capacity */);
103174006804065941841883c4b46ee785070164023fJamie Gennis        info.mGraphicBuffer = graphicBuffer;
103274006804065941841883c4b46ee785070164023fJamie Gennis        mBuffers[kPortIndexOutput].push(info);
103374006804065941841883c4b46ee785070164023fJamie Gennis
1034f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id bufferId;
1035f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        err = mOMX->useGraphicBuffer(mNode, kPortIndexOutput, graphicBuffer,
1036f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &bufferId);
1037f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != 0) {
1038609b815a3131d22da38b2f452faa9f89daad4039Andy Hung            ALOGE("registering GraphicBuffer %u with OMX IL component failed: "
103974006804065941841883c4b46ee785070164023fJamie Gennis                 "%d", i, err);
1040f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1041f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1042f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
104374006804065941841883c4b46ee785070164023fJamie Gennis        mBuffers[kPortIndexOutput].editItemAt(i).mBufferID = bufferId;
104474006804065941841883c4b46ee785070164023fJamie Gennis
1045609b815a3131d22da38b2f452faa9f89daad4039Andy Hung        ALOGV("[%s] Registered graphic buffer with ID %u (pointer = %p)",
1046f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             mComponentName.c_str(),
1047f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             bufferId, graphicBuffer.get());
1048f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1049f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1050f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 cancelStart;
1051f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 cancelEnd;
1052f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1053f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
1054f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // If an error occurred while dequeuing we need to cancel any buffers
1055f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // that were dequeued.
1056f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        cancelStart = 0;
105774006804065941841883c4b46ee785070164023fJamie Gennis        cancelEnd = mBuffers[kPortIndexOutput].size();
1058f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
1059054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        // Return the required minimum undequeued buffers to the native window.
1060054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        cancelStart = bufferCount - minUndequeuedBuffers;
1061054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        cancelEnd = bufferCount;
1062f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1063f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1064f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (OMX_U32 i = cancelStart; i < cancelEnd; i++) {
1065f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[kPortIndexOutput].editItemAt(i);
10663fb9f68dea5d991288f0ea8037742b50c7df5767Wei Jia        status_t error = cancelBufferToNativeWindow(info);
10673fb9f68dea5d991288f0ea8037742b50c7df5767Wei Jia        if (err == 0) {
10683fb9f68dea5d991288f0ea8037742b50c7df5767Wei Jia            err = error;
10693fb9f68dea5d991288f0ea8037742b50c7df5767Wei Jia        }
1070f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1071f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1072054219874873b41f1c815552987c10465c34ba2bLajos Molnar    if (!storingMetadataInDecodedBuffers()) {
10733298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia        static_cast<Surface*>(mNativeWindow.get())
10743298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia                ->getIGraphicBufferProducer()->allowAllocation(false);
10753298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia    }
10763298d6fa642c11c5b004bdfc375252ff4c4536e3Wei Jia
1077f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
1078f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1079f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1080054219874873b41f1c815552987c10465c34ba2bLajos Molnarstatus_t ACodec::allocateOutputMetadataBuffers() {
1081054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    OMX_U32 bufferCount, bufferSize, minUndequeuedBuffers;
1082054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    status_t err = configureOutputBuffersFromNativeWindow(
1083054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar            &bufferCount, &bufferSize, &minUndequeuedBuffers);
1084054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    if (err != 0)
1085054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        return err;
1086e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    mNumUndequeuedBuffers = minUndequeuedBuffers;
1087054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1088e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar    ALOGV("[%s] Allocating %u meta buffers on output port",
1089e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar         mComponentName.c_str(), bufferCount);
1090e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1091054219874873b41f1c815552987c10465c34ba2bLajos Molnar    size_t bufSize = mOutputMetadataType == kMetadataBufferTypeANWBuffer ?
1092054219874873b41f1c815552987c10465c34ba2bLajos Molnar            sizeof(struct VideoNativeMetadata) : sizeof(struct VideoGrallocMetadata);
1093054219874873b41f1c815552987c10465c34ba2bLajos Molnar    size_t totalSize = bufferCount * bufSize;
1094e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar    mDealer[kPortIndexOutput] = new MemoryDealer(totalSize, "ACodec");
1095e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1096e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar    // Dequeue buffers and send them to OMX
1097e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar    for (OMX_U32 i = 0; i < bufferCount; i++) {
1098e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        BufferInfo info;
1099e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        info.mStatus = BufferInfo::OWNED_BY_NATIVE_WINDOW;
110015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        info.mFenceFd = -1;
110190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        info.mRenderInfo = NULL;
1102e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        info.mGraphicBuffer = NULL;
1103e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        info.mDequeuedAt = mDequeueCounter;
1104e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1105054219874873b41f1c815552987c10465c34ba2bLajos Molnar        sp<IMemory> mem = mDealer[kPortIndexOutput]->allocate(bufSize);
11065581770ee0dde70e2e9c50533be35e537a5800efChong Zhang        if (mem == NULL || mem->pointer() == NULL) {
1107777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            return NO_MEMORY;
1108777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        }
1109054219874873b41f1c815552987c10465c34ba2bLajos Molnar        if (mOutputMetadataType == kMetadataBufferTypeANWBuffer) {
1110054219874873b41f1c815552987c10465c34ba2bLajos Molnar            ((VideoNativeMetadata *)mem->pointer())->nFenceFd = -1;
1111054219874873b41f1c815552987c10465c34ba2bLajos Molnar        }
1112e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        info.mData = new ABuffer(mem->pointer(), mem->size());
1113e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1114e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        // we use useBuffer for metadata regardless of quirks
1115e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        err = mOMX->useBuffer(
1116cc7cc67349b7a3f498882087aa42ffc05a2daf11Lajos Molnar                mNode, kPortIndexOutput, mem, &info.mBufferID, mem->size());
1117e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1118e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        mBuffers[kPortIndexOutput].push(info);
1119e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1120e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        ALOGV("[%s] allocated meta buffer with ID %u (pointer = %p)",
1121e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar             mComponentName.c_str(), info.mBufferID, mem->pointer());
1122e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar    }
1123e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1124011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar    if (mLegacyAdaptiveExperiment) {
1125e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar        // preallocate and preregister buffers
1126011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        static_cast<Surface *>(mNativeWindow.get())
1127011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                ->getIGraphicBufferProducer()->allowAllocation(true);
1128011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar
1129011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        ALOGV("[%s] Allocating %u buffers from a native window of size %u on "
1130011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar             "output port",
1131011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar             mComponentName.c_str(), bufferCount, bufferSize);
1132011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar
1133011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        // Dequeue buffers then cancel them all
1134011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        for (OMX_U32 i = 0; i < bufferCount; i++) {
1135e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar            BufferInfo *info = &mBuffers[kPortIndexOutput].editItemAt(i);
1136e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar
1137011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            ANativeWindowBuffer *buf;
113815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            int fenceFd;
113915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf, &fenceFd);
1140011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            if (err != 0) {
1141011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                ALOGE("dequeueBuffer failed: %s (%d)", strerror(-err), -err);
1142011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                break;
1143011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            }
1144011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar
1145011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(buf, false));
1146e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar            mOMX->updateGraphicBufferInMeta(
1147e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar                    mNode, kPortIndexOutput, graphicBuffer, info->mBufferID);
1148e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar            info->mStatus = BufferInfo::OWNED_BY_US;
114915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            info->setWriteFence(fenceFd, "allocateOutputMetadataBuffers for legacy");
1150e66c5a42f01e2f445857b54a9d7bbb8ee062edebLajos Molnar            info->mGraphicBuffer = graphicBuffer;
1151011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        }
1152011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar
1153011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        for (OMX_U32 i = 0; i < mBuffers[kPortIndexOutput].size(); i++) {
1154011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            BufferInfo *info = &mBuffers[kPortIndexOutput].editItemAt(i);
1155011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            status_t error = cancelBufferToNativeWindow(info);
1156011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            if (err == OK) {
1157011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                err = error;
1158011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            }
1159011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        }
1160011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar
1161011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar        static_cast<Surface*>(mNativeWindow.get())
1162011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                ->getIGraphicBufferProducer()->allowAllocation(false);
1163011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar    }
1164011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar
1165054219874873b41f1c815552987c10465c34ba2bLajos Molnar    mMetadataBuffersToSubmit = bufferCount - minUndequeuedBuffers;
1166054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    return err;
1167054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar}
1168054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1169054219874873b41f1c815552987c10465c34ba2bLajos Molnarstatus_t ACodec::submitOutputMetadataBuffer() {
1170054219874873b41f1c815552987c10465c34ba2bLajos Molnar    CHECK(storingMetadataInDecodedBuffers());
1171054219874873b41f1c815552987c10465c34ba2bLajos Molnar    if (mMetadataBuffersToSubmit == 0)
1172054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        return OK;
1173054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1174054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    BufferInfo *info = dequeueBufferFromNativeWindow();
11750806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (info == NULL) {
1176054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        return ERROR_IO;
11770806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
1178054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1179609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    ALOGV("[%s] submitting output meta buffer ID %u for graphic buffer %p",
1180054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar          mComponentName.c_str(), info->mBufferID, info->mGraphicBuffer.get());
1181054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1182054219874873b41f1c815552987c10465c34ba2bLajos Molnar    --mMetadataBuffersToSubmit;
118315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    info->checkWriteFence("submitOutputMetadataBuffer");
118415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    status_t err = mOMX->fillBuffer(mNode, info->mBufferID, info->mFenceFd);
118515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    info->mFenceFd = -1;
1186777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (err == OK) {
1187777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1188777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
1189054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1190777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    return err;
1191054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar}
1192054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
119315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnarstatus_t ACodec::waitForFence(int fd, const char *dbg ) {
119415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    status_t res = OK;
119515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (fd >= 0) {
119615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        sp<Fence> fence = new Fence(fd);
119715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        res = fence->wait(IOMX::kFenceTimeoutMs);
119815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        ALOGW_IF(res != OK, "FENCE TIMEOUT for %d in %s", fd, dbg);
119915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
120015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    return res;
120115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar}
120215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
12030806340688c937e7b78c2d89db3809274130df4eLajos Molnar// static
12040806340688c937e7b78c2d89db3809274130df4eLajos Molnarconst char *ACodec::_asString(BufferInfo::Status s) {
12050806340688c937e7b78c2d89db3809274130df4eLajos Molnar    switch (s) {
12060806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::OWNED_BY_US:            return "OUR";
12070806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::OWNED_BY_COMPONENT:     return "COMPONENT";
12080806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::OWNED_BY_UPSTREAM:      return "UPSTREAM";
12090806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::OWNED_BY_DOWNSTREAM:    return "DOWNSTREAM";
12100806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::OWNED_BY_NATIVE_WINDOW: return "SURFACE";
12110806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::UNRECOGNIZED:           return "UNRECOGNIZED";
12120806340688c937e7b78c2d89db3809274130df4eLajos Molnar        default:                                 return "?";
12130806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
12140806340688c937e7b78c2d89db3809274130df4eLajos Molnar}
12150806340688c937e7b78c2d89db3809274130df4eLajos Molnar
12160806340688c937e7b78c2d89db3809274130df4eLajos Molnarvoid ACodec::dumpBuffers(OMX_U32 portIndex) {
12170806340688c937e7b78c2d89db3809274130df4eLajos Molnar    CHECK(portIndex == kPortIndexInput || portIndex == kPortIndexOutput);
12180806340688c937e7b78c2d89db3809274130df4eLajos Molnar    ALOGI("[%s] %s port has %zu buffers:", mComponentName.c_str(),
12190806340688c937e7b78c2d89db3809274130df4eLajos Molnar            portIndex == kPortIndexInput ? "input" : "output", mBuffers[portIndex].size());
12200806340688c937e7b78c2d89db3809274130df4eLajos Molnar    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
12210806340688c937e7b78c2d89db3809274130df4eLajos Molnar        const BufferInfo &info = mBuffers[portIndex][i];
12220806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGI("  slot %2zu: #%8u %p/%p %s(%d) dequeued:%u",
12230806340688c937e7b78c2d89db3809274130df4eLajos Molnar                i, info.mBufferID, info.mGraphicBuffer.get(),
12240806340688c937e7b78c2d89db3809274130df4eLajos Molnar                info.mGraphicBuffer == NULL ? NULL : info.mGraphicBuffer->getNativeBuffer(),
12250806340688c937e7b78c2d89db3809274130df4eLajos Molnar                _asString(info.mStatus), info.mStatus, info.mDequeuedAt);
12260806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
12270806340688c937e7b78c2d89db3809274130df4eLajos Molnar}
12280806340688c937e7b78c2d89db3809274130df4eLajos Molnar
1229f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::cancelBufferToNativeWindow(BufferInfo *info) {
1230f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
1231f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1232609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    ALOGV("[%s] Calling cancelBuffer on buffer %u",
1233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mComponentName.c_str(), info->mBufferID);
1234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
123515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    info->checkWriteFence("cancelBufferToNativeWindow");
1236f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int err = mNativeWindow->cancelBuffer(
123715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        mNativeWindow.get(), info->mGraphicBuffer.get(), info->mFenceFd);
123815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    info->mFenceFd = -1;
1239f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
12403fb9f68dea5d991288f0ea8037742b50c7df5767Wei Jia    ALOGW_IF(err != 0, "[%s] can not return buffer %u to native window",
12413fb9f68dea5d991288f0ea8037742b50c7df5767Wei Jia            mComponentName.c_str(), info->mBufferID);
12420806340688c937e7b78c2d89db3809274130df4eLajos Molnar    // change ownership even if cancelBuffer fails
1243f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_NATIVE_WINDOW;
1244f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
12453fb9f68dea5d991288f0ea8037742b50c7df5767Wei Jia    return err;
1246f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1247f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
124890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnarvoid ACodec::updateRenderInfoForDequeuedBuffer(
124990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        ANativeWindowBuffer *buf, int fenceFd, BufferInfo *info) {
125090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
125190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    info->mRenderInfo =
125290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        mRenderTracker.updateInfoForDequeuedBuffer(
125390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                buf, fenceFd, info - &mBuffers[kPortIndexOutput][0]);
125490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
125590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    // check for any fences already signaled
125690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    notifyOfRenderedFrames(false /* dropIncomplete */, info->mRenderInfo);
125790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar}
125890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
125990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnarvoid ACodec::onFrameRendered(int64_t mediaTimeUs, nsecs_t systemNano) {
126090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    if (mRenderTracker.onFrameRendered(mediaTimeUs, systemNano) != OK) {
126190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        mRenderTracker.dumpRenderQueue();
126290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    }
126390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar}
126490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
126590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnarvoid ACodec::notifyOfRenderedFrames(bool dropIncomplete, FrameRenderTracker::Info *until) {
126690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    sp<AMessage> msg = mNotify->dup();
126790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    msg->setInt32("what", CodecBase::kWhatOutputFramesRendered);
126890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    std::list<FrameRenderTracker::Info> done =
126990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        mRenderTracker.checkFencesAndGetRenderedFrames(until, dropIncomplete);
127090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
127190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    // unlink untracked frames
127290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    for (std::list<FrameRenderTracker::Info>::const_iterator it = done.cbegin();
127390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar            it != done.cend(); ++it) {
1274604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar        ssize_t index = it->getIndex();
1275604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar        if (index >= 0 && (size_t)index < mBuffers[kPortIndexOutput].size()) {
1276604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar            mBuffers[kPortIndexOutput].editItemAt(index).mRenderInfo = NULL;
1277604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar        } else if (index >= 0) {
1278604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar            // THIS SHOULD NEVER HAPPEN
1279604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar            ALOGE("invalid index %zd in %zu", index, mBuffers[kPortIndexOutput].size());
128090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        }
128190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    }
128290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
128390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    if (MediaCodec::CreateFramesRenderedMessage(done, msg)) {
128490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        msg->post();
128590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    }
128690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar}
128790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
1288f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BufferInfo *ACodec::dequeueBufferFromNativeWindow() {
12898ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev    ANativeWindowBuffer *buf;
1290054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    CHECK(mNativeWindow.get() != NULL);
1291ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad
1292ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad    if (mTunneled) {
1293ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        ALOGW("dequeueBufferFromNativeWindow() should not be called in tunnel"
1294ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad              " video playback mode mode!");
1295ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad        return NULL;
1296ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad    }
1297ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad
129815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    int fenceFd = -1;
1299dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar    do {
130015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        status_t err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf, &fenceFd);
130115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        if (err != 0) {
130215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            ALOGE("dequeueBuffer failed: %s(%d).", asString(err), err);
1303dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            return NULL;
1304dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        }
1305f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1306dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        bool stale = false;
1307dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        for (size_t i = mBuffers[kPortIndexOutput].size(); i-- > 0;) {
1308dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            BufferInfo *info = &mBuffers[kPortIndexOutput].editItemAt(i);
1309dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar
1310dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            if (info->mGraphicBuffer != NULL &&
131190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                    info->mGraphicBuffer->handle == buf->handle) {
1312dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                // Since consumers can attach buffers to BufferQueues, it is possible
1313dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                // that a known yet stale buffer can return from a surface that we
1314dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                // once used.  We can simply ignore this as we have already dequeued
1315dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                // this buffer properly.  NOTE: this does not eliminate all cases,
1316dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                // e.g. it is possible that we have queued the valid buffer to the
1317dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                // NW, and a stale copy of the same buffer gets dequeued - which will
1318dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                // be treated as the valid buffer by ACodec.
1319dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                if (info->mStatus != BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1320dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                    ALOGI("dequeued stale buffer %p. discarding", buf);
1321dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                    stale = true;
1322dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                    break;
1323dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                }
132490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
1325dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                ALOGV("dequeued buffer %p", info->mGraphicBuffer->getNativeBuffer());
1326dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                info->mStatus = BufferInfo::OWNED_BY_US;
132715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info->setWriteFence(fenceFd, "dequeueBufferFromNativeWindow");
132890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                updateRenderInfoForDequeuedBuffer(buf, fenceFd, info);
1329dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar                return info;
1330dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            }
1331dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        }
1332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1333dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        // It is also possible to receive a previously unregistered buffer
1334dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        // in non-meta mode. These should be treated as stale buffers. The
1335dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        // same is possible in meta mode, in which case, it will be treated
1336dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        // as a normal buffer, which is not desirable.
1337dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        // TODO: fix this.
1338054219874873b41f1c815552987c10465c34ba2bLajos Molnar        if (!stale && (!storingMetadataInDecodedBuffers() || mLegacyAdaptiveExperiment)) {
1339dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            ALOGI("dequeued unrecognized (stale) buffer %p. discarding", buf);
1340dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            stale = true;
1341dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        }
1342dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        if (stale) {
1343dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            // TODO: detach stale buffer, but there is no API yet to do it.
1344dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            buf = NULL;
1345f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1346dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar    } while (buf == NULL);
1347054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1348dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar    // get oldest undequeued buffer
1349dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar    BufferInfo *oldest = NULL;
1350dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar    for (size_t i = mBuffers[kPortIndexOutput].size(); i-- > 0;) {
1351dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar        BufferInfo *info =
1352dd248abd2c3e5e82b0049d40c7e9fb741fd4540cLajos Molnar            &mBuffers[kPortIndexOutput].editItemAt(i);
1353054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW &&
1354054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar            (oldest == NULL ||
1355054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar             // avoid potential issues from counter rolling over
1356054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar             mDequeueCounter - info->mDequeuedAt >
1357054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                    mDequeueCounter - oldest->mDequeuedAt)) {
1358054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar            oldest = info;
1359054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        }
1360054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    }
1361054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
13620806340688c937e7b78c2d89db3809274130df4eLajos Molnar    // it is impossible dequeue a buffer when there are no buffers with ANW
13630806340688c937e7b78c2d89db3809274130df4eLajos Molnar    CHECK(oldest != NULL);
13640806340688c937e7b78c2d89db3809274130df4eLajos Molnar    // it is impossible to dequeue an unknown buffer in non-meta mode, as the
13650806340688c937e7b78c2d89db3809274130df4eLajos Molnar    // while loop above does not complete
1366054219874873b41f1c815552987c10465c34ba2bLajos Molnar    CHECK(storingMetadataInDecodedBuffers());
1367054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
13680806340688c937e7b78c2d89db3809274130df4eLajos Molnar    // discard buffer in LRU info and replace with new buffer
13690806340688c937e7b78c2d89db3809274130df4eLajos Molnar    oldest->mGraphicBuffer = new GraphicBuffer(buf, false);
13700806340688c937e7b78c2d89db3809274130df4eLajos Molnar    oldest->mStatus = BufferInfo::OWNED_BY_US;
137115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    oldest->setWriteFence(fenceFd, "dequeueBufferFromNativeWindow for oldest");
137290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    mRenderTracker.untrackFrame(oldest->mRenderInfo);
137390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    oldest->mRenderInfo = NULL;
1374d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar
13750806340688c937e7b78c2d89db3809274130df4eLajos Molnar    mOMX->updateGraphicBufferInMeta(
13760806340688c937e7b78c2d89db3809274130df4eLajos Molnar            mNode, kPortIndexOutput, oldest->mGraphicBuffer,
13770806340688c937e7b78c2d89db3809274130df4eLajos Molnar            oldest->mBufferID);
1378054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1379054219874873b41f1c815552987c10465c34ba2bLajos Molnar    if (mOutputMetadataType == kMetadataBufferTypeGrallocSource) {
1380054219874873b41f1c815552987c10465c34ba2bLajos Molnar        VideoGrallocMetadata *grallocMeta =
1381054219874873b41f1c815552987c10465c34ba2bLajos Molnar            reinterpret_cast<VideoGrallocMetadata *>(oldest->mData->base());
1382054219874873b41f1c815552987c10465c34ba2bLajos Molnar        ALOGV("replaced oldest buffer #%u with age %u (%p/%p stored in %p)",
1383054219874873b41f1c815552987c10465c34ba2bLajos Molnar                (unsigned)(oldest - &mBuffers[kPortIndexOutput][0]),
1384054219874873b41f1c815552987c10465c34ba2bLajos Molnar                mDequeueCounter - oldest->mDequeuedAt,
13859847fcefb183e1cb09eb48e17a09577392b0e8f4Lajos Molnar                grallocMeta->pHandle,
1386054219874873b41f1c815552987c10465c34ba2bLajos Molnar                oldest->mGraphicBuffer->handle, oldest->mData->base());
1387054219874873b41f1c815552987c10465c34ba2bLajos Molnar    } else if (mOutputMetadataType == kMetadataBufferTypeANWBuffer) {
1388054219874873b41f1c815552987c10465c34ba2bLajos Molnar        VideoNativeMetadata *nativeMeta =
1389054219874873b41f1c815552987c10465c34ba2bLajos Molnar            reinterpret_cast<VideoNativeMetadata *>(oldest->mData->base());
1390054219874873b41f1c815552987c10465c34ba2bLajos Molnar        ALOGV("replaced oldest buffer #%u with age %u (%p/%p stored in %p)",
1391054219874873b41f1c815552987c10465c34ba2bLajos Molnar                (unsigned)(oldest - &mBuffers[kPortIndexOutput][0]),
1392054219874873b41f1c815552987c10465c34ba2bLajos Molnar                mDequeueCounter - oldest->mDequeuedAt,
1393054219874873b41f1c815552987c10465c34ba2bLajos Molnar                nativeMeta->pBuffer,
1394054219874873b41f1c815552987c10465c34ba2bLajos Molnar                oldest->mGraphicBuffer->getNativeBuffer(), oldest->mData->base());
1395054219874873b41f1c815552987c10465c34ba2bLajos Molnar    }
1396f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
139790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    updateRenderInfoForDequeuedBuffer(buf, fenceFd, oldest);
13980806340688c937e7b78c2d89db3809274130df4eLajos Molnar    return oldest;
1399f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1400f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1401f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::freeBuffersOnPort(OMX_U32 portIndex) {
1402777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    status_t err = OK;
1403938e2b34b16c3c1fd29c753eeb53ee95a2b2e2b3Nick Kralevich    for (size_t i = mBuffers[portIndex].size(); i > 0;) {
1404938e2b34b16c3c1fd29c753eeb53ee95a2b2e2b3Nick Kralevich        i--;
1405777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        status_t err2 = freeBuffer(portIndex, i);
1406777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        if (err == OK) {
1407777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            err = err2;
1408777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        }
1409f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1410f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
14110806340688c937e7b78c2d89db3809274130df4eLajos Molnar    // clear mDealer even on an error
1412f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDealer[portIndex].clear();
1413777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    return err;
1414f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1415f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1416349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huberstatus_t ACodec::freeOutputBuffersNotOwnedByComponent() {
1417777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    status_t err = OK;
1418938e2b34b16c3c1fd29c753eeb53ee95a2b2e2b3Nick Kralevich    for (size_t i = mBuffers[kPortIndexOutput].size(); i > 0;) {
1419938e2b34b16c3c1fd29c753eeb53ee95a2b2e2b3Nick Kralevich        i--;
1420f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info =
1421f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &mBuffers[kPortIndexOutput].editItemAt(i);
1422f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
14232ded8b53014602d25b20bade8ce46db95a8da4b5Lajos Molnar        // At this time some buffers may still be with the component
14242ded8b53014602d25b20bade8ce46db95a8da4b5Lajos Molnar        // or being drained.
14252ded8b53014602d25b20bade8ce46db95a8da4b5Lajos Molnar        if (info->mStatus != BufferInfo::OWNED_BY_COMPONENT &&
14262ded8b53014602d25b20bade8ce46db95a8da4b5Lajos Molnar            info->mStatus != BufferInfo::OWNED_BY_DOWNSTREAM) {
1427777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            status_t err2 = freeBuffer(kPortIndexOutput, i);
1428777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            if (err == OK) {
1429777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                err = err2;
1430777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            }
1431f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1432f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1433f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1434777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    return err;
1435f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1436f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1437f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::freeBuffer(OMX_U32 portIndex, size_t i) {
1438f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
14390806340688c937e7b78c2d89db3809274130df4eLajos Molnar    status_t err = OK;
1440f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
144115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    // there should not be any fences in the metadata
144215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    MetadataBufferType type =
144315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        portIndex == kPortIndexOutput ? mOutputMetadataType : mInputMetadataType;
144415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (type == kMetadataBufferTypeANWBuffer && info->mData != NULL
144515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            && info->mData->size() >= sizeof(VideoNativeMetadata)) {
144615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        int fenceFd = ((VideoNativeMetadata *)info->mData->data())->nFenceFd;
144715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        if (fenceFd >= 0) {
144815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            ALOGW("unreleased fence (%d) in %s metadata buffer %zu",
144915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    fenceFd, portIndex == kPortIndexInput ? "input" : "output", i);
145015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        }
145115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
145215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
14530806340688c937e7b78c2d89db3809274130df4eLajos Molnar    switch (info->mStatus) {
14540806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::OWNED_BY_US:
14550806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (portIndex == kPortIndexOutput && mNativeWindow != NULL) {
14560806340688c937e7b78c2d89db3809274130df4eLajos Molnar                (void)cancelBufferToNativeWindow(info);
14570806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
14580806340688c937e7b78c2d89db3809274130df4eLajos Molnar            // fall through
1459f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
14600806340688c937e7b78c2d89db3809274130df4eLajos Molnar        case BufferInfo::OWNED_BY_NATIVE_WINDOW:
14610806340688c937e7b78c2d89db3809274130df4eLajos Molnar            err = mOMX->freeBuffer(mNode, portIndex, info->mBufferID);
14620806340688c937e7b78c2d89db3809274130df4eLajos Molnar            break;
14630806340688c937e7b78c2d89db3809274130df4eLajos Molnar
14640806340688c937e7b78c2d89db3809274130df4eLajos Molnar        default:
14650806340688c937e7b78c2d89db3809274130df4eLajos Molnar            ALOGE("trying to free buffer not owned by us or ANW (%d)", info->mStatus);
14660806340688c937e7b78c2d89db3809274130df4eLajos Molnar            err = FAILED_TRANSACTION;
14670806340688c937e7b78c2d89db3809274130df4eLajos Molnar            break;
1468f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1469f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
147015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (info->mFenceFd >= 0) {
147115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        ::close(info->mFenceFd);
147215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
147315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
1474604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar    if (portIndex == kPortIndexOutput) {
1475604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar        mRenderTracker.untrackFrame(info->mRenderInfo, i);
1476604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar        info->mRenderInfo = NULL;
1477604bb9ea6e9bec763ae231330066ecffa90a2786Lajos Molnar    }
147890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
1479777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    // remove buffer even if mOMX->freeBuffer fails
1480f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mBuffers[portIndex].removeAt(i);
1481777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    return err;
1482f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1483f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1484f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BufferInfo *ACodec::findBufferByID(
14850806340688c937e7b78c2d89db3809274130df4eLajos Molnar        uint32_t portIndex, IOMX::buffer_id bufferID, ssize_t *index) {
1486f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
1487f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
1488f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1489f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mBufferID == bufferID) {
1490f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (index != NULL) {
1491f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                *index = i;
1492f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1493f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return info;
1494f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1495f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1496f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1497777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    ALOGE("Could not find buffer with ID %u", bufferID);
1498f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return NULL;
1499f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1500f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
15015778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setComponentRole(
1502f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        bool isEncoder, const char *mime) {
1503f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    struct MimeToRole {
1504f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime;
1505f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *decoderRole;
1506f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *encoderRole;
1507f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
1508f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1509f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    static const MimeToRole kMimeToRole[] = {
1510f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_MPEG,
1511f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.mp3", "audio_encoder.mp3" },
15122944eca607304a095ea43ba2b8f0b9de61249f9fAndreas Huber        { MEDIA_MIMETYPE_AUDIO_MPEG_LAYER_I,
15132944eca607304a095ea43ba2b8f0b9de61249f9fAndreas Huber            "audio_decoder.mp1", "audio_encoder.mp1" },
15142944eca607304a095ea43ba2b8f0b9de61249f9fAndreas Huber        { MEDIA_MIMETYPE_AUDIO_MPEG_LAYER_II,
15152944eca607304a095ea43ba2b8f0b9de61249f9fAndreas Huber            "audio_decoder.mp2", "audio_encoder.mp2" },
1516f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_NB,
1517f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.amrnb", "audio_encoder.amrnb" },
1518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_WB,
1519f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.amrwb", "audio_encoder.amrwb" },
1520f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AAC,
1521f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.aac", "audio_encoder.aac" },
1522729de186450f78c099637e1fce743fe531862c52Andreas Huber        { MEDIA_MIMETYPE_AUDIO_VORBIS,
1523729de186450f78c099637e1fce743fe531862c52Andreas Huber            "audio_decoder.vorbis", "audio_encoder.vorbis" },
1524bf927f8ec7979f2b64331c2b2f12a6a5dba05bcaVignesh Venkatasubramanian        { MEDIA_MIMETYPE_AUDIO_OPUS,
1525bf927f8ec7979f2b64331c2b2f12a6a5dba05bcaVignesh Venkatasubramanian            "audio_decoder.opus", "audio_encoder.opus" },
1526c5255ac5b1bd313dcb50159566163b24dce7483fAndreas Huber        { MEDIA_MIMETYPE_AUDIO_G711_MLAW,
1527c5255ac5b1bd313dcb50159566163b24dce7483fAndreas Huber            "audio_decoder.g711mlaw", "audio_encoder.g711mlaw" },
1528c5255ac5b1bd313dcb50159566163b24dce7483fAndreas Huber        { MEDIA_MIMETYPE_AUDIO_G711_ALAW,
1529c5255ac5b1bd313dcb50159566163b24dce7483fAndreas Huber            "audio_decoder.g711alaw", "audio_encoder.g711alaw" },
1530f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_AVC,
1531f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.avc", "video_encoder.avc" },
15322472b1c0d63454e5d90a982bd6c555de6c3127bdRachad        { MEDIA_MIMETYPE_VIDEO_HEVC,
15332472b1c0d63454e5d90a982bd6c555de6c3127bdRachad            "video_decoder.hevc", "video_encoder.hevc" },
1534f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_MPEG4,
1535f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.mpeg4", "video_encoder.mpeg4" },
1536f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_H263,
1537f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.h263", "video_encoder.h263" },
153894705aff3c9eef58cbb72ec6fe5d2dcfd9481646hkuang        { MEDIA_MIMETYPE_VIDEO_VP8,
153994705aff3c9eef58cbb72ec6fe5d2dcfd9481646hkuang            "video_decoder.vp8", "video_encoder.vp8" },
154094705aff3c9eef58cbb72ec6fe5d2dcfd9481646hkuang        { MEDIA_MIMETYPE_VIDEO_VP9,
154194705aff3c9eef58cbb72ec6fe5d2dcfd9481646hkuang            "video_decoder.vp9", "video_encoder.vp9" },
1542ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber        { MEDIA_MIMETYPE_AUDIO_RAW,
1543ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber            "audio_decoder.raw", "audio_encoder.raw" },
15442f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        { MEDIA_MIMETYPE_AUDIO_FLAC,
15452f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            "audio_decoder.flac", "audio_encoder.flac" },
1546ba933df89521d63f75ca66af12ce9d7ae9496b9eMarco Nelissen        { MEDIA_MIMETYPE_AUDIO_MSGSM,
1547ba933df89521d63f75ca66af12ce9d7ae9496b9eMarco Nelissen            "audio_decoder.gsm", "audio_encoder.gsm" },
1548774eb18c40c3a7da0bc1636a9779f02315ddbad8Changwan Ryu        { MEDIA_MIMETYPE_VIDEO_MPEG2,
1549774eb18c40c3a7da0bc1636a9779f02315ddbad8Changwan Ryu            "video_decoder.mpeg2", "video_encoder.mpeg2" },
155097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        { MEDIA_MIMETYPE_AUDIO_AC3,
155197358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            "audio_decoder.ac3", "audio_encoder.ac3" },
15528a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        { MEDIA_MIMETYPE_AUDIO_EAC3,
15538a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            "audio_decoder.eac3", "audio_encoder.eac3" },
1554f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
1555f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1556f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    static const size_t kNumMimeToRole =
1557f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sizeof(kMimeToRole) / sizeof(kMimeToRole[0]);
1558f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1559f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    size_t i;
1560f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (i = 0; i < kNumMimeToRole; ++i) {
1561f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (!strcasecmp(mime, kMimeToRole[i].mime)) {
1562f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1563f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1564f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1565f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1566f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (i == kNumMimeToRole) {
15675778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return ERROR_UNSUPPORTED;
1568f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1569f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1570f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    const char *role =
1571f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        isEncoder ? kMimeToRole[i].encoderRole
1572f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                  : kMimeToRole[i].decoderRole;
1573f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1574f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (role != NULL) {
1575f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_PARAM_COMPONENTROLETYPE roleParams;
1576f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        InitOMXParams(&roleParams);
1577f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1578f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        strncpy((char *)roleParams.cRole,
1579f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                role, OMX_MAX_STRINGNAME_SIZE - 1);
1580f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1581f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        roleParams.cRole[OMX_MAX_STRINGNAME_SIZE - 1] = '\0';
1582f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1583f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        status_t err = mOMX->setParameter(
1584f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, OMX_IndexParamStandardComponentRole,
1585f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &roleParams, sizeof(roleParams));
1586f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1587f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
15885ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("[%s] Failed to set standard component role '%s'.",
1589f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 mComponentName.c_str(), role);
15905778822d86b0337407514b9372562b86edfa91cdAndreas Huber
15915778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
1592f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1593f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
15945778822d86b0337407514b9372562b86edfa91cdAndreas Huber
15955778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
1596f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1597f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
15985778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::configureCodec(
1599f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime, const sp<AMessage> &msg) {
16005778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t encoder;
16015778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("encoder", &encoder)) {
16025778822d86b0337407514b9372562b86edfa91cdAndreas Huber        encoder = false;
16035778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
1604f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1605e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    sp<AMessage> inputFormat = new AMessage();
16064e865a3cfe4c955e0890321a6b488cf661808b63Lajos Molnar    sp<AMessage> outputFormat = mNotify->dup(); // will use this for kWhatOutputFormatChanged
1607e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
16085778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mIsEncoder = encoder;
1609f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1610054219874873b41f1c815552987c10465c34ba2bLajos Molnar    mInputMetadataType = kMetadataBufferTypeInvalid;
1611054219874873b41f1c815552987c10465c34ba2bLajos Molnar    mOutputMetadataType = kMetadataBufferTypeInvalid;
16128db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu
16135778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = setComponentRole(encoder /* isEncoder */, mime);
16145778822d86b0337407514b9372562b86edfa91cdAndreas Huber
16155778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
16165778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
16175778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
16185778822d86b0337407514b9372562b86edfa91cdAndreas Huber
16195778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t bitRate = 0;
16202f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi    // FLAC encoder doesn't need a bitrate, other encoders do
16212f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi    if (encoder && strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_FLAC)
16222f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            && !msg->findInt32("bitrate", &bitRate)) {
16235778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return INVALID_OPERATION;
16245778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
16255778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1626d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t storeMeta;
1627d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (encoder
1628d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            && msg->findInt32("store-metadata-in-buffers", &storeMeta)
1629d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            && storeMeta != 0) {
1630054219874873b41f1c815552987c10465c34ba2bLajos Molnar        err = mOMX->storeMetaDataInBuffers(mNode, kPortIndexInput, OMX_TRUE, &mInputMetadataType);
1631d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (err != OK) {
1632054219874873b41f1c815552987c10465c34ba2bLajos Molnar            ALOGE("[%s] storeMetaDataInBuffers (input) failed w/ err %d",
1633308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                    mComponentName.c_str(), err);
1634d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1635054219874873b41f1c815552987c10465c34ba2bLajos Molnar            return err;
1636054219874873b41f1c815552987c10465c34ba2bLajos Molnar        }
1637054219874873b41f1c815552987c10465c34ba2bLajos Molnar        // For this specific case we could be using camera source even if storeMetaDataInBuffers
1638054219874873b41f1c815552987c10465c34ba2bLajos Molnar        // returns Gralloc source. Pretend that we are; this will force us to use nBufferSize.
1639054219874873b41f1c815552987c10465c34ba2bLajos Molnar        if (mInputMetadataType == kMetadataBufferTypeGrallocSource) {
1640054219874873b41f1c815552987c10465c34ba2bLajos Molnar            mInputMetadataType = kMetadataBufferTypeCameraSource;
1641054219874873b41f1c815552987c10465c34ba2bLajos Molnar        }
1642054219874873b41f1c815552987c10465c34ba2bLajos Molnar    }
1643d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1644308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    int32_t prependSPSPPS = 0;
16453a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber    if (encoder
16463a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            && msg->findInt32("prepend-sps-pps-to-idr-frames", &prependSPSPPS)
16473a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            && prependSPSPPS != 0) {
16483a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber        OMX_INDEXTYPE index;
16493a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber        err = mOMX->getExtensionIndex(
16503a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber                mNode,
16513a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber                "OMX.google.android.index.prependSPSPPSToIDRFrames",
16523a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber                &index);
16533a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber
16543a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber        if (err == OK) {
16553a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            PrependSPSPPSToIDRFramesParams params;
16563a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            InitOMXParams(&params);
16573a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            params.bEnable = OMX_TRUE;
16583a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber
16593a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            err = mOMX->setParameter(
16603a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber                    mNode, index, &params, sizeof(params));
16613a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber        }
16623a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber
16633a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber        if (err != OK) {
16643a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            ALOGE("Encoder could not be configured to emit SPS/PPS before "
16653a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber                  "IDR frames. (err %d)", err);
16663a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber
16673a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber            return err;
16683a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber        }
16693a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber    }
16703a0ef0dc71f3776d115f30d87f1d8867daefea88Andreas Huber
1671308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    // Only enable metadata mode on encoder output if encoder can prepend
1672308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    // sps/pps to idr frames, since in metadata mode the bitstream is in an
1673308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    // opaque handle, to which we don't have access.
1674308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    int32_t video = !strncasecmp(mime, "video/", 6);
16758db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu    mIsVideo = video;
1676308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    if (encoder && video) {
1677308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang        OMX_BOOL enable = (OMX_BOOL) (prependSPSPPS
1678308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang            && msg->findInt32("store-metadata-in-buffers-output", &storeMeta)
1679308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang            && storeMeta != 0);
1680308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang
1681054219874873b41f1c815552987c10465c34ba2bLajos Molnar        err = mOMX->storeMetaDataInBuffers(mNode, kPortIndexOutput, enable, &mOutputMetadataType);
1682308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang        if (err != OK) {
1683308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang            ALOGE("[%s] storeMetaDataInBuffers (output) failed w/ err %d",
1684308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                mComponentName.c_str(), err);
1685308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang        }
1686a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber
1687a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber        if (!msg->findInt64(
1688a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                    "repeat-previous-frame-after",
1689a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                    &mRepeatFrameDelayUs)) {
1690a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber            mRepeatFrameDelayUs = -1ll;
1691a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber        }
169294ee4b708acfa941581160b267afb79192b1d816Chong Zhang
169394ee4b708acfa941581160b267afb79192b1d816Chong Zhang        if (!msg->findInt64("max-pts-gap-to-encoder", &mMaxPtsGapUs)) {
16942c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang            mMaxPtsGapUs = -1ll;
16952c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        }
16962c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang
169737b2b389139ed638831e49708c947863eef631efRonghua Wu        if (!msg->findFloat("max-fps-to-encoder", &mMaxFps)) {
169837b2b389139ed638831e49708c947863eef631efRonghua Wu            mMaxFps = -1;
169937b2b389139ed638831e49708c947863eef631efRonghua Wu        }
170037b2b389139ed638831e49708c947863eef631efRonghua Wu
17012c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        if (!msg->findInt64("time-lapse", &mTimePerCaptureUs)) {
17022c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang            mTimePerCaptureUs = -1ll;
170394ee4b708acfa941581160b267afb79192b1d816Chong Zhang        }
170472cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang
170572cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        if (!msg->findInt32(
170672cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                    "create-input-buffers-suspended",
170772cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                    (int32_t*)&mCreateInputBuffersSuspended)) {
170872cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang            mCreateInputBuffersSuspended = false;
170972cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        }
1710308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    }
1711308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang
17123a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar    // NOTE: we only use native window for video decoders
1713054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    sp<RefBase> obj;
17140d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    bool haveNativeWindow = msg->findObject("native-window", &obj)
17153a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            && obj != NULL && video && !encoder;
1716011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar    mLegacyAdaptiveExperiment = false;
1717e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    if (video && !encoder) {
1718e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        inputFormat->setInt32("adaptive-playback", false);
17191713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang
17201713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang        int32_t usageProtected;
17211713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang        if (msg->findInt32("protected", &usageProtected) && usageProtected) {
17221713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang            if (!haveNativeWindow) {
17231713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang                ALOGE("protected output buffers must be sent to an ANativeWindow");
17241713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang                return PERMISSION_DENIED;
17251713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang            }
17261713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang            mFlags |= kFlagIsGrallocUsageProtected;
17271713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang            mFlags |= kFlagPushBlankBuffersToNativeWindowOnShutdown;
17281713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang        }
1729e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    }
17303a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar    if (haveNativeWindow) {
17311de1e25cba872bd4c077c2e394f8ca9c70b65856Lajos Molnar        sp<ANativeWindow> nativeWindow =
17321de1e25cba872bd4c077c2e394f8ca9c70b65856Lajos Molnar            static_cast<ANativeWindow *>(static_cast<Surface *>(obj.get()));
17335a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
17346597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar        // START of temporary support for automatic FRC - THIS WILL BE REMOVED
17356597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar        int32_t autoFrc;
17366597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar        if (msg->findInt32("auto-frc", &autoFrc)) {
17376597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            bool enabled = autoFrc;
17386597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            OMX_CONFIG_BOOLEANTYPE config;
17396597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            InitOMXParams(&config);
17406597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            config.bEnabled = (OMX_BOOL)enabled;
17416597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            status_t temp = mOMX->setConfig(
17426597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar                    mNode, (OMX_INDEXTYPE)OMX_IndexConfigAutoFramerateConversion,
17436597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar                    &config, sizeof(config));
17446597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            if (temp == OK) {
17456597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar                outputFormat->setInt32("auto-frc", enabled);
17466597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            } else if (enabled) {
17476597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar                ALOGI("codec does not support requested auto-frc (err %d)", temp);
17486597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar            }
17496597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar        }
17506597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar        // END of temporary support for automatic FRC
17516597c00f7015e5ec9a07601d7a760169ca7266d3Lajos Molnar
17525a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad        int32_t tunneled;
17535a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad        if (msg->findInt32("feature-tunneled-playback", &tunneled) &&
17545a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            tunneled != 0) {
17555a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            ALOGI("Configuring TUNNELED video playback.");
1756ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad            mTunneled = true;
17575a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
175897827bd7c7e64dec22c8fe0f9e734a3c432ad7eeRachad            int32_t audioHwSync = 0;
175997827bd7c7e64dec22c8fe0f9e734a3c432ad7eeRachad            if (!msg->findInt32("audio-hw-sync", &audioHwSync)) {
17605a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                ALOGW("No Audio HW Sync provided for video tunnel");
17615a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            }
17625a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            err = configureTunneledVideoPlayback(audioHwSync, nativeWindow);
17635a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            if (err != OK) {
176497827bd7c7e64dec22c8fe0f9e734a3c432ad7eeRachad                ALOGE("configureTunneledVideoPlayback(%d,%p) failed!",
17655a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        audioHwSync, nativeWindow.get());
17665a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                return err;
1767fce0d1883cdbcb7d501625fb43844043cd28a267Lajos Molnar            }
1768fce0d1883cdbcb7d501625fb43844043cd28a267Lajos Molnar
1769d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad            int32_t maxWidth = 0, maxHeight = 0;
1770d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad            if (msg->findInt32("max-width", &maxWidth) &&
1771d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                    msg->findInt32("max-height", &maxHeight)) {
1772d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad
1773d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                err = mOMX->prepareForAdaptivePlayback(
1774d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                        mNode, kPortIndexOutput, OMX_TRUE, maxWidth, maxHeight);
1775d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                if (err != OK) {
1776d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                    ALOGW("[%s] prepareForAdaptivePlayback failed w/ err %d",
1777d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                            mComponentName.c_str(), err);
17783a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    // allow failure
17793a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    err = OK;
1780d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                } else {
1781d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                    inputFormat->setInt32("max-width", maxWidth);
1782d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                    inputFormat->setInt32("max-height", maxHeight);
1783d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                    inputFormat->setInt32("adaptive-playback", true);
1784d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad                }
1785d0b9a2b8538c9e4538d92fd675cf6786644ccb85Rachad            }
17865a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad        } else {
1787ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad            ALOGV("Configuring CPU controlled video playback.");
1788ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad            mTunneled = false;
1789ab76066c11e988ca3d3a5d6d74dd510ae080322eRachad
17903fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad            // Explicity reset the sideband handle of the window for
17913fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad            // non-tunneled video in case the window was previously used
17923fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad            // for a tunneled video playback.
17933fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad            err = native_window_set_sideband_stream(nativeWindow.get(), NULL);
17943fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad            if (err != OK) {
17953fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad                ALOGE("set_sideband_stream(NULL) failed! (err %d).", err);
17963fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad                return err;
17973fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad            }
17983fb3917ae19f07ddfb2176a9da3c7cfa514522a5Rachad
17995a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            // Always try to enable dynamic output buffers on native surface
18005a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            err = mOMX->storeMetaDataInBuffers(
1801054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    mNode, kPortIndexOutput, OMX_TRUE, &mOutputMetadataType);
18025a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            if (err != OK) {
18035a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                ALOGE("[%s] storeMetaDataInBuffers failed w/ err %d",
1804fce0d1883cdbcb7d501625fb43844043cd28a267Lajos Molnar                        mComponentName.c_str(), err);
1805e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
18065a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                // if adaptive playback has been requested, try JB fallback
18075a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                // NOTE: THIS FALLBACK MECHANISM WILL BE REMOVED DUE TO ITS
18085a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                // LARGE MEMORY REQUIREMENT
18095a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
18105a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                // we will not do adaptive playback on software accessed
18115a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                // surfaces as they never had to respond to changes in the
18125a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                // crop window, and we don't trust that they will be able to.
18135a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                int usageBits = 0;
18145a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                bool canDoAdaptivePlayback;
18155a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
18165a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                if (nativeWindow->query(
18175a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        nativeWindow.get(),
18185a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        NATIVE_WINDOW_CONSUMER_USAGE_BITS,
18195a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        &usageBits) != OK) {
18205a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    canDoAdaptivePlayback = false;
18215a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                } else {
18225a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    canDoAdaptivePlayback =
18235a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        (usageBits &
18245a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                                (GRALLOC_USAGE_SW_READ_MASK |
18255a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                                 GRALLOC_USAGE_SW_WRITE_MASK)) == 0;
18265a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                }
18275a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
18285a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                int32_t maxWidth = 0, maxHeight = 0;
18295a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                if (canDoAdaptivePlayback &&
18305a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        msg->findInt32("max-width", &maxWidth) &&
18315a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        msg->findInt32("max-height", &maxHeight)) {
18325a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    ALOGV("[%s] prepareForAdaptivePlayback(%dx%d)",
18335a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                            mComponentName.c_str(), maxWidth, maxHeight);
18345a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
18355a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    err = mOMX->prepareForAdaptivePlayback(
18365a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                            mNode, kPortIndexOutput, OMX_TRUE, maxWidth,
18375a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                            maxHeight);
18385a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    ALOGW_IF(err != OK,
18395a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                            "[%s] prepareForAdaptivePlayback failed w/ err %d",
18405a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                            mComponentName.c_str(), err);
18415a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
18425a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    if (err == OK) {
18435a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        inputFormat->setInt32("max-width", maxWidth);
18445a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        inputFormat->setInt32("max-height", maxHeight);
18455a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        inputFormat->setInt32("adaptive-playback", true);
18465a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    }
1847e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                }
18485a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                // allow failure
18495a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                err = OK;
18505a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            } else {
18515a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                ALOGV("[%s] storeMetaDataInBuffers succeeded",
18525a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                        mComponentName.c_str());
1853054219874873b41f1c815552987c10465c34ba2bLajos Molnar                CHECK(storingMetadataInDecodedBuffers());
1854011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                mLegacyAdaptiveExperiment = ADebug::isExperimentEnabled(
1855011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                        "legacy-adaptive", !msg->contains("no-experiments"));
1856011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar
18575a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                inputFormat->setInt32("adaptive-playback", true);
1858fce0d1883cdbcb7d501625fb43844043cd28a267Lajos Molnar            }
18590167414e261f88a96b5e4bf6cb592e6ca11e5a95Andreas Huber
18605a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            int32_t push;
18615a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            if (msg->findInt32("push-blank-buffers-on-shutdown", &push)
18625a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                    && push != 0) {
18635a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                mFlags |= kFlagPushBlankBuffersToNativeWindowOnShutdown;
18645a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            }
18650167414e261f88a96b5e4bf6cb592e6ca11e5a95Andreas Huber        }
1866e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang
1867e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang        int32_t rotationDegrees;
1868e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang        if (msg->findInt32("rotation-degrees", &rotationDegrees)) {
1869e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang            mRotationDegrees = rotationDegrees;
1870e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang        } else {
1871e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang            mRotationDegrees = 0;
1872e9e63bcf6c36351f1129b0bdc5e93f17f0f9f0b4Chong Zhang        }
1873054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    }
1874054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
1875308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang    if (video) {
18763a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        // determine need for software renderer
18773a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        bool usingSwRenderer = false;
18783a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        if (haveNativeWindow && mComponentName.startsWith("OMX.google.")) {
18793a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            usingSwRenderer = true;
18803a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            haveNativeWindow = false;
18813a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        }
18823a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar
18835778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (encoder) {
18845778822d86b0337407514b9372562b86edfa91cdAndreas Huber            err = setupVideoEncoder(mime, msg);
18855778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else {
18860d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            err = setupVideoDecoder(mime, msg, haveNativeWindow);
18875778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
18883a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar
18893a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        if (err != OK) {
18903a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            return err;
18913a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        }
18923a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar
18933a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        if (haveNativeWindow) {
18941de1e25cba872bd4c077c2e394f8ca9c70b65856Lajos Molnar            mNativeWindow = static_cast<Surface *>(obj.get());
18953a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        }
18963a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar
18973a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        // initialize native window now to get actual output format
18983a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        // TODO: this is needed for some encoders even though they don't use native window
1899777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        err = initNativeWindow();
1900777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        if (err != OK) {
1901777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            return err;
1902777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        }
19033a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar
19043a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        // fallback for devices that do not handle flex-YUV for native buffers
19053a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        if (haveNativeWindow) {
19063a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            int32_t requestedColorFormat = OMX_COLOR_FormatUnused;
19073a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            if (msg->findInt32("color-format", &requestedColorFormat) &&
19083a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    requestedColorFormat == OMX_COLOR_FormatYUV420Flexible) {
1909777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                status_t err = getPortFormat(kPortIndexOutput, outputFormat);
1910777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                if (err != OK) {
1911777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    return err;
1912777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                }
19133a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                int32_t colorFormat = OMX_COLOR_FormatUnused;
19143a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                OMX_U32 flexibleEquivalent = OMX_COLOR_FormatUnused;
1915777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                if (!outputFormat->findInt32("color-format", &colorFormat)) {
1916777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    ALOGE("ouptut port did not have a color format (wrong domain?)");
1917777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    return BAD_VALUE;
1918777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                }
19193a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                ALOGD("[%s] Requested output format %#x and got %#x.",
19203a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        mComponentName.c_str(), requestedColorFormat, colorFormat);
19213a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                if (!isFlexibleColorFormat(
19223a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                                mOMX, mNode, colorFormat, haveNativeWindow, &flexibleEquivalent)
19233a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        || flexibleEquivalent != (OMX_U32)requestedColorFormat) {
19243a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    // device did not handle flex-YUV request for native window, fall back
19253a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    // to SW renderer
19263a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    ALOGI("[%s] Falling back to software renderer", mComponentName.c_str());
19273a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    mNativeWindow.clear();
19283a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    haveNativeWindow = false;
19293a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    usingSwRenderer = true;
1930054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    if (storingMetadataInDecodedBuffers()) {
1931054219874873b41f1c815552987c10465c34ba2bLajos Molnar                        err = mOMX->storeMetaDataInBuffers(
1932054219874873b41f1c815552987c10465c34ba2bLajos Molnar                                mNode, kPortIndexOutput, OMX_FALSE, &mOutputMetadataType);
1933054219874873b41f1c815552987c10465c34ba2bLajos Molnar                        mOutputMetadataType = kMetadataBufferTypeInvalid; // just in case
19343a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        // TODO: implement adaptive-playback support for bytebuffer mode.
19353a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        // This is done by SW codecs, but most HW codecs don't support it.
19363a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        inputFormat->setInt32("adaptive-playback", false);
19373a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    }
19383a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    if (err == OK) {
19393a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        err = mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_FALSE);
19403a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    }
19413a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    if (mFlags & kFlagIsGrallocUsageProtected) {
19423a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        // fallback is not supported for protected playback
19433a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        err = PERMISSION_DENIED;
19443a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    } else if (err == OK) {
19453a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                        err = setupVideoDecoder(mime, msg, false);
19463a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                    }
19473a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar                }
19483a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            }
19493a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        }
19503a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar
19513a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        if (usingSwRenderer) {
19523a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar            outputFormat->setInt32("using-sw-renderer", 1);
19533a01a71dcbb467d06cc5da4a72a82bb588648cfcLajos Molnar        }
195442392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_MPEG)) {
195542392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber        int32_t numChannels, sampleRate;
195642392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber        if (!msg->findInt32("channel-count", &numChannels)
195742392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber                || !msg->findInt32("sample-rate", &sampleRate)) {
195842392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber            // Since we did not always check for these, leave them optional
195942392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber            // and have the decoder figure it all out.
196042392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber            err = OK;
196142392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber        } else {
196242392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber            err = setupRawAudioFormat(
196342392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber                    encoder ? kPortIndexInput : kPortIndexOutput,
196442392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber                    sampleRate,
196542392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber                    numChannels);
196642392e49e167c6a0c573e55e1c1b4c7fa0ceb213Andreas Huber        }
1967f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_AAC)) {
1968f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int32_t numChannels, sampleRate;
19695778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("channel-count", &numChannels)
19705778822d86b0337407514b9372562b86edfa91cdAndreas Huber                || !msg->findInt32("sample-rate", &sampleRate)) {
19715778822d86b0337407514b9372562b86edfa91cdAndreas Huber            err = INVALID_OPERATION;
19725778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else {
1973aeb8fd460ed87d032b3fb8bb61e21eb542ce0f5bDave Burke            int32_t isADTS, aacProfile;
1974b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            int32_t sbrMode;
19758045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            int32_t maxOutputChannelCount;
19762965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang            int32_t pcmLimiterEnable;
19778045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            drcParams_t drc;
1978ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber            if (!msg->findInt32("is-adts", &isADTS)) {
1979ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                isADTS = 0;
1980ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber            }
1981aeb8fd460ed87d032b3fb8bb61e21eb542ce0f5bDave Burke            if (!msg->findInt32("aac-profile", &aacProfile)) {
1982aeb8fd460ed87d032b3fb8bb61e21eb542ce0f5bDave Burke                aacProfile = OMX_AUDIO_AACObjectNull;
1983aeb8fd460ed87d032b3fb8bb61e21eb542ce0f5bDave Burke            }
1984b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            if (!msg->findInt32("aac-sbr-mode", &sbrMode)) {
1985b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi                sbrMode = -1;
1986b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            }
1987ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
19888045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            if (!msg->findInt32("aac-max-output-channel_count", &maxOutputChannelCount)) {
19898045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                maxOutputChannelCount = -1;
19908045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            }
19912965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang            if (!msg->findInt32("aac-pcm-limiter-enable", &pcmLimiterEnable)) {
19922965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang                // value is unknown
19932965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang                pcmLimiterEnable = -1;
19942965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang            }
19958045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            if (!msg->findInt32("aac-encoded-target-level", &drc.encodedTargetLevel)) {
19968045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                // value is unknown
19978045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                drc.encodedTargetLevel = -1;
19988045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            }
19998045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            if (!msg->findInt32("aac-drc-cut-level", &drc.drcCut)) {
20008045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                // value is unknown
20018045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                drc.drcCut = -1;
20028045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            }
20038045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            if (!msg->findInt32("aac-drc-boost-level", &drc.drcBoost)) {
20048045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                // value is unknown
20058045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                drc.drcBoost = -1;
20068045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            }
20078045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            if (!msg->findInt32("aac-drc-heavy-compression", &drc.heavyCompression)) {
20088045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                // value is unknown
20098045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                drc.heavyCompression = -1;
20108045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            }
20118045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            if (!msg->findInt32("aac-target-ref-level", &drc.targetRefLevel)) {
20128045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                // value is unknown
20138045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                drc.targetRefLevel = -1;
20148045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi            }
20158045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi
2016ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber            err = setupAACCodec(
20174471e47ece77be5b4159cc9cbbcef4d43900d36fAndreas Huber                    encoder, numChannels, sampleRate, bitRate, aacProfile,
20182965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang                    isADTS != 0, sbrMode, maxOutputChannelCount, drc,
20192965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang                    pcmLimiterEnable);
20205778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
2021729de186450f78c099637e1fce743fe531862c52Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_AMR_NB)) {
20225778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = setupAMRCodec(encoder, false /* isWAMR */, bitRate);
2023729de186450f78c099637e1fce743fe531862c52Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_AMR_WB)) {
20245778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = setupAMRCodec(encoder, true /* isWAMR */, bitRate);
2025729de186450f78c099637e1fce743fe531862c52Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_G711_ALAW)
2026729de186450f78c099637e1fce743fe531862c52Andreas Huber            || !strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_G711_MLAW)) {
2027729de186450f78c099637e1fce743fe531862c52Andreas Huber        // These are PCM-like formats with a fixed sample rate but
2028729de186450f78c099637e1fce743fe531862c52Andreas Huber        // a variable number of channels.
2029729de186450f78c099637e1fce743fe531862c52Andreas Huber
2030729de186450f78c099637e1fce743fe531862c52Andreas Huber        int32_t numChannels;
20315778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("channel-count", &numChannels)) {
20325778822d86b0337407514b9372562b86edfa91cdAndreas Huber            err = INVALID_OPERATION;
20335778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else {
203417c39e708ed657b8fa66f8acce5128e51696915cMarco Nelissen            int32_t sampleRate;
203517c39e708ed657b8fa66f8acce5128e51696915cMarco Nelissen            if (!msg->findInt32("sample-rate", &sampleRate)) {
203617c39e708ed657b8fa66f8acce5128e51696915cMarco Nelissen                sampleRate = 8000;
203717c39e708ed657b8fa66f8acce5128e51696915cMarco Nelissen            }
203817c39e708ed657b8fa66f8acce5128e51696915cMarco Nelissen            err = setupG711Codec(encoder, sampleRate, numChannels);
20395778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
20402f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_FLAC)) {
2041ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar        int32_t numChannels = 0, sampleRate = 0, compressionLevel = -1;
20422f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        if (encoder &&
20432f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi                (!msg->findInt32("channel-count", &numChannels)
20442f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi                        || !msg->findInt32("sample-rate", &sampleRate))) {
20452f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            ALOGE("missing channel count or sample rate for FLAC encoder");
20462f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            err = INVALID_OPERATION;
20472f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        } else {
20482f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            if (encoder) {
2049516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                if (!msg->findInt32(
2050cd39746f8d83bb3f12e8f613e77c3c3b5f77c077Lajos Molnar                            "complexity", &compressionLevel) &&
2051cd39746f8d83bb3f12e8f613e77c3c3b5f77c077Lajos Molnar                    !msg->findInt32(
2052516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                            "flac-compression-level", &compressionLevel)) {
2053cd39746f8d83bb3f12e8f613e77c3c3b5f77c077Lajos Molnar                    compressionLevel = 5; // default FLAC compression level
20542f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi                } else if (compressionLevel < 0) {
2055516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                    ALOGW("compression level %d outside [0..8] range, "
2056516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                          "using 0",
2057516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                          compressionLevel);
20582f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi                    compressionLevel = 0;
20592f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi                } else if (compressionLevel > 8) {
2060516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                    ALOGW("compression level %d outside [0..8] range, "
2061516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                          "using 8",
2062516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                          compressionLevel);
20632f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi                    compressionLevel = 8;
20642f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi                }
20652f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            }
2066516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber            err = setupFlacCodec(
2067516dacfb02d0b0eafe21114330c98ce0e7d90da9Andreas Huber                    encoder, numChannels, sampleRate, compressionLevel);
20682f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        }
2069ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_RAW)) {
2070ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber        int32_t numChannels, sampleRate;
2071ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber        if (encoder
2072ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber                || !msg->findInt32("channel-count", &numChannels)
2073ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber                || !msg->findInt32("sample-rate", &sampleRate)) {
2074ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber            err = INVALID_OPERATION;
2075ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber        } else {
2076ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber            err = setupRawAudioFormat(kPortIndexInput, sampleRate, numChannels);
2077ecdd39c5af016e2fa57cbfd837aa670b706dabd3Andreas Huber        }
207897358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_AC3)) {
207997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        int32_t numChannels;
208097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        int32_t sampleRate;
208197358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        if (!msg->findInt32("channel-count", &numChannels)
208297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                || !msg->findInt32("sample-rate", &sampleRate)) {
208397358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            err = INVALID_OPERATION;
208497358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        } else {
208597358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            err = setupAC3Codec(encoder, numChannels, sampleRate);
208697358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        }
20878a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_EAC3)) {
20888a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        int32_t numChannels;
20898a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        int32_t sampleRate;
20908a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        if (!msg->findInt32("channel-count", &numChannels)
20918a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                || !msg->findInt32("sample-rate", &sampleRate)) {
20928a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            err = INVALID_OPERATION;
20938a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        } else {
20948a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            err = setupEAC3Codec(encoder, numChannels, sampleRate);
20958a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        }
20965778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
2097729de186450f78c099637e1fce743fe531862c52Andreas Huber
20984471e47ece77be5b4159cc9cbbcef4d43900d36fAndreas Huber    if (err != OK) {
20994471e47ece77be5b4159cc9cbbcef4d43900d36fAndreas Huber        return err;
21004471e47ece77be5b4159cc9cbbcef4d43900d36fAndreas Huber    }
21014471e47ece77be5b4159cc9cbbcef4d43900d36fAndreas Huber
21028b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen    if (!msg->findInt32("encoder-delay", &mEncoderDelay)) {
21038b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen        mEncoderDelay = 0;
21048b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen    }
21059806555d3930be43e11106281dee354820ac1c88Andreas Huber
21068b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen    if (!msg->findInt32("encoder-padding", &mEncoderPadding)) {
21078b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen        mEncoderPadding = 0;
21088b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen    }
21098b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen
21109806555d3930be43e11106281dee354820ac1c88Andreas Huber    if (msg->findInt32("channel-mask", &mChannelMask)) {
21119806555d3930be43e11106281dee354820ac1c88Andreas Huber        mChannelMaskPresent = true;
21129806555d3930be43e11106281dee354820ac1c88Andreas Huber    } else {
21139806555d3930be43e11106281dee354820ac1c88Andreas Huber        mChannelMaskPresent = false;
21149806555d3930be43e11106281dee354820ac1c88Andreas Huber    }
21159806555d3930be43e11106281dee354820ac1c88Andreas Huber
2116f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t maxInputSize;
2117f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (msg->findInt32("max-input-size", &maxInputSize)) {
21185778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = setMinBufferSize(kPortIndexInput, (size_t)maxInputSize);
2119f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcmp("OMX.Nvidia.aac.decoder", mComponentName.c_str())) {
21205778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = setMinBufferSize(kPortIndexInput, 8192);  // XXX
2121f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
21225778822d86b0337407514b9372562b86edfa91cdAndreas Huber
21238b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    int32_t priority;
21248b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    if (msg->findInt32("priority", &priority)) {
21258b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu        err = setPriority(priority);
21268b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    }
21278b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu
2128ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    int32_t rateInt = -1;
2129ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    float rateFloat = -1;
2130ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    if (!msg->findFloat("operating-rate", &rateFloat)) {
2131ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        msg->findInt32("operating-rate", &rateInt);
2132ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        rateFloat = (float)rateInt;  // 16MHz (FLINTMAX) is OK for upper bound.
2133ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    }
2134ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    if (rateFloat > 0) {
2135ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        err = setOperatingRate(rateFloat, video);
2136ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    }
2137ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu
21384e865a3cfe4c955e0890321a6b488cf661808b63Lajos Molnar    mBaseOutputFormat = outputFormat;
21394e865a3cfe4c955e0890321a6b488cf661808b63Lajos Molnar
2140777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    err = getPortFormat(kPortIndexInput, inputFormat);
2141777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (err == OK) {
2142777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        err = getPortFormat(kPortIndexOutput, outputFormat);
2143777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        if (err == OK) {
2144777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            mInputFormat = inputFormat;
2145777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            mOutputFormat = outputFormat;
2146777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        }
2147777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
21485778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return err;
2149f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
21518b806ea894ca098366629458bfdd1df4866afcdfRonghua Wustatus_t ACodec::setPriority(int32_t priority) {
21528b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    if (priority < 0) {
21538b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu        return BAD_VALUE;
21548b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    }
21558b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    OMX_PARAM_U32TYPE config;
21568b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    InitOMXParams(&config);
21578b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    config.nU32 = (OMX_U32)priority;
21588b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    status_t temp = mOMX->setConfig(
21598b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu            mNode, (OMX_INDEXTYPE)OMX_IndexConfigPriority,
21608b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu            &config, sizeof(config));
21618b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    if (temp != OK) {
21628b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu        ALOGI("codec does not support config priority (err %d)", temp);
21638b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    }
21648b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu    return OK;
21658b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu}
21668b806ea894ca098366629458bfdd1df4866afcdfRonghua Wu
2167ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wustatus_t ACodec::setOperatingRate(float rateFloat, bool isVideo) {
2168ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    if (rateFloat < 0) {
2169ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        return BAD_VALUE;
2170ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    }
2171ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    OMX_U32 rate;
2172ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    if (isVideo) {
2173ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        if (rateFloat > 65535) {
2174ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu            return BAD_VALUE;
2175ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        }
2176ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        rate = (OMX_U32)(rateFloat * 65536.0f + 0.5f);
2177ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    } else {
2178ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        if (rateFloat > UINT_MAX) {
2179ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu            return BAD_VALUE;
2180ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        }
2181ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        rate = (OMX_U32)(rateFloat);
2182ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    }
2183ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    OMX_PARAM_U32TYPE config;
2184ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    InitOMXParams(&config);
2185ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    config.nU32 = rate;
2186ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    status_t err = mOMX->setConfig(
2187ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu            mNode, (OMX_INDEXTYPE)OMX_IndexConfigOperatingRate,
2188ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu            &config, sizeof(config));
2189ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    if (err != OK) {
2190ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu        ALOGI("codec does not support config operating rate (err %d)", err);
2191ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    }
2192ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu    return OK;
2193ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu}
2194ed5554fb1c837f6b1008d3910018c979c57da502Ronghua Wu
2195f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setMinBufferSize(OMX_U32 portIndex, size_t size) {
2196f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
2197f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
2198f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
2199f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2200f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
2201f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
2202f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2203f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2204f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2205f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2207f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (def.nBufferSize >= size) {
2208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return OK;
2209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2210f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2211f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nBufferSize = size;
2212f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2213f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
2214f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
2215f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2216f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2217f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2218f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2219f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2220f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->getParameter(
2221f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
2222f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2223f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2224f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2225f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2226f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2227777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (def.nBufferSize < size) {
2228777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        ALOGE("failed to set min buffer size to %zu (is still %u)", size, def.nBufferSize);
2229777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return FAILED_TRANSACTION;
2230777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
2231f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2232f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
2233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
22355778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::selectAudioPortFormat(
22365778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMX_U32 portIndex, OMX_AUDIO_CODINGTYPE desiredFormat) {
22375778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_AUDIO_PARAM_PORTFORMATTYPE format;
22385778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&format);
22395778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22405778822d86b0337407514b9372562b86edfa91cdAndreas Huber    format.nPortIndex = portIndex;
22415778822d86b0337407514b9372562b86edfa91cdAndreas Huber    for (OMX_U32 index = 0;; ++index) {
22425778822d86b0337407514b9372562b86edfa91cdAndreas Huber        format.nIndex = index;
22435778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22445778822d86b0337407514b9372562b86edfa91cdAndreas Huber        status_t err = mOMX->getParameter(
22455778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mNode, OMX_IndexParamAudioPortFormat,
22465778822d86b0337407514b9372562b86edfa91cdAndreas Huber                &format, sizeof(format));
22475778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22485778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
22495778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
22505778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
22515778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22525778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (format.eEncoding == desiredFormat) {
22535778822d86b0337407514b9372562b86edfa91cdAndreas Huber            break;
22545778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
22555778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
22565778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22575778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return mOMX->setParameter(
22585778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamAudioPortFormat, &format, sizeof(format));
22595778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
22605778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22615778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupAACCodec(
2262aeb8fd460ed87d032b3fb8bb61e21eb542ce0f5bDave Burke        bool encoder, int32_t numChannels, int32_t sampleRate,
22638045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi        int32_t bitRate, int32_t aacProfile, bool isADTS, int32_t sbrMode,
22642965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang        int32_t maxOutputChannelCount, const drcParams_t& drc,
22652965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang        int32_t pcmLimiterEnable) {
2266ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    if (encoder && isADTS) {
2267ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        return -EINVAL;
2268ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    }
2269ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
22705778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = setupRawAudioFormat(
22715778822d86b0337407514b9372562b86edfa91cdAndreas Huber            encoder ? kPortIndexInput : kPortIndexOutput,
22725778822d86b0337407514b9372562b86edfa91cdAndreas Huber            sampleRate,
22735778822d86b0337407514b9372562b86edfa91cdAndreas Huber            numChannels);
22745778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22755778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
22765778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
22775778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
22785778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22795778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (encoder) {
22805778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = selectAudioPortFormat(kPortIndexOutput, OMX_AUDIO_CodingAAC);
22815778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22825778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
22835778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
22845778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
22855778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22865778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMX_PARAM_PORTDEFINITIONTYPE def;
22875778822d86b0337407514b9372562b86edfa91cdAndreas Huber        InitOMXParams(&def);
22885778822d86b0337407514b9372562b86edfa91cdAndreas Huber        def.nPortIndex = kPortIndexOutput;
22895778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22905778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = mOMX->getParameter(
22915778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
22925778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22935778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
22945778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
22955778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
22965778822d86b0337407514b9372562b86edfa91cdAndreas Huber
22975778822d86b0337407514b9372562b86edfa91cdAndreas Huber        def.format.audio.bFlagErrorConcealment = OMX_TRUE;
22985778822d86b0337407514b9372562b86edfa91cdAndreas Huber        def.format.audio.eEncoding = OMX_AUDIO_CodingAAC;
22995778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23005778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = mOMX->setParameter(
23015778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
23025778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23035778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
23045778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
23055778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
23065778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23075778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMX_AUDIO_PARAM_AACPROFILETYPE profile;
23085778822d86b0337407514b9372562b86edfa91cdAndreas Huber        InitOMXParams(&profile);
23095778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nPortIndex = kPortIndexOutput;
23105778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23115778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = mOMX->getParameter(
23125778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
23135778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23145778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
23155778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
23165778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
23175778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23185778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nChannels = numChannels;
23195778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23205778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.eChannelMode =
23215778822d86b0337407514b9372562b86edfa91cdAndreas Huber            (numChannels == 1)
23225778822d86b0337407514b9372562b86edfa91cdAndreas Huber                ? OMX_AUDIO_ChannelModeMono: OMX_AUDIO_ChannelModeStereo;
23235778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23245778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nSampleRate = sampleRate;
23255778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nBitRate = bitRate;
23265778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nAudioBandWidth = 0;
23275778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nFrameLength = 0;
23285778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nAACtools = OMX_AUDIO_AACToolAll;
23295778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.nAACERtools = OMX_AUDIO_AACERNone;
2330aeb8fd460ed87d032b3fb8bb61e21eb542ce0f5bDave Burke        profile.eAACProfile = (OMX_AUDIO_AACPROFILETYPE) aacProfile;
23315778822d86b0337407514b9372562b86edfa91cdAndreas Huber        profile.eAACStreamFormat = OMX_AUDIO_AACStreamFormatMP4FF;
2332b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi        switch (sbrMode) {
2333b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi        case 0:
2334b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            // disable sbr
2335b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools &= ~OMX_AUDIO_AACToolAndroidSSBR;
2336b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools &= ~OMX_AUDIO_AACToolAndroidDSBR;
2337b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            break;
2338b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi        case 1:
2339b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            // enable single-rate sbr
2340b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools |= OMX_AUDIO_AACToolAndroidSSBR;
2341b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools &= ~OMX_AUDIO_AACToolAndroidDSBR;
2342b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            break;
2343b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi        case 2:
2344b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            // enable dual-rate sbr
2345b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools &= ~OMX_AUDIO_AACToolAndroidSSBR;
2346b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools |= OMX_AUDIO_AACToolAndroidDSBR;
2347b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            break;
2348b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi        case -1:
2349b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            // enable both modes -> the codec will decide which mode should be used
2350b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools |= OMX_AUDIO_AACToolAndroidSSBR;
2351b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            profile.nAACtools |= OMX_AUDIO_AACToolAndroidDSBR;
2352b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            break;
2353b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi        default:
2354b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            // unsupported sbr mode
2355b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi            return BAD_VALUE;
2356b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi        }
2357b97cc6a9a3c81215e365c61b071a932073b56ff2Jean-Michel Trivi
23585778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23595778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = mOMX->setParameter(
23605778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
23615778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23625778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
23635778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
23645778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
23655778822d86b0337407514b9372562b86edfa91cdAndreas Huber
23665778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
23675778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
23685778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2369f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_AUDIO_PARAM_AACPROFILETYPE profile;
2370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&profile);
2371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nPortIndex = kPortIndexInput;
2372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
23735778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->getParameter(
2374f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
2375f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2376f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2377f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2378f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nChannels = numChannels;
2381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nSampleRate = sampleRate;
2382ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
2383ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    profile.eAACStreamFormat =
2384ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        isADTS
2385ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber            ? OMX_AUDIO_AACStreamFormatMP4ADTS
2386ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber            : OMX_AUDIO_AACStreamFormatMP4FF;
2387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
23888045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    OMX_AUDIO_PARAM_ANDROID_AACPRESENTATIONTYPE presentation;
23898045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    presentation.nMaxOutputChannels = maxOutputChannelCount;
23908045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    presentation.nDrcCut = drc.drcCut;
23918045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    presentation.nDrcBoost = drc.drcBoost;
23928045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    presentation.nHeavyCompression = drc.heavyCompression;
23938045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    presentation.nTargetReferenceLevel = drc.targetRefLevel;
23948045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    presentation.nEncodedTargetLevel = drc.encodedTargetLevel;
23952965f4eb7dceaf1173f0e2d93c11c28293aeead7Chong Zhang    presentation.nPCMLimiterEnable = pcmLimiterEnable;
23968045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi
23978045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    status_t res = mOMX->setParameter(mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
23988045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    if (res == OK) {
23998045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi        // optional parameters, will not cause configuration failure
24008045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi        mOMX->setParameter(mNode, (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidAacPresentation,
24018045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi                &presentation, sizeof(presentation));
24028045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    } else {
24038045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi        ALOGW("did not set AudioAndroidAacPresentation due to error %d when setting AudioAac", res);
24048045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    }
24058045853d03649f43ea2f7107e7d2dbb9b2d20855Jean-Michel Trivi    return res;
24065778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
2407f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
240897358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryustatus_t ACodec::setupAC3Codec(
240997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        bool encoder, int32_t numChannels, int32_t sampleRate) {
241097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    status_t err = setupRawAudioFormat(
241197358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            encoder ? kPortIndexInput : kPortIndexOutput, sampleRate, numChannels);
241297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
241397358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    if (err != OK) {
241497358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        return err;
241597358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    }
241697358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
241797358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    if (encoder) {
241897358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        ALOGW("AC3 encoding is not supported.");
241997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        return INVALID_OPERATION;
242097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    }
242197358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
242297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    OMX_AUDIO_PARAM_ANDROID_AC3TYPE def;
242397358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    InitOMXParams(&def);
242497358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    def.nPortIndex = kPortIndexInput;
242597358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
242697358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    err = mOMX->getParameter(
242797358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            mNode,
242897358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidAc3,
242997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            &def,
243097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            sizeof(def));
243197358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
243297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    if (err != OK) {
243397358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu        return err;
243497358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    }
243597358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
243697358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    def.nChannels = numChannels;
243797358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    def.nSampleRate = sampleRate;
243897358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
243997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu    return mOMX->setParameter(
244097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            mNode,
244197358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidAc3,
244297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            &def,
244397358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            sizeof(def));
244497358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu}
244597358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
24468a4728966dc9c78e21c3c93a927707e93c05e5e0Rachadstatus_t ACodec::setupEAC3Codec(
24478a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        bool encoder, int32_t numChannels, int32_t sampleRate) {
24488a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    status_t err = setupRawAudioFormat(
24498a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            encoder ? kPortIndexInput : kPortIndexOutput, sampleRate, numChannels);
24508a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24518a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    if (err != OK) {
24528a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        return err;
24538a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    }
24548a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24558a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    if (encoder) {
24568a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        ALOGW("EAC3 encoding is not supported.");
24578a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        return INVALID_OPERATION;
24588a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    }
24598a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24608a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    OMX_AUDIO_PARAM_ANDROID_EAC3TYPE def;
24618a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    InitOMXParams(&def);
24628a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    def.nPortIndex = kPortIndexInput;
24638a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24648a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    err = mOMX->getParameter(
24658a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            mNode,
24668a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidEac3,
24678a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            &def,
24688a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            sizeof(def));
24698a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24708a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    if (err != OK) {
24718a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad        return err;
24728a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    }
24738a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24748a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    def.nChannels = numChannels;
24758a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    def.nSampleRate = sampleRate;
24768a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24778a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad    return mOMX->setParameter(
24788a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            mNode,
24798a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidEac3,
24808a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            &def,
24818a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad            sizeof(def));
24828a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad}
24838a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
24845778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatic OMX_AUDIO_AMRBANDMODETYPE pickModeFromBitRate(
24855778822d86b0337407514b9372562b86edfa91cdAndreas Huber        bool isAMRWB, int32_t bps) {
24865778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (isAMRWB) {
24875778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (bps <= 6600) {
24885778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB0;
24895778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 8850) {
24905778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB1;
24915778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 12650) {
24925778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB2;
24935778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 14250) {
24945778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB3;
24955778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 15850) {
24965778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB4;
24975778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 18250) {
24985778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB5;
24995778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 19850) {
25005778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB6;
25015778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 23050) {
25025778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeWB7;
25035778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
25045778822d86b0337407514b9372562b86edfa91cdAndreas Huber
25055778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // 23850 bps
25065778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return OMX_AUDIO_AMRBandModeWB8;
25075778822d86b0337407514b9372562b86edfa91cdAndreas Huber    } else {  // AMRNB
25085778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (bps <= 4750) {
25095778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeNB0;
25105778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 5150) {
25115778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeNB1;
25125778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 5900) {
25135778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeNB2;
25145778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 6700) {
25155778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeNB3;
25165778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 7400) {
25175778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeNB4;
25185778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 7950) {
25195778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeNB5;
25205778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (bps <= 10200) {
25215778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OMX_AUDIO_AMRBandModeNB6;
25225778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
25235778822d86b0337407514b9372562b86edfa91cdAndreas Huber
25245778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // 12200 bps
25255778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return OMX_AUDIO_AMRBandModeNB7;
25265778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
2527f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2528f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
25295778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupAMRCodec(bool encoder, bool isWAMR, int32_t bitrate) {
2530729de186450f78c099637e1fce743fe531862c52Andreas Huber    OMX_AUDIO_PARAM_AMRTYPE def;
2531729de186450f78c099637e1fce743fe531862c52Andreas Huber    InitOMXParams(&def);
25325778822d86b0337407514b9372562b86edfa91cdAndreas Huber    def.nPortIndex = encoder ? kPortIndexOutput : kPortIndexInput;
2533729de186450f78c099637e1fce743fe531862c52Andreas Huber
2534729de186450f78c099637e1fce743fe531862c52Andreas Huber    status_t err =
2535729de186450f78c099637e1fce743fe531862c52Andreas Huber        mOMX->getParameter(mNode, OMX_IndexParamAudioAmr, &def, sizeof(def));
2536729de186450f78c099637e1fce743fe531862c52Andreas Huber
2537729de186450f78c099637e1fce743fe531862c52Andreas Huber    if (err != OK) {
2538729de186450f78c099637e1fce743fe531862c52Andreas Huber        return err;
2539729de186450f78c099637e1fce743fe531862c52Andreas Huber    }
2540729de186450f78c099637e1fce743fe531862c52Andreas Huber
2541729de186450f78c099637e1fce743fe531862c52Andreas Huber    def.eAMRFrameFormat = OMX_AUDIO_AMRFrameFormatFSF;
25425778822d86b0337407514b9372562b86edfa91cdAndreas Huber    def.eAMRBandMode = pickModeFromBitRate(isWAMR, bitrate);
25435778822d86b0337407514b9372562b86edfa91cdAndreas Huber
25445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->setParameter(
25455778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamAudioAmr, &def, sizeof(def));
2546729de186450f78c099637e1fce743fe531862c52Andreas Huber
25475778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
25485778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
25495778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
2550729de186450f78c099637e1fce743fe531862c52Andreas Huber
25515778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return setupRawAudioFormat(
25525778822d86b0337407514b9372562b86edfa91cdAndreas Huber            encoder ? kPortIndexInput : kPortIndexOutput,
25535778822d86b0337407514b9372562b86edfa91cdAndreas Huber            isWAMR ? 16000 : 8000 /* sampleRate */,
25545778822d86b0337407514b9372562b86edfa91cdAndreas Huber            1 /* numChannels */);
2555729de186450f78c099637e1fce743fe531862c52Andreas Huber}
2556729de186450f78c099637e1fce743fe531862c52Andreas Huber
255717c39e708ed657b8fa66f8acce5128e51696915cMarco Nelissenstatus_t ACodec::setupG711Codec(bool encoder, int32_t sampleRate, int32_t numChannels) {
2558777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (encoder) {
2559777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return INVALID_OPERATION;
2560777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
25615778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2562729de186450f78c099637e1fce743fe531862c52Andreas Huber    return setupRawAudioFormat(
256317c39e708ed657b8fa66f8acce5128e51696915cMarco Nelissen            kPortIndexInput, sampleRate, numChannels);
2564729de186450f78c099637e1fce743fe531862c52Andreas Huber}
2565729de186450f78c099637e1fce743fe531862c52Andreas Huber
25662f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivistatus_t ACodec::setupFlacCodec(
25672f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        bool encoder, int32_t numChannels, int32_t sampleRate, int32_t compressionLevel) {
25682f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi
25692f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi    if (encoder) {
25702f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        OMX_AUDIO_PARAM_FLACTYPE def;
25712f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        InitOMXParams(&def);
25722f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        def.nPortIndex = kPortIndexOutput;
25732f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi
25742f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        // configure compression level
25752f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        status_t err = mOMX->getParameter(mNode, OMX_IndexParamAudioFlac, &def, sizeof(def));
25762f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        if (err != OK) {
25772f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            ALOGE("setupFlacCodec(): Error %d getting OMX_IndexParamAudioFlac parameter", err);
25782f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            return err;
25792f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        }
25802f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        def.nCompressionLevel = compressionLevel;
25812f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        err = mOMX->setParameter(mNode, OMX_IndexParamAudioFlac, &def, sizeof(def));
25822f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        if (err != OK) {
25832f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            ALOGE("setupFlacCodec(): Error %d setting OMX_IndexParamAudioFlac parameter", err);
25842f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            return err;
25852f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi        }
25862f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi    }
25872f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi
25882f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi    return setupRawAudioFormat(
25892f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            encoder ? kPortIndexInput : kPortIndexOutput,
25902f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            sampleRate,
25912f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi            numChannels);
25922f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi}
25932f74ef3cdc192f817ee1121f41765f0852c1d81eJean-Michel Trivi
2594729de186450f78c099637e1fce743fe531862c52Andreas Huberstatus_t ACodec::setupRawAudioFormat(
2595729de186450f78c099637e1fce743fe531862c52Andreas Huber        OMX_U32 portIndex, int32_t sampleRate, int32_t numChannels) {
2596729de186450f78c099637e1fce743fe531862c52Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
2597729de186450f78c099637e1fce743fe531862c52Andreas Huber    InitOMXParams(&def);
2598729de186450f78c099637e1fce743fe531862c52Andreas Huber    def.nPortIndex = portIndex;
2599729de186450f78c099637e1fce743fe531862c52Andreas Huber
2600729de186450f78c099637e1fce743fe531862c52Andreas Huber    status_t err = mOMX->getParameter(
2601729de186450f78c099637e1fce743fe531862c52Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
2602729de186450f78c099637e1fce743fe531862c52Andreas Huber
2603729de186450f78c099637e1fce743fe531862c52Andreas Huber    if (err != OK) {
2604729de186450f78c099637e1fce743fe531862c52Andreas Huber        return err;
2605729de186450f78c099637e1fce743fe531862c52Andreas Huber    }
2606729de186450f78c099637e1fce743fe531862c52Andreas Huber
2607729de186450f78c099637e1fce743fe531862c52Andreas Huber    def.format.audio.eEncoding = OMX_AUDIO_CodingPCM;
2608729de186450f78c099637e1fce743fe531862c52Andreas Huber
2609729de186450f78c099637e1fce743fe531862c52Andreas Huber    err = mOMX->setParameter(
2610729de186450f78c099637e1fce743fe531862c52Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
2611729de186450f78c099637e1fce743fe531862c52Andreas Huber
2612729de186450f78c099637e1fce743fe531862c52Andreas Huber    if (err != OK) {
2613729de186450f78c099637e1fce743fe531862c52Andreas Huber        return err;
2614729de186450f78c099637e1fce743fe531862c52Andreas Huber    }
2615729de186450f78c099637e1fce743fe531862c52Andreas Huber
2616729de186450f78c099637e1fce743fe531862c52Andreas Huber    OMX_AUDIO_PARAM_PCMMODETYPE pcmParams;
2617729de186450f78c099637e1fce743fe531862c52Andreas Huber    InitOMXParams(&pcmParams);
2618729de186450f78c099637e1fce743fe531862c52Andreas Huber    pcmParams.nPortIndex = portIndex;
2619729de186450f78c099637e1fce743fe531862c52Andreas Huber
2620729de186450f78c099637e1fce743fe531862c52Andreas Huber    err = mOMX->getParameter(
2621729de186450f78c099637e1fce743fe531862c52Andreas Huber            mNode, OMX_IndexParamAudioPcm, &pcmParams, sizeof(pcmParams));
2622729de186450f78c099637e1fce743fe531862c52Andreas Huber
2623729de186450f78c099637e1fce743fe531862c52Andreas Huber    if (err != OK) {
2624729de186450f78c099637e1fce743fe531862c52Andreas Huber        return err;
2625729de186450f78c099637e1fce743fe531862c52Andreas Huber    }
2626729de186450f78c099637e1fce743fe531862c52Andreas Huber
2627729de186450f78c099637e1fce743fe531862c52Andreas Huber    pcmParams.nChannels = numChannels;
2628729de186450f78c099637e1fce743fe531862c52Andreas Huber    pcmParams.eNumData = OMX_NumericalDataSigned;
2629729de186450f78c099637e1fce743fe531862c52Andreas Huber    pcmParams.bInterleaved = OMX_TRUE;
2630729de186450f78c099637e1fce743fe531862c52Andreas Huber    pcmParams.nBitPerSample = 16;
2631729de186450f78c099637e1fce743fe531862c52Andreas Huber    pcmParams.nSamplingRate = sampleRate;
2632729de186450f78c099637e1fce743fe531862c52Andreas Huber    pcmParams.ePCMMode = OMX_AUDIO_PCMModeLinear;
2633729de186450f78c099637e1fce743fe531862c52Andreas Huber
2634c1d8115e8a0bdaeb2b723d395b9a85a02c90c933Andreas Huber    if (getOMXChannelMapping(numChannels, pcmParams.eChannelMapping) != OK) {
2635c1d8115e8a0bdaeb2b723d395b9a85a02c90c933Andreas Huber        return OMX_ErrorNone;
2636729de186450f78c099637e1fce743fe531862c52Andreas Huber    }
2637729de186450f78c099637e1fce743fe531862c52Andreas Huber
2638729de186450f78c099637e1fce743fe531862c52Andreas Huber    return mOMX->setParameter(
2639729de186450f78c099637e1fce743fe531862c52Andreas Huber            mNode, OMX_IndexParamAudioPcm, &pcmParams, sizeof(pcmParams));
2640729de186450f78c099637e1fce743fe531862c52Andreas Huber}
2641729de186450f78c099637e1fce743fe531862c52Andreas Huber
26425a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachadstatus_t ACodec::configureTunneledVideoPlayback(
264397827bd7c7e64dec22c8fe0f9e734a3c432ad7eeRachad        int32_t audioHwSync, const sp<ANativeWindow> &nativeWindow) {
26445a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    native_handle_t* sidebandHandle;
26455a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
26465a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    status_t err = mOMX->configureVideoTunnelMode(
26475a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad            mNode, kPortIndexOutput, OMX_TRUE, audioHwSync, &sidebandHandle);
26485a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    if (err != OK) {
26495a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad        ALOGE("configureVideoTunnelMode failed! (err %d).", err);
26505a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad        return err;
26515a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    }
26525a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
26535a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    err = native_window_set_sideband_stream(nativeWindow.get(), sidebandHandle);
26545a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    if (err != OK) {
26555a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad        ALOGE("native_window_set_sideband_stream(%p) failed! (err %d).",
26565a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad                sidebandHandle, err);
26575a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad        return err;
26585a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    }
26595a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
26605a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad    return OK;
26615a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad}
26625a446aafff3020d607ad6fb14cc7ae76dd8f7947Rachad
2663f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setVideoPortFormatType(
2664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex,
2665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_VIDEO_CODINGTYPE compressionFormat,
26660d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        OMX_COLOR_FORMATTYPE colorFormat,
26670d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        bool usingNativeBuffers) {
2668f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PARAM_PORTFORMATTYPE format;
2669f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&format);
2670f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nPortIndex = portIndex;
2671f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nIndex = 0;
2672f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool found = false;
2673f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2674f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 index = 0;
2675f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (;;) {
2676f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        format.nIndex = index;
2677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        status_t err = mOMX->getParameter(
2678f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, OMX_IndexParamVideoPortFormat,
2679f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &format, sizeof(format));
2680f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2681f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
2682f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return err;
2683f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2684f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2685229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        // substitute back flexible color format to codec supported format
2686229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        OMX_U32 flexibleEquivalent;
26870d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        if (compressionFormat == OMX_VIDEO_CodingUnused
26880d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                && isFlexibleColorFormat(
26890d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        mOMX, mNode, format.eColorFormat, usingNativeBuffers, &flexibleEquivalent)
26900d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                && colorFormat == flexibleEquivalent) {
2691229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            ALOGI("[%s] using color format %#x in place of %#x",
2692229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar                    mComponentName.c_str(), format.eColorFormat, colorFormat);
2693229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            colorFormat = format.eColorFormat;
2694229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        }
2695229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar
2696f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // The following assertion is violated by TI's video decoder.
2697f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // CHECK_EQ(format.nIndex, index);
2698f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2699f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (!strcmp("OMX.TI.Video.encoder", mComponentName.c_str())) {
2700f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (portIndex == kPortIndexInput
2701f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    && colorFormat == format.eColorFormat) {
2702f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // eCompressionFormat does not seem right.
2703f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                found = true;
2704f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
2705f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
2706f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (portIndex == kPortIndexOutput
2707f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    && compressionFormat == format.eCompressionFormat) {
2708f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // eColorFormat does not seem right.
2709f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                found = true;
2710f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
2711f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
2712f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2713f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2714f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (format.eCompressionFormat == compressionFormat
2715f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && format.eColorFormat == colorFormat) {
2716f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            found = true;
2717f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2718f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2719f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2720f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ++index;
2721f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2722f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2723f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!found) {
2724f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return UNKNOWN_ERROR;
2725f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2726f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2727f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->setParameter(
2728f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
2729f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
2730f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2731f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
2732f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2733f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
27340d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// Set optimal output format. OMX component lists output formats in the order
27350d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// of preference, but this got more complicated since the introduction of flexible
27360d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// YUV formats. We support a legacy behavior for applications that do not use
27370d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// surface output, do not specify an output format, but expect a "usable" standard
27380d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// OMX format. SW readable and standard formats must be flex-YUV.
27390d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar//
27400d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// Suggested preference order:
27410d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// - optimal format for texture rendering (mediaplayer behavior)
27420d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// - optimal SW readable & texture renderable format (flex-YUV support)
27430d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// - optimal SW readable non-renderable format (flex-YUV bytebuffer support)
27440d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// - legacy "usable" standard formats
27450d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar//
27460d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// For legacy support, we prefer a standard format, but will settle for a SW readable
27470d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar// flex-YUV format.
27480d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnarstatus_t ACodec::setSupportedOutputFormat(bool getLegacyFlexibleFormat) {
27490d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    OMX_VIDEO_PARAM_PORTFORMATTYPE format, legacyFormat;
2750f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&format);
2751f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nPortIndex = kPortIndexOutput;
2752f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
27530d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    InitOMXParams(&legacyFormat);
27540d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    // this field will change when we find a suitable legacy format
27550d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    legacyFormat.eColorFormat = OMX_COLOR_FormatUnused;
2756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
27570d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    for (OMX_U32 index = 0; ; ++index) {
27580d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        format.nIndex = index;
27590d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        status_t err = mOMX->getParameter(
27600d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                mNode, OMX_IndexParamVideoPortFormat,
27610d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                &format, sizeof(format));
27620d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        if (err != OK) {
27630d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            // no more formats, pick legacy format if found
27640d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            if (legacyFormat.eColorFormat != OMX_COLOR_FormatUnused) {
27650d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                 memcpy(&format, &legacyFormat, sizeof(format));
27660d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                 break;
27670d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            }
27680d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            return err;
27690d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        }
27700d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        if (format.eCompressionFormat != OMX_VIDEO_CodingUnused) {
27710d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            return OMX_ErrorBadParameter;
27720d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        }
27730d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        if (!getLegacyFlexibleFormat) {
27740d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            break;
27750d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        }
27760d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        // standard formats that were exposed to users before
27770d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        if (format.eColorFormat == OMX_COLOR_FormatYUV420Planar
27780d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                || format.eColorFormat == OMX_COLOR_FormatYUV420PackedPlanar
27790d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                || format.eColorFormat == OMX_COLOR_FormatYUV420SemiPlanar
27800d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                || format.eColorFormat == OMX_COLOR_FormatYUV420PackedSemiPlanar
27810d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                || format.eColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar) {
27820d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            break;
27830d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        }
27840d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        // find best legacy non-standard format
27850d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        OMX_U32 flexibleEquivalent;
27860d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        if (legacyFormat.eColorFormat == OMX_COLOR_FormatUnused
27870d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                && isFlexibleColorFormat(
27880d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        mOMX, mNode, format.eColorFormat, false /* usingNativeBuffers */,
27890d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        &flexibleEquivalent)
27900d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                && flexibleEquivalent == OMX_COLOR_FormatYUV420Flexible) {
27910d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            memcpy(&legacyFormat, &format, sizeof(format));
27920d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        }
27930d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    }
2794f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return mOMX->setParameter(
2795f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
2796f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
2797f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2798f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2799e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huberstatic const struct VideoCodingMapEntry {
2800e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    const char *mMime;
2801e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    OMX_VIDEO_CODINGTYPE mVideoCodingType;
2802e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber} kVideoCodingMapEntry[] = {
2803e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    { MEDIA_MIMETYPE_VIDEO_AVC, OMX_VIDEO_CodingAVC },
28042472b1c0d63454e5d90a982bd6c555de6c3127bdRachad    { MEDIA_MIMETYPE_VIDEO_HEVC, OMX_VIDEO_CodingHEVC },
2805e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG4, OMX_VIDEO_CodingMPEG4 },
2806e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    { MEDIA_MIMETYPE_VIDEO_H263, OMX_VIDEO_CodingH263 },
2807e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    { MEDIA_MIMETYPE_VIDEO_MPEG2, OMX_VIDEO_CodingMPEG2 },
280894705aff3c9eef58cbb72ec6fe5d2dcfd9481646hkuang    { MEDIA_MIMETYPE_VIDEO_VP8, OMX_VIDEO_CodingVP8 },
280994705aff3c9eef58cbb72ec6fe5d2dcfd9481646hkuang    { MEDIA_MIMETYPE_VIDEO_VP9, OMX_VIDEO_CodingVP9 },
2810e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber};
2811e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
28125778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatic status_t GetVideoCodingTypeFromMime(
28135778822d86b0337407514b9372562b86edfa91cdAndreas Huber        const char *mime, OMX_VIDEO_CODINGTYPE *codingType) {
2814e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    for (size_t i = 0;
2815e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber         i < sizeof(kVideoCodingMapEntry) / sizeof(kVideoCodingMapEntry[0]);
2816e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber         ++i) {
2817e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber        if (!strcasecmp(mime, kVideoCodingMapEntry[i].mMime)) {
2818e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber            *codingType = kVideoCodingMapEntry[i].mVideoCodingType;
2819e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber            return OK;
2820e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber        }
2821f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2822f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2823e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    *codingType = OMX_VIDEO_CodingUnused;
2824e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
2825e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    return ERROR_UNSUPPORTED;
2826e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber}
2827e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
2828e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huberstatic status_t GetMimeTypeForVideoCoding(
2829e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber        OMX_VIDEO_CODINGTYPE codingType, AString *mime) {
2830e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    for (size_t i = 0;
2831e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber         i < sizeof(kVideoCodingMapEntry) / sizeof(kVideoCodingMapEntry[0]);
2832e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber         ++i) {
2833e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber        if (codingType == kVideoCodingMapEntry[i].mVideoCodingType) {
2834e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber            *mime = kVideoCodingMapEntry[i].mMime;
2835e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber            return OK;
2836e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber        }
2837e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    }
2838e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
2839e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    mime->clear();
2840e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
2841e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber    return ERROR_UNSUPPORTED;
28425778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
28435778822d86b0337407514b9372562b86edfa91cdAndreas Huber
28445778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupVideoDecoder(
28450d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        const char *mime, const sp<AMessage> &msg, bool haveNativeWindow) {
284689869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar    int32_t width, height;
284789869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar    if (!msg->findInt32("width", &width)
284889869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar            || !msg->findInt32("height", &height)) {
284989869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar        return INVALID_OPERATION;
285089869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar    }
285189869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar
28525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_CODINGTYPE compressionFormat;
28535778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = GetVideoCodingTypeFromMime(mime, &compressionFormat);
28545778822d86b0337407514b9372562b86edfa91cdAndreas Huber
28555778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
28565778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
28575778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
28585778822d86b0337407514b9372562b86edfa91cdAndreas Huber
28595778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = setVideoPortFormatType(
2860f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexInput, compressionFormat, OMX_COLOR_FormatUnused);
2861f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2862f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2863f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2864f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2865f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
286689869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar    int32_t tmp;
286789869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar    if (msg->findInt32("color-format", &tmp)) {
286889869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar        OMX_COLOR_FORMATTYPE colorFormat =
286989869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar            static_cast<OMX_COLOR_FORMATTYPE>(tmp);
287089869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar        err = setVideoPortFormatType(
28710d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                kPortIndexOutput, OMX_VIDEO_CodingUnused, colorFormat, haveNativeWindow);
287289869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar        if (err != OK) {
287389869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar            ALOGW("[%s] does not support color format %d",
287489869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar                  mComponentName.c_str(), colorFormat);
28750d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar            err = setSupportedOutputFormat(!haveNativeWindow /* getLegacyFlexibleFormat */);
287689869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar        }
287789869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar    } else {
28780d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar        err = setSupportedOutputFormat(!haveNativeWindow /* getLegacyFlexibleFormat */);
287989869f692c35d0ca914c7de4a1f5ff63c9920634Lajos Molnar    }
2880f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2881f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2882f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2883f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2884f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
288578b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad    int32_t frameRateInt;
288678b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad    float frameRateFloat;
288778b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad    if (!msg->findFloat("frame-rate", &frameRateFloat)) {
288878b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad        if (!msg->findInt32("frame-rate", &frameRateInt)) {
288978b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad            frameRateInt = -1;
289078b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad        }
289178b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad        frameRateFloat = (float)frameRateInt;
289278b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad    }
289378b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad
2894f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setVideoFormatOnPort(
289578b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad            kPortIndexInput, width, height, compressionFormat, frameRateFloat);
2896f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2897f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2898f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2899f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2900f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2901f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setVideoFormatOnPort(
2902f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexOutput, width, height, OMX_VIDEO_CodingUnused);
2903f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2904f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
2905f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
2906f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2907f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2908f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
2909f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2910f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
29115778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupVideoEncoder(const char *mime, const sp<AMessage> &msg) {
29125778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t tmp;
29135778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("color-format", &tmp)) {
29145778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return INVALID_OPERATION;
29155778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29165778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29175778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_COLOR_FORMATTYPE colorFormat =
29185778822d86b0337407514b9372562b86edfa91cdAndreas Huber        static_cast<OMX_COLOR_FORMATTYPE>(tmp);
29195778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29205778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = setVideoPortFormatType(
29215778822d86b0337407514b9372562b86edfa91cdAndreas Huber            kPortIndexInput, OMX_VIDEO_CodingUnused, colorFormat);
29225778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29235778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
29245778822d86b0337407514b9372562b86edfa91cdAndreas Huber        ALOGE("[%s] does not support color format %d",
29255778822d86b0337407514b9372562b86edfa91cdAndreas Huber              mComponentName.c_str(), colorFormat);
29265778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29275778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
29285778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29295778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29305778822d86b0337407514b9372562b86edfa91cdAndreas Huber    /* Input port configuration */
29315778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29325778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
29335778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&def);
29345778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29355778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_PORTDEFINITIONTYPE *video_def = &def.format.video;
29365778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29375778822d86b0337407514b9372562b86edfa91cdAndreas Huber    def.nPortIndex = kPortIndexInput;
29385778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29395778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->getParameter(
29405778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
29415778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29425778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
29435778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
29445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29455778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29465778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t width, height, bitrate;
29475778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("width", &width)
29485778822d86b0337407514b9372562b86edfa91cdAndreas Huber            || !msg->findInt32("height", &height)
29495778822d86b0337407514b9372562b86edfa91cdAndreas Huber            || !msg->findInt32("bitrate", &bitrate)) {
29505778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return INVALID_OPERATION;
29515778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29525778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29535778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->nFrameWidth = width;
29545778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->nFrameHeight = height;
29555778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29565778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t stride;
29575778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("stride", &stride)) {
29585778822d86b0337407514b9372562b86edfa91cdAndreas Huber        stride = width;
29595778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29605778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29615778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->nStride = stride;
29625778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29635778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t sliceHeight;
29645778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("slice-height", &sliceHeight)) {
29655778822d86b0337407514b9372562b86edfa91cdAndreas Huber        sliceHeight = height;
29665778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29675778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29685778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->nSliceHeight = sliceHeight;
29695778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29705778822d86b0337407514b9372562b86edfa91cdAndreas Huber    def.nBufferSize = (video_def->nStride * video_def->nSliceHeight * 3) / 2;
29715778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29725778822d86b0337407514b9372562b86edfa91cdAndreas Huber    float frameRate;
29735778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findFloat("frame-rate", &frameRate)) {
29745778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t tmp;
29755778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("frame-rate", &tmp)) {
29765778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return INVALID_OPERATION;
29775778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
29785778822d86b0337407514b9372562b86edfa91cdAndreas Huber        frameRate = (float)tmp;
29792c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        mTimePerFrameUs = (int64_t) (1000000.0f / frameRate);
29805778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29815778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29825778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->xFramerate = (OMX_U32)(frameRate * 65536.0f);
29835778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->eCompressionFormat = OMX_VIDEO_CodingUnused;
2984a32d5435d9585794b72dd12546054f13adb845f2Lajos Molnar    // this is redundant as it was already set up in setVideoPortFormatType
2985a32d5435d9585794b72dd12546054f13adb845f2Lajos Molnar    // FIXME for now skip this only for flexible YUV formats
2986a32d5435d9585794b72dd12546054f13adb845f2Lajos Molnar    if (colorFormat != OMX_COLOR_FormatYUV420Flexible) {
2987a32d5435d9585794b72dd12546054f13adb845f2Lajos Molnar        video_def->eColorFormat = colorFormat;
2988a32d5435d9585794b72dd12546054f13adb845f2Lajos Molnar    }
29895778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29905778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->setParameter(
29915778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
29925778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29935778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
29945778822d86b0337407514b9372562b86edfa91cdAndreas Huber        ALOGE("[%s] failed to set input port definition parameters.",
29955778822d86b0337407514b9372562b86edfa91cdAndreas Huber              mComponentName.c_str());
29965778822d86b0337407514b9372562b86edfa91cdAndreas Huber
29975778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
29985778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
29995778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30005778822d86b0337407514b9372562b86edfa91cdAndreas Huber    /* Output port configuration */
30015778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30025778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_CODINGTYPE compressionFormat;
30035778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = GetVideoCodingTypeFromMime(mime, &compressionFormat);
30045778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30055778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
30065778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
30075778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
30085778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30095778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = setVideoPortFormatType(
30105778822d86b0337407514b9372562b86edfa91cdAndreas Huber            kPortIndexOutput, compressionFormat, OMX_COLOR_FormatUnused);
30115778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30125778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
30135778822d86b0337407514b9372562b86edfa91cdAndreas Huber        ALOGE("[%s] does not support compression format %d",
30145778822d86b0337407514b9372562b86edfa91cdAndreas Huber             mComponentName.c_str(), compressionFormat);
30155778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30165778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
30175778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
30185778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30195778822d86b0337407514b9372562b86edfa91cdAndreas Huber    def.nPortIndex = kPortIndexOutput;
30205778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30215778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->getParameter(
30225778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
30235778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30245778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
30255778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
30265778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
30275778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30285778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->nFrameWidth = width;
30295778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->nFrameHeight = height;
30305778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->xFramerate = 0;
30315778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->nBitrate = bitrate;
30325778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->eCompressionFormat = compressionFormat;
30335778822d86b0337407514b9372562b86edfa91cdAndreas Huber    video_def->eColorFormat = OMX_COLOR_FormatUnused;
30345778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30355778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->setParameter(
30365778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
30375778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30385778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
30395778822d86b0337407514b9372562b86edfa91cdAndreas Huber        ALOGE("[%s] failed to set output port definition parameters.",
30405778822d86b0337407514b9372562b86edfa91cdAndreas Huber              mComponentName.c_str());
30415778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30425778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
30435778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
30445778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30455778822d86b0337407514b9372562b86edfa91cdAndreas Huber    switch (compressionFormat) {
30465778822d86b0337407514b9372562b86edfa91cdAndreas Huber        case OMX_VIDEO_CodingMPEG4:
30475778822d86b0337407514b9372562b86edfa91cdAndreas Huber            err = setupMPEG4EncoderParameters(msg);
30485778822d86b0337407514b9372562b86edfa91cdAndreas Huber            break;
30495778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30505778822d86b0337407514b9372562b86edfa91cdAndreas Huber        case OMX_VIDEO_CodingH263:
30515778822d86b0337407514b9372562b86edfa91cdAndreas Huber            err = setupH263EncoderParameters(msg);
30525778822d86b0337407514b9372562b86edfa91cdAndreas Huber            break;
30535778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30545778822d86b0337407514b9372562b86edfa91cdAndreas Huber        case OMX_VIDEO_CodingAVC:
30555778822d86b0337407514b9372562b86edfa91cdAndreas Huber            err = setupAVCEncoderParameters(msg);
30565778822d86b0337407514b9372562b86edfa91cdAndreas Huber            break;
30575778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3058c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        case OMX_VIDEO_CodingHEVC:
3059c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            err = setupHEVCEncoderParameters(msg);
3060c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            break;
3061c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
306289b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber        case OMX_VIDEO_CodingVP8:
306389b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber        case OMX_VIDEO_CodingVP9:
306489b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber            err = setupVPXEncoderParameters(msg);
306589b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber            break;
306689b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber
30675778822d86b0337407514b9372562b86edfa91cdAndreas Huber        default:
30685778822d86b0337407514b9372562b86edfa91cdAndreas Huber            break;
30695778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
30705778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3071d36a023778256f0550e9a09e9993c066e8a3c7d7Ronghua Wu    if (err == OK) {
3072d36a023778256f0550e9a09e9993c066e8a3c7d7Ronghua Wu        ALOGI("setupVideoEncoder succeeded");
3073d36a023778256f0550e9a09e9993c066e8a3c7d7Ronghua Wu    }
30745778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30755778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return err;
30765778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
30775778822d86b0337407514b9372562b86edfa91cdAndreas Huber
30780dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dongstatus_t ACodec::setCyclicIntraMacroblockRefresh(const sp<AMessage> &msg, int32_t mode) {
30790dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    OMX_VIDEO_PARAM_INTRAREFRESHTYPE params;
30800dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    InitOMXParams(&params);
30810dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    params.nPortIndex = kPortIndexOutput;
30820dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong
30830dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    params.eRefreshMode = static_cast<OMX_VIDEO_INTRAREFRESHTYPE>(mode);
30840dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong
30850dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    if (params.eRefreshMode == OMX_VIDEO_IntraRefreshCyclic ||
30860dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            params.eRefreshMode == OMX_VIDEO_IntraRefreshBoth) {
30870dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        int32_t mbs;
30880dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        if (!msg->findInt32("intra-refresh-CIR-mbs", &mbs)) {
30890dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            return INVALID_OPERATION;
30900dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        }
30910dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        params.nCirMBs = mbs;
30920dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    }
30930dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong
30940dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    if (params.eRefreshMode == OMX_VIDEO_IntraRefreshAdaptive ||
30950dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            params.eRefreshMode == OMX_VIDEO_IntraRefreshBoth) {
30960dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        int32_t mbs;
30970dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        if (!msg->findInt32("intra-refresh-AIR-mbs", &mbs)) {
30980dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            return INVALID_OPERATION;
30990dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        }
31000dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        params.nAirMBs = mbs;
31010dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong
31020dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        int32_t ref;
31030dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        if (!msg->findInt32("intra-refresh-AIR-ref", &ref)) {
31040dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            return INVALID_OPERATION;
31050dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        }
31060dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        params.nAirRef = ref;
31070dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    }
31080dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong
31090dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    status_t err = mOMX->setParameter(
31100dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            mNode, OMX_IndexParamVideoIntraRefresh,
31110dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            &params, sizeof(params));
31120dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    return err;
31130dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong}
31140dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong
31155778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatic OMX_U32 setPFramesSpacing(int32_t iFramesInterval, int32_t frameRate) {
31165778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (iFramesInterval < 0) {
31175778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return 0xFFFFFFFF;
31185778822d86b0337407514b9372562b86edfa91cdAndreas Huber    } else if (iFramesInterval == 0) {
31195778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return 0;
31205778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
31215778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_U32 ret = frameRate * iFramesInterval;
31225778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return ret;
31235778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
31245778822d86b0337407514b9372562b86edfa91cdAndreas Huber
312596076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huberstatic OMX_VIDEO_CONTROLRATETYPE getBitrateMode(const sp<AMessage> &msg) {
312696076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    int32_t tmp;
312796076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    if (!msg->findInt32("bitrate-mode", &tmp)) {
312896076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber        return OMX_Video_ControlRateVariable;
312996076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    }
313096076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber
313196076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    return static_cast<OMX_VIDEO_CONTROLRATETYPE>(tmp);
313296076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber}
313396076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber
31345778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupMPEG4EncoderParameters(const sp<AMessage> &msg) {
31355778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t bitrate, iFrameInterval;
31365778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("bitrate", &bitrate)
31375778822d86b0337407514b9372562b86edfa91cdAndreas Huber            || !msg->findInt32("i-frame-interval", &iFrameInterval)) {
31385778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return INVALID_OPERATION;
31395778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
31405778822d86b0337407514b9372562b86edfa91cdAndreas Huber
314196076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    OMX_VIDEO_CONTROLRATETYPE bitrateMode = getBitrateMode(msg);
314296076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber
31435778822d86b0337407514b9372562b86edfa91cdAndreas Huber    float frameRate;
31445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findFloat("frame-rate", &frameRate)) {
31455778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t tmp;
31465778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("frame-rate", &tmp)) {
31475778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return INVALID_OPERATION;
31485778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
31495778822d86b0337407514b9372562b86edfa91cdAndreas Huber        frameRate = (float)tmp;
31505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
31515778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_PARAM_MPEG4TYPE mpeg4type;
31535778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&mpeg4type);
31545778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nPortIndex = kPortIndexOutput;
31555778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31565778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = mOMX->getParameter(
31575778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoMpeg4, &mpeg4type, sizeof(mpeg4type));
31585778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31595778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
31605778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
31615778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
31625778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31635778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nSliceHeaderSpacing = 0;
31645778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.bSVH = OMX_FALSE;
31655778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.bGov = OMX_FALSE;
31665778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31675778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nAllowedPictureTypes =
31685778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMX_VIDEO_PictureTypeI | OMX_VIDEO_PictureTypeP;
31695778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31705778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nPFrames = setPFramesSpacing(iFrameInterval, frameRate);
31715778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (mpeg4type.nPFrames == 0) {
31725778822d86b0337407514b9372562b86edfa91cdAndreas Huber        mpeg4type.nAllowedPictureTypes = OMX_VIDEO_PictureTypeI;
31735778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
31745778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nBFrames = 0;
31755778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nIDCVLCThreshold = 0;
31765778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.bACPred = OMX_TRUE;
31775778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nMaxPacketSize = 256;
31785778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nTimeIncRes = 1000;
31795778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.nHeaderExtension = 0;
31805778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mpeg4type.bReversibleVLC = OMX_FALSE;
31815778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31825778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t profile;
31835778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (msg->findInt32("profile", &profile)) {
31845778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t level;
31855778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("level", &level)) {
31865778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return INVALID_OPERATION;
31875778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
31885778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31895778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = verifySupportForProfileAndLevel(profile, level);
31905778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31915778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
31925778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
31935778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
31945778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31955778822d86b0337407514b9372562b86edfa91cdAndreas Huber        mpeg4type.eProfile = static_cast<OMX_VIDEO_MPEG4PROFILETYPE>(profile);
31965778822d86b0337407514b9372562b86edfa91cdAndreas Huber        mpeg4type.eLevel = static_cast<OMX_VIDEO_MPEG4LEVELTYPE>(level);
31975778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
31985778822d86b0337407514b9372562b86edfa91cdAndreas Huber
31995778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->setParameter(
32005778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoMpeg4, &mpeg4type, sizeof(mpeg4type));
32015778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32025778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
32035778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
32045778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32055778822d86b0337407514b9372562b86edfa91cdAndreas Huber
320696076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    err = configureBitrate(bitrate, bitrateMode);
32075778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32085778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
32095778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
32105778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32115778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32125778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return setupErrorCorrectionParameters();
32135778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
32145778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32155778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupH263EncoderParameters(const sp<AMessage> &msg) {
32165778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t bitrate, iFrameInterval;
32175778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("bitrate", &bitrate)
32185778822d86b0337407514b9372562b86edfa91cdAndreas Huber            || !msg->findInt32("i-frame-interval", &iFrameInterval)) {
32195778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return INVALID_OPERATION;
32205778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32215778822d86b0337407514b9372562b86edfa91cdAndreas Huber
322296076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    OMX_VIDEO_CONTROLRATETYPE bitrateMode = getBitrateMode(msg);
322396076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber
32245778822d86b0337407514b9372562b86edfa91cdAndreas Huber    float frameRate;
32255778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findFloat("frame-rate", &frameRate)) {
32265778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t tmp;
32275778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("frame-rate", &tmp)) {
32285778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return INVALID_OPERATION;
32295778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
32305778822d86b0337407514b9372562b86edfa91cdAndreas Huber        frameRate = (float)tmp;
32315778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32325778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32335778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_PARAM_H263TYPE h263type;
32345778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&h263type);
32355778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.nPortIndex = kPortIndexOutput;
32365778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32375778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = mOMX->getParameter(
32385778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoH263, &h263type, sizeof(h263type));
32395778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32405778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
32415778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
32425778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32435778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.nAllowedPictureTypes =
32455778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMX_VIDEO_PictureTypeI | OMX_VIDEO_PictureTypeP;
32465778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32475778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.nPFrames = setPFramesSpacing(iFrameInterval, frameRate);
32485778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (h263type.nPFrames == 0) {
32495778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h263type.nAllowedPictureTypes = OMX_VIDEO_PictureTypeI;
32505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32515778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.nBFrames = 0;
32525778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32535778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t profile;
32545778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (msg->findInt32("profile", &profile)) {
32555778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t level;
32565778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("level", &level)) {
32575778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return INVALID_OPERATION;
32585778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
32595778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32605778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = verifySupportForProfileAndLevel(profile, level);
32615778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32625778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
32635778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
32645778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
32655778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32665778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h263type.eProfile = static_cast<OMX_VIDEO_H263PROFILETYPE>(profile);
32675778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h263type.eLevel = static_cast<OMX_VIDEO_H263LEVELTYPE>(level);
32685778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32695778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32705778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.bPLUSPTYPEAllowed = OMX_FALSE;
32715778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.bForceRoundingTypeToZero = OMX_FALSE;
32725778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.nPictureHeaderRepetition = 0;
32735778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h263type.nGOBHeaderInterval = 0;
32745778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32755778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->setParameter(
32765778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoH263, &h263type, sizeof(h263type));
32775778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32785778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
32795778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
32805778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32815778822d86b0337407514b9372562b86edfa91cdAndreas Huber
328296076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    err = configureBitrate(bitrate, bitrateMode);
32835778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32845778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
32855778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
32865778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
32875778822d86b0337407514b9372562b86edfa91cdAndreas Huber
32885778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return setupErrorCorrectionParameters();
32895778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
32905778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3291a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar// static
3292a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnarint /* OMX_VIDEO_AVCLEVELTYPE */ ACodec::getAVCLevelFor(
3293a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        int width, int height, int rate, int bitrate,
3294a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        OMX_VIDEO_AVCPROFILETYPE profile) {
3295a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    // convert bitrate to main/baseline profile kbps equivalent
3296a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    switch (profile) {
3297a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        case OMX_VIDEO_AVCProfileHigh10:
3298a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar            bitrate = divUp(bitrate, 3000); break;
3299a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        case OMX_VIDEO_AVCProfileHigh:
3300a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar            bitrate = divUp(bitrate, 1250); break;
3301a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        default:
3302a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar            bitrate = divUp(bitrate, 1000); break;
3303a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    }
3304a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar
3305a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    // convert size and rate to MBs
3306a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    width = divUp(width, 16);
3307a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    height = divUp(height, 16);
3308a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    int mbs = width * height;
3309a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    rate *= mbs;
3310a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    int maxDimension = max(width, height);
3311a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar
3312a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    static const int limits[][5] = {
3313a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        /*   MBps     MB   dim  bitrate        level */
3314a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {    1485,    99,  28,     64, OMX_VIDEO_AVCLevel1  },
3315a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {    1485,    99,  28,    128, OMX_VIDEO_AVCLevel1b },
3316a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {    3000,   396,  56,    192, OMX_VIDEO_AVCLevel11 },
3317a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {    6000,   396,  56,    384, OMX_VIDEO_AVCLevel12 },
3318a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {   11880,   396,  56,    768, OMX_VIDEO_AVCLevel13 },
3319a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {   11880,   396,  56,   2000, OMX_VIDEO_AVCLevel2  },
3320a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {   19800,   792,  79,   4000, OMX_VIDEO_AVCLevel21 },
3321a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {   20250,  1620, 113,   4000, OMX_VIDEO_AVCLevel22 },
3322a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {   40500,  1620, 113,  10000, OMX_VIDEO_AVCLevel3  },
3323a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {  108000,  3600, 169,  14000, OMX_VIDEO_AVCLevel31 },
3324a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {  216000,  5120, 202,  20000, OMX_VIDEO_AVCLevel32 },
3325a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {  245760,  8192, 256,  20000, OMX_VIDEO_AVCLevel4  },
3326a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {  245760,  8192, 256,  50000, OMX_VIDEO_AVCLevel41 },
3327a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {  522240,  8704, 263,  50000, OMX_VIDEO_AVCLevel42 },
3328a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {  589824, 22080, 420, 135000, OMX_VIDEO_AVCLevel5  },
3329a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        {  983040, 36864, 543, 240000, OMX_VIDEO_AVCLevel51 },
3330a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        { 2073600, 36864, 543, 240000, OMX_VIDEO_AVCLevel52 },
3331a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    };
3332a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar
3333a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    for (size_t i = 0; i < ARRAY_SIZE(limits); i++) {
3334a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        const int (&limit)[5] = limits[i];
3335a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        if (rate <= limit[0] && mbs <= limit[1] && maxDimension <= limit[2]
3336a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar                && bitrate <= limit[3]) {
3337a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar            return limit[4];
3338a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar        }
3339a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    }
3340a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar    return 0;
3341a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar}
3342a147b4f91143d9f2fb608e22f9fca14bbd029573Lajos Molnar
33435778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupAVCEncoderParameters(const sp<AMessage> &msg) {
33445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t bitrate, iFrameInterval;
33455778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findInt32("bitrate", &bitrate)
33465778822d86b0337407514b9372562b86edfa91cdAndreas Huber            || !msg->findInt32("i-frame-interval", &iFrameInterval)) {
33475778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return INVALID_OPERATION;
33485778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
33495778822d86b0337407514b9372562b86edfa91cdAndreas Huber
335096076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    OMX_VIDEO_CONTROLRATETYPE bitrateMode = getBitrateMode(msg);
335196076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber
33525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    float frameRate;
33535778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!msg->findFloat("frame-rate", &frameRate)) {
33545778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t tmp;
33555778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("frame-rate", &tmp)) {
33565778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return INVALID_OPERATION;
33575778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
33585778822d86b0337407514b9372562b86edfa91cdAndreas Huber        frameRate = (float)tmp;
33595778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
33605778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33610dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    status_t err = OK;
33620dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    int32_t intraRefreshMode = 0;
33630dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    if (msg->findInt32("intra-refresh-mode", &intraRefreshMode)) {
33640dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        err = setCyclicIntraMacroblockRefresh(msg, intraRefreshMode);
33650dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        if (err != OK) {
33660dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            ALOGE("Setting intra macroblock refresh mode (%d) failed: 0x%x",
33670dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong                    err, intraRefreshMode);
33680dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong            return err;
33690dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong        }
33700dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    }
33710dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong
33725778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_PARAM_AVCTYPE h264type;
33735778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&h264type);
33745778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.nPortIndex = kPortIndexOutput;
33755778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33760dbe5a9321b24b6883fbb2fe97cd9d525128b0b5James Dong    err = mOMX->getParameter(
33775778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoAvc, &h264type, sizeof(h264type));
33785778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33795778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
33805778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
33815778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
33825778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33835778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.nAllowedPictureTypes =
33845778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMX_VIDEO_PictureTypeI | OMX_VIDEO_PictureTypeP;
33855778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33865778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t profile;
33875778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (msg->findInt32("profile", &profile)) {
33885778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t level;
33895778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("level", &level)) {
33905778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return INVALID_OPERATION;
33915778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
33925778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33935778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = verifySupportForProfileAndLevel(profile, level);
33945778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33955778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
33965778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
33975778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
33985778822d86b0337407514b9372562b86edfa91cdAndreas Huber
33995778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.eProfile = static_cast<OMX_VIDEO_AVCPROFILETYPE>(profile);
34005778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.eLevel = static_cast<OMX_VIDEO_AVCLEVELTYPE>(level);
34015778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
34025778822d86b0337407514b9372562b86edfa91cdAndreas Huber
34035778822d86b0337407514b9372562b86edfa91cdAndreas Huber    // XXX
34047c25df82dfc8bbedb58608242f0d923a4594bb14James Dong    if (h264type.eProfile != OMX_VIDEO_AVCProfileBaseline) {
34057c25df82dfc8bbedb58608242f0d923a4594bb14James Dong        ALOGW("Use baseline profile instead of %d for AVC recording",
34067c25df82dfc8bbedb58608242f0d923a4594bb14James Dong            h264type.eProfile);
34075778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.eProfile = OMX_VIDEO_AVCProfileBaseline;
34085778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
34095778822d86b0337407514b9372562b86edfa91cdAndreas Huber
34105778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (h264type.eProfile == OMX_VIDEO_AVCProfileBaseline) {
34115778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nSliceHeaderSpacing = 0;
34125778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.bUseHadamard = OMX_TRUE;
34135778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nRefFrames = 1;
34145778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nBFrames = 0;
34155778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nPFrames = setPFramesSpacing(iFrameInterval, frameRate);
34165778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (h264type.nPFrames == 0) {
34175778822d86b0337407514b9372562b86edfa91cdAndreas Huber            h264type.nAllowedPictureTypes = OMX_VIDEO_PictureTypeI;
34185778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
34195778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nRefIdx10ActiveMinus1 = 0;
34205778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nRefIdx11ActiveMinus1 = 0;
34215778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.bEntropyCodingCABAC = OMX_FALSE;
34225778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.bWeightedPPrediction = OMX_FALSE;
34235778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.bconstIpred = OMX_FALSE;
34245778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.bDirect8x8Inference = OMX_FALSE;
34255778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.bDirectSpatialTemporal = OMX_FALSE;
34265778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nCabacInitIdc = 0;
34275778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
34285778822d86b0337407514b9372562b86edfa91cdAndreas Huber
34295778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (h264type.nBFrames != 0) {
34305778822d86b0337407514b9372562b86edfa91cdAndreas Huber        h264type.nAllowedPictureTypes |= OMX_VIDEO_PictureTypeB;
34315778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
34325778822d86b0337407514b9372562b86edfa91cdAndreas Huber
34335778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.bEnableUEP = OMX_FALSE;
34345778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.bEnableFMO = OMX_FALSE;
34355778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.bEnableASO = OMX_FALSE;
34365778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.bEnableRS = OMX_FALSE;
34375778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.bFrameMBsOnly = OMX_TRUE;
34385778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.bMBAFF = OMX_FALSE;
34395778822d86b0337407514b9372562b86edfa91cdAndreas Huber    h264type.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterEnable;
34405778822d86b0337407514b9372562b86edfa91cdAndreas Huber
34415778822d86b0337407514b9372562b86edfa91cdAndreas Huber    err = mOMX->setParameter(
34425778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoAvc, &h264type, sizeof(h264type));
34435778822d86b0337407514b9372562b86edfa91cdAndreas Huber
34445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
34455778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
34465778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
34475778822d86b0337407514b9372562b86edfa91cdAndreas Huber
344896076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    return configureBitrate(bitrate, bitrateMode);
34495778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
34505778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3451c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachadstatus_t ACodec::setupHEVCEncoderParameters(const sp<AMessage> &msg) {
3452c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    int32_t bitrate, iFrameInterval;
3453c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    if (!msg->findInt32("bitrate", &bitrate)
3454c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            || !msg->findInt32("i-frame-interval", &iFrameInterval)) {
3455c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        return INVALID_OPERATION;
3456c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    }
3457c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3458c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    OMX_VIDEO_CONTROLRATETYPE bitrateMode = getBitrateMode(msg);
3459c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3460c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    float frameRate;
3461c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    if (!msg->findFloat("frame-rate", &frameRate)) {
3462c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        int32_t tmp;
3463c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        if (!msg->findInt32("frame-rate", &tmp)) {
3464c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            return INVALID_OPERATION;
3465c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        }
3466c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        frameRate = (float)tmp;
3467c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    }
3468c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3469c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    OMX_VIDEO_PARAM_HEVCTYPE hevcType;
3470c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    InitOMXParams(&hevcType);
3471c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    hevcType.nPortIndex = kPortIndexOutput;
3472c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3473c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    status_t err = OK;
3474c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    err = mOMX->getParameter(
3475c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            mNode, (OMX_INDEXTYPE)OMX_IndexParamVideoHevc, &hevcType, sizeof(hevcType));
3476c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    if (err != OK) {
3477c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        return err;
3478c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    }
3479c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3480c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    int32_t profile;
3481c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    if (msg->findInt32("profile", &profile)) {
3482c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        int32_t level;
3483c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        if (!msg->findInt32("level", &level)) {
3484c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            return INVALID_OPERATION;
3485c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        }
3486c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3487c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        err = verifySupportForProfileAndLevel(profile, level);
3488c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        if (err != OK) {
3489c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            return err;
3490c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        }
3491c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3492c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        hevcType.eProfile = static_cast<OMX_VIDEO_HEVCPROFILETYPE>(profile);
3493c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        hevcType.eLevel = static_cast<OMX_VIDEO_HEVCLEVELTYPE>(level);
3494c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    }
3495c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3496c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    // TODO: Need OMX structure definition for setting iFrameInterval
3497c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3498c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    err = mOMX->setParameter(
3499c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad            mNode, (OMX_INDEXTYPE)OMX_IndexParamVideoHevc, &hevcType, sizeof(hevcType));
3500c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    if (err != OK) {
3501c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad        return err;
3502c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    }
3503c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
3504c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad    return configureBitrate(bitrate, bitrateMode);
3505c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad}
3506c8efda9e9cd61dfe8e486c93fa8940b77cc3ccebRachad
350789b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huberstatus_t ACodec::setupVPXEncoderParameters(const sp<AMessage> &msg) {
350889b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber    int32_t bitrate;
35094154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    int32_t iFrameInterval = 0;
35104154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    size_t tsLayers = 0;
35114154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    OMX_VIDEO_ANDROID_VPXTEMPORALLAYERPATTERNTYPE pattern =
35124154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        OMX_VIDEO_VPXTemporalLayerPatternNone;
35134154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    static const uint32_t kVp8LayerRateAlloction
35144154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        [OMX_VIDEO_ANDROID_MAXVP8TEMPORALLAYERS]
35154154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        [OMX_VIDEO_ANDROID_MAXVP8TEMPORALLAYERS] = {
35164154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        {100, 100, 100},  // 1 layer
35174154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        { 60, 100, 100},  // 2 layers {60%, 40%}
35184154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        { 40,  60, 100},  // 3 layers {40%, 20%, 40%}
35194154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    };
352089b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber    if (!msg->findInt32("bitrate", &bitrate)) {
352189b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber        return INVALID_OPERATION;
352289b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber    }
35234154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    msg->findInt32("i-frame-interval", &iFrameInterval);
352489b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber
352589b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber    OMX_VIDEO_CONTROLRATETYPE bitrateMode = getBitrateMode(msg);
352689b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber
35274154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    float frameRate;
35284154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    if (!msg->findFloat("frame-rate", &frameRate)) {
35294154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        int32_t tmp;
35304154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        if (!msg->findInt32("frame-rate", &tmp)) {
35314154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            return INVALID_OPERATION;
35324154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        }
35334154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        frameRate = (float)tmp;
35344154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    }
35354154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev
35364154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    AString tsSchema;
35374154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    if (msg->findString("ts-schema", &tsSchema)) {
35384154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        if (tsSchema == "webrtc.vp8.1-layer") {
35394154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            pattern = OMX_VIDEO_VPXTemporalLayerPatternWebRTC;
35404154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            tsLayers = 1;
35414154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        } else if (tsSchema == "webrtc.vp8.2-layer") {
35424154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            pattern = OMX_VIDEO_VPXTemporalLayerPatternWebRTC;
35434154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            tsLayers = 2;
35444154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        } else if (tsSchema == "webrtc.vp8.3-layer") {
35454154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            pattern = OMX_VIDEO_VPXTemporalLayerPatternWebRTC;
35464154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            tsLayers = 3;
35474154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        } else {
35484154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            ALOGW("Unsupported ts-schema [%s]", tsSchema.c_str());
35494154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        }
35504154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    }
35514154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev
35524154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    OMX_VIDEO_PARAM_ANDROID_VP8ENCODERTYPE vp8type;
35534154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    InitOMXParams(&vp8type);
35544154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    vp8type.nPortIndex = kPortIndexOutput;
35554154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    status_t err = mOMX->getParameter(
35564154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            mNode, (OMX_INDEXTYPE)OMX_IndexParamVideoAndroidVp8Encoder,
35574154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            &vp8type, sizeof(vp8type));
35584154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev
35594154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    if (err == OK) {
35604154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        if (iFrameInterval > 0) {
35614154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            vp8type.nKeyFrameInterval = setPFramesSpacing(iFrameInterval, frameRate);
35624154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        }
35634154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        vp8type.eTemporalPattern = pattern;
35644154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        vp8type.nTemporalLayerCount = tsLayers;
35654154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        if (tsLayers > 0) {
35664154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            for (size_t i = 0; i < OMX_VIDEO_ANDROID_MAXVP8TEMPORALLAYERS; i++) {
35674154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev                vp8type.nTemporalLayerBitrateRatio[i] =
35684154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev                    kVp8LayerRateAlloction[tsLayers - 1][i];
35694154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            }
35704154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        }
35714154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        if (bitrateMode == OMX_Video_ControlRateConstant) {
35724154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            vp8type.nMinQuantizer = 2;
35734154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            vp8type.nMaxQuantizer = 63;
35744154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        }
35754154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev
35764154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        err = mOMX->setParameter(
35774154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev                mNode, (OMX_INDEXTYPE)OMX_IndexParamVideoAndroidVp8Encoder,
35784154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev                &vp8type, sizeof(vp8type));
35794154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        if (err != OK) {
35804154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev            ALOGW("Extended VP8 parameters set failed: %d", err);
35814154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev        }
35824154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev    }
35834154795d5526750b7aec5f774831a9e1ba0a3f15Alex Glaznev
358489b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber    return configureBitrate(bitrate, bitrateMode);
358589b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber}
358689b31a3f670d8bec87bed50aaad9bcd8edec66b6Andreas Huber
35875778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::verifySupportForProfileAndLevel(
35885778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t profile, int32_t level) {
35895778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_PARAM_PROFILELEVELTYPE params;
35905778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&params);
35915778822d86b0337407514b9372562b86edfa91cdAndreas Huber    params.nPortIndex = kPortIndexOutput;
35925778822d86b0337407514b9372562b86edfa91cdAndreas Huber
35935778822d86b0337407514b9372562b86edfa91cdAndreas Huber    for (params.nProfileIndex = 0;; ++params.nProfileIndex) {
35945778822d86b0337407514b9372562b86edfa91cdAndreas Huber        status_t err = mOMX->getParameter(
35955778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mNode,
35965778822d86b0337407514b9372562b86edfa91cdAndreas Huber                OMX_IndexParamVideoProfileLevelQuerySupported,
35975778822d86b0337407514b9372562b86edfa91cdAndreas Huber                &params,
35985778822d86b0337407514b9372562b86edfa91cdAndreas Huber                sizeof(params));
35995778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36005778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (err != OK) {
36015778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return err;
36025778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
36035778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36045778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t supportedProfile = static_cast<int32_t>(params.eProfile);
36055778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t supportedLevel = static_cast<int32_t>(params.eLevel);
36065778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36075778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (profile == supportedProfile && level <= supportedLevel) {
36085778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OK;
36095778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
36105778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
36115778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
36125778822d86b0337407514b9372562b86edfa91cdAndreas Huber
361396076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huberstatus_t ACodec::configureBitrate(
361496076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber        int32_t bitrate, OMX_VIDEO_CONTROLRATETYPE bitrateMode) {
36155778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_PARAM_BITRATETYPE bitrateType;
36165778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&bitrateType);
36175778822d86b0337407514b9372562b86edfa91cdAndreas Huber    bitrateType.nPortIndex = kPortIndexOutput;
36185778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36195778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = mOMX->getParameter(
36205778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoBitrate,
36215778822d86b0337407514b9372562b86edfa91cdAndreas Huber            &bitrateType, sizeof(bitrateType));
36225778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36235778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
36245778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return err;
36255778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
36265778822d86b0337407514b9372562b86edfa91cdAndreas Huber
362796076964863ee6887f0bed9d0f11f424b48ab9b9Andreas Huber    bitrateType.eControlRate = bitrateMode;
36285778822d86b0337407514b9372562b86edfa91cdAndreas Huber    bitrateType.nTargetBitrate = bitrate;
36295778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36305778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return mOMX->setParameter(
36315778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoBitrate,
36325778822d86b0337407514b9372562b86edfa91cdAndreas Huber            &bitrateType, sizeof(bitrateType));
36335778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
36345778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36355778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t ACodec::setupErrorCorrectionParameters() {
36365778822d86b0337407514b9372562b86edfa91cdAndreas Huber    OMX_VIDEO_PARAM_ERRORCORRECTIONTYPE errorCorrectionType;
36375778822d86b0337407514b9372562b86edfa91cdAndreas Huber    InitOMXParams(&errorCorrectionType);
36385778822d86b0337407514b9372562b86edfa91cdAndreas Huber    errorCorrectionType.nPortIndex = kPortIndexOutput;
36395778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36405778822d86b0337407514b9372562b86edfa91cdAndreas Huber    status_t err = mOMX->getParameter(
36415778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoErrorCorrection,
36425778822d86b0337407514b9372562b86edfa91cdAndreas Huber            &errorCorrectionType, sizeof(errorCorrectionType));
36435778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
36455778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return OK;  // Optional feature. Ignore this failure
36465778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
36475778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36485778822d86b0337407514b9372562b86edfa91cdAndreas Huber    errorCorrectionType.bEnableHEC = OMX_FALSE;
36495778822d86b0337407514b9372562b86edfa91cdAndreas Huber    errorCorrectionType.bEnableResync = OMX_TRUE;
36505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    errorCorrectionType.nResynchMarkerSpacing = 256;
36515778822d86b0337407514b9372562b86edfa91cdAndreas Huber    errorCorrectionType.bEnableDataPartitioning = OMX_FALSE;
36525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    errorCorrectionType.bEnableRVLC = OMX_FALSE;
36535778822d86b0337407514b9372562b86edfa91cdAndreas Huber
36545778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return mOMX->setParameter(
36555778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mNode, OMX_IndexParamVideoErrorCorrection,
36565778822d86b0337407514b9372562b86edfa91cdAndreas Huber            &errorCorrectionType, sizeof(errorCorrectionType));
36575778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
36585778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3659f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setVideoFormatOnPort(
3660f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex,
366178b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad        int32_t width, int32_t height, OMX_VIDEO_CODINGTYPE compressionFormat,
366278b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad        float frameRate) {
3663f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
3664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
3665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
3666f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3667f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PORTDEFINITIONTYPE *video_def = &def.format.video;
3668f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3669f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
3670f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3671777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (err != OK) {
3672777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return err;
3673777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
3674f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3675f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexInput) {
3676f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // XXX Need a (much) better heuristic to compute input buffer sizes.
3677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const size_t X = 64 * 1024;
3678f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (def.nBufferSize < X) {
3679f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.nBufferSize = X;
3680f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
3681f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
3682f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3683777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (def.eDomain != OMX_PortDomainVideo) {
3684777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        ALOGE("expected video port, got %s(%d)", asString(def.eDomain), def.eDomain);
3685777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return FAILED_TRANSACTION;
3686777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
3687f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3688f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    video_def->nFrameWidth = width;
3689f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    video_def->nFrameHeight = height;
3690f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3691f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexInput) {
3692f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        video_def->eCompressionFormat = compressionFormat;
3693f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        video_def->eColorFormat = OMX_COLOR_FormatUnused;
369478b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad        if (frameRate >= 0) {
369578b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad            video_def->xFramerate = (OMX_U32)(frameRate * 65536.0f);
369678b01639c08fe5e7e9c1be5e9dc5de560f1383f9Rachad        }
3697f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
3698f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3699f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
3700f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3701f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3702f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
3703f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
3704f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3705f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::initNativeWindow() {
3706f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mNativeWindow != NULL) {
3707f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_TRUE);
3708f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
3709f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3710f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_FALSE);
3711f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
3712f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
3713f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3714d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Hubersize_t ACodec::countBuffersOwnedByComponent(OMX_U32 portIndex) const {
3715d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber    size_t n = 0;
3716d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber
3717d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
3718d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber        const BufferInfo &info = mBuffers[portIndex].itemAt(i);
3719d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber
3720d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber        if (info.mStatus == BufferInfo::OWNED_BY_COMPONENT) {
3721d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber            ++n;
3722d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber        }
3723d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber    }
3724d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber
3725d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber    return n;
3726d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber}
3727d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber
37287e7013392e302a28364df1dcee79b82ad90978b4Andreas Hubersize_t ACodec::countBuffersOwnedByNativeWindow() const {
37297e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber    size_t n = 0;
37307e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber
37317e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber    for (size_t i = 0; i < mBuffers[kPortIndexOutput].size(); ++i) {
37327e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber        const BufferInfo &info = mBuffers[kPortIndexOutput].itemAt(i);
37337e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber
37347e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber        if (info.mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
37357e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber            ++n;
37367e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber        }
37377e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber    }
37387e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber
37397e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber    return n;
37407e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber}
37417e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber
37427e7013392e302a28364df1dcee79b82ad90978b4Andreas Hubervoid ACodec::waitUntilAllPossibleNativeWindowBuffersAreReturnedToUs() {
37437e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber    if (mNativeWindow == NULL) {
37447e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber        return;
37457e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber    }
37467e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber
3747e257e5ebefdd50f808ee3b4d1596db261c9b62ddLajos Molnar    while (countBuffersOwnedByNativeWindow() > mNumUndequeuedBuffers
37487e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber            && dequeueBufferFromNativeWindow() != NULL) {
3749c38fcfba95f711e5738e4c72bd5499317a2f30d9Lajos Molnar        // these buffers will be submitted as regular buffers; account for this
3750054219874873b41f1c815552987c10465c34ba2bLajos Molnar        if (storingMetadataInDecodedBuffers() && mMetadataBuffersToSubmit > 0) {
3751054219874873b41f1c815552987c10465c34ba2bLajos Molnar            --mMetadataBuffersToSubmit;
3752c38fcfba95f711e5738e4c72bd5499317a2f30d9Lajos Molnar        }
37537e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber    }
37547e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber}
37557e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber
3756f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::allYourBuffersAreBelongToUs(
3757f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
3758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
3759f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
3760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3761f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus != BufferInfo::OWNED_BY_US
3762f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                && info->mStatus != BufferInfo::OWNED_BY_NATIVE_WINDOW) {
3763609b815a3131d22da38b2f452faa9f89daad4039Andy Hung            ALOGV("[%s] Buffer %u on port %u still has status %d",
3764f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mComponentName.c_str(),
3765f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mBufferID, portIndex, info->mStatus);
3766f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
3767f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
3768f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
3769f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3770f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
3771f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
3772f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3773f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::allYourBuffersAreBelongToUs() {
3774f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return allYourBuffersAreBelongToUs(kPortIndexInput)
3775f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        && allYourBuffersAreBelongToUs(kPortIndexOutput);
3776f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
3777f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3778f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::deferMessage(const sp<AMessage> &msg) {
3779f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDeferredQueue.push_back(msg);
3780f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
3781f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3782f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::processDeferredMessages() {
3783f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    List<sp<AMessage> > queue = mDeferredQueue;
3784f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDeferredQueue.clear();
3785f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
3786f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    List<sp<AMessage> >::iterator it = queue.begin();
3787f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    while (it != queue.end()) {
3788f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        onMessageReceived(*it++);
3789f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
3790f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
3791f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
379203c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar// static
3793229d242665c612fd97431d1e7ac004823b47f181Lajos Molnarbool ACodec::describeDefaultColorFormat(DescribeColorFormatParams &params) {
379403c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    MediaImage &image = params.sMediaImage;
379503c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    memset(&image, 0, sizeof(image));
379603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
379703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mType = MediaImage::MEDIA_IMAGE_TYPE_UNKNOWN;
379803c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mNumPlanes = 0;
379903c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
380003c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    const OMX_COLOR_FORMATTYPE fmt = params.eColorFormat;
380103c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mWidth = params.nFrameWidth;
380203c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mHeight = params.nFrameHeight;
380303c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
380403c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    // only supporting YUV420
380503c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    if (fmt != OMX_COLOR_FormatYUV420Planar &&
380603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar        fmt != OMX_COLOR_FormatYUV420PackedPlanar &&
380703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar        fmt != OMX_COLOR_FormatYUV420SemiPlanar &&
38085a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar        fmt != OMX_COLOR_FormatYUV420PackedSemiPlanar &&
38095a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar        fmt != HAL_PIXEL_FORMAT_YV12) {
381003c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar        ALOGW("do not know color format 0x%x = %d", fmt, fmt);
3811229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        return false;
381203c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    }
381303c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
3814b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar    // TEMPORARY FIX for some vendors that advertise sliceHeight as 0
3815b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar    if (params.nStride != 0 && params.nSliceHeight == 0) {
3816b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar        ALOGW("using sliceHeight=%u instead of what codec advertised (=0)",
3817b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar                params.nFrameHeight);
3818b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar        params.nSliceHeight = params.nFrameHeight;
3819b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar    }
3820b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar
3821b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar    // we need stride and slice-height to be non-zero
3822b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar    if (params.nStride == 0 || params.nSliceHeight == 0) {
3823b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar        ALOGW("cannot describe color format 0x%x = %d with stride=%u and sliceHeight=%u",
3824b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar                fmt, fmt, params.nStride, params.nSliceHeight);
3825b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar        return false;
3826b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar    }
3827b32ebac7e3afb49b41eeccf130c8a96c1dae85d1Lajos Molnar
382803c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    // set-up YUV format
382903c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mType = MediaImage::MEDIA_IMAGE_TYPE_YUV;
383003c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mNumPlanes = 3;
383103c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mBitDepth = 8;
383203c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mPlane[image.Y].mOffset = 0;
383303c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mPlane[image.Y].mColInc = 1;
383403c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mPlane[image.Y].mRowInc = params.nStride;
383503c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mPlane[image.Y].mHorizSubsampling = 1;
383603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    image.mPlane[image.Y].mVertSubsampling = 1;
383703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
38385a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar    switch ((int)fmt) {
38395a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar        case HAL_PIXEL_FORMAT_YV12:
38405a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar            if (params.bUsingNativeBuffers) {
38415a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                size_t ystride = align(params.nStride, 16);
38425a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                size_t cstride = align(params.nStride / 2, 16);
38435a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.Y].mRowInc = ystride;
38445a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar
38455a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.V].mOffset = ystride * params.nSliceHeight;
38465a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.V].mColInc = 1;
38475a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.V].mRowInc = cstride;
38485a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.V].mHorizSubsampling = 2;
38495a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.V].mVertSubsampling = 2;
38505a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar
38515a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.U].mOffset = image.mPlane[image.V].mOffset
38525a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                        + (cstride * params.nSliceHeight / 2);
38535a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.U].mColInc = 1;
38545a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.U].mRowInc = cstride;
38555a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.U].mHorizSubsampling = 2;
38565a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                image.mPlane[image.U].mVertSubsampling = 2;
38575a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                break;
38585a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar            } else {
38595a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                // fall through as YV12 is used for YUV420Planar by some codecs
38605a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar            }
38615a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar
38625a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar        case OMX_COLOR_FormatYUV420Planar:
386303c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar        case OMX_COLOR_FormatYUV420PackedPlanar:
386403c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mOffset = params.nStride * params.nSliceHeight;
386503c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mColInc = 1;
386603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mRowInc = params.nStride / 2;
386703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mHorizSubsampling = 2;
386803c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mVertSubsampling = 2;
386903c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
387003c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mOffset = image.mPlane[image.U].mOffset
387103c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar                    + (params.nStride * params.nSliceHeight / 4);
387203c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mColInc = 1;
387303c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mRowInc = params.nStride / 2;
387403c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mHorizSubsampling = 2;
387503c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mVertSubsampling = 2;
387603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            break;
387703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
387803c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar        case OMX_COLOR_FormatYUV420SemiPlanar:
387903c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            // FIXME: NV21 for sw-encoder, NV12 for decoder and hw-encoder
388003c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar        case OMX_COLOR_FormatYUV420PackedSemiPlanar:
388103c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            // NV12
388203c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mOffset = params.nStride * params.nSliceHeight;
388303c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mColInc = 2;
388403c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mRowInc = params.nStride;
388503c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mHorizSubsampling = 2;
388603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.U].mVertSubsampling = 2;
388703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
388803c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mOffset = image.mPlane[image.U].mOffset + 1;
388903c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mColInc = 2;
389003c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mRowInc = params.nStride;
389103c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mHorizSubsampling = 2;
389203c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            image.mPlane[image.V].mVertSubsampling = 2;
389303c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            break;
389403c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
389503c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar        default:
389603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar            TRESPASS();
389703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar    }
3898229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    return true;
3899229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar}
3900229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar
3901229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar// static
3902229d242665c612fd97431d1e7ac004823b47f181Lajos Molnarbool ACodec::describeColorFormat(
3903229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        const sp<IOMX> &omx, IOMX::node_id node,
3904229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        DescribeColorFormatParams &describeParams)
3905229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar{
3906229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    OMX_INDEXTYPE describeColorFormatIndex;
3907229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    if (omx->getExtensionIndex(
3908229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            node, "OMX.google.android.index.describeColorFormat",
3909229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            &describeColorFormatIndex) != OK ||
3910229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        omx->getParameter(
3911229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            node, describeColorFormatIndex,
3912229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            &describeParams, sizeof(describeParams)) != OK) {
3913229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        return describeDefaultColorFormat(describeParams);
3914229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    }
3915229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    return describeParams.sMediaImage.mType !=
3916229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            MediaImage::MEDIA_IMAGE_TYPE_UNKNOWN;
3917229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar}
3918229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar
3919229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar// static
3920229d242665c612fd97431d1e7ac004823b47f181Lajos Molnarbool ACodec::isFlexibleColorFormat(
3921229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar         const sp<IOMX> &omx, IOMX::node_id node,
39220d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar         uint32_t colorFormat, bool usingNativeBuffers, OMX_U32 *flexibleEquivalent) {
3923229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    DescribeColorFormatParams describeParams;
3924229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    InitOMXParams(&describeParams);
3925229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    describeParams.eColorFormat = (OMX_COLOR_FORMATTYPE)colorFormat;
3926229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    // reasonable dummy values
3927229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    describeParams.nFrameWidth = 128;
3928229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    describeParams.nFrameHeight = 128;
3929229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    describeParams.nStride = 128;
3930229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    describeParams.nSliceHeight = 128;
39310d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar    describeParams.bUsingNativeBuffers = (OMX_BOOL)usingNativeBuffers;
3932229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar
3933229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    CHECK(flexibleEquivalent != NULL);
3934229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar
3935229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    if (!describeColorFormat(omx, node, describeParams)) {
3936229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        return false;
3937229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    }
3938229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar
3939229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    const MediaImage &img = describeParams.sMediaImage;
3940229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    if (img.mType == MediaImage::MEDIA_IMAGE_TYPE_YUV) {
3941229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        if (img.mNumPlanes != 3 ||
3942229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            img.mPlane[img.Y].mHorizSubsampling != 1 ||
3943229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            img.mPlane[img.Y].mVertSubsampling != 1) {
3944229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            return false;
3945229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        }
3946229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar
3947229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        // YUV 420
3948229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        if (img.mPlane[img.U].mHorizSubsampling == 2
3949229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar                && img.mPlane[img.U].mVertSubsampling == 2
3950229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar                && img.mPlane[img.V].mHorizSubsampling == 2
3951229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar                && img.mPlane[img.V].mVertSubsampling == 2) {
3952229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            // possible flexible YUV420 format
3953229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            if (img.mBitDepth <= 8) {
3954229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar               *flexibleEquivalent = OMX_COLOR_FormatYUV420Flexible;
3955229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar               return true;
3956229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar            }
3957229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar        }
3958229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    }
3959229d242665c612fd97431d1e7ac004823b47f181Lajos Molnar    return false;
396003c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar}
396103c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
3962e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnarstatus_t ACodec::getPortFormat(OMX_U32 portIndex, sp<AMessage> &notify) {
3963777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    const char *niceIndex = portIndex == kPortIndexInput ? "input" : "output";
396431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
396531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    InitOMXParams(&def);
3966e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    def.nPortIndex = portIndex;
396731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
3968777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    status_t err = mOMX->getParameter(mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
3969777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (err != OK) {
3970777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return err;
3971777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
397231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
3973777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (def.eDir != (portIndex == kPortIndexOutput ? OMX_DirOutput : OMX_DirInput)) {
3974777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        ALOGE("unexpected dir: %s(%d) on %s port", asString(def.eDir), def.eDir, niceIndex);
3975777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return BAD_VALUE;
3976777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
397731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
397831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    switch (def.eDomain) {
397931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        case OMX_PortDomainVideo:
398031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        {
398131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_VIDEO_PORTDEFINITIONTYPE *videoDef = &def.format.video;
3982e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar            switch ((int)videoDef->eCompressionFormat) {
3983e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                case OMX_VIDEO_CodingUnused:
3984e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                {
3985e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    CHECK(mIsEncoder ^ (portIndex == kPortIndexOutput));
3986e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setString("mime", MEDIA_MIMETYPE_VIDEO_RAW);
3987e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
3988e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setInt32("stride", videoDef->nStride);
3989e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setInt32("slice-height", videoDef->nSliceHeight);
3990e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setInt32("color-format", videoDef->eColorFormat);
3991e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
39920d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                    if (mNativeWindow == NULL) {
39930d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        DescribeColorFormatParams describeParams;
39940d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        InitOMXParams(&describeParams);
39950d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        describeParams.eColorFormat = videoDef->eColorFormat;
39960d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        describeParams.nFrameWidth = videoDef->nFrameWidth;
39970d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        describeParams.nFrameHeight = videoDef->nFrameHeight;
39980d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        describeParams.nStride = videoDef->nStride;
39990d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        describeParams.nSliceHeight = videoDef->nSliceHeight;
40000d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        describeParams.bUsingNativeBuffers = OMX_FALSE;
40010d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar
40020d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        if (describeColorFormat(mOMX, mNode, describeParams)) {
40030d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                            notify->setBuffer(
40040d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                                    "image-data",
40050d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                                    ABuffer::CreateAsCopy(
40060d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                                            &describeParams.sMediaImage,
40070d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                                            sizeof(describeParams.sMediaImage)));
40085a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar
40095a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                            MediaImage *img = &describeParams.sMediaImage;
40105a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                            ALOGV("[%s] MediaImage { F(%zux%zu) @%zu+%zu+%zu @%zu+%zu+%zu @%zu+%zu+%zu }",
40115a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                                    mComponentName.c_str(), img->mWidth, img->mHeight,
40125a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                                    img->mPlane[0].mOffset, img->mPlane[0].mColInc, img->mPlane[0].mRowInc,
40135a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                                    img->mPlane[1].mOffset, img->mPlane[1].mColInc, img->mPlane[1].mRowInc,
40145a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                                    img->mPlane[2].mOffset, img->mPlane[2].mColInc, img->mPlane[2].mRowInc);
40150d09182a1dde960f7acda1c28469e5deead1b996Lajos Molnar                        }
401603c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar                    }
401703c556ae1eb409ad088c49037e185946c54e1d25Lajos Molnar
401891a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                    if (portIndex != kPortIndexOutput) {
401991a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                        // TODO: also get input crop
402091a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                        break;
402191a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                    }
402291a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar
4023e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    OMX_CONFIG_RECTTYPE rect;
4024e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    InitOMXParams(&rect);
402591a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                    rect.nPortIndex = portIndex;
4026e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4027e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    if (mOMX->getConfig(
402891a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                                mNode,
402991a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                                (portIndex == kPortIndexOutput ?
403091a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                                        OMX_IndexConfigCommonOutputCrop :
403191a3cc00db31a713a25848f345bd624ac2ad8dc5Lajos Molnar                                        OMX_IndexConfigCommonInputCrop),
4032e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                                &rect, sizeof(rect)) != OK) {
4033e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                        rect.nLeft = 0;
4034e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                        rect.nTop = 0;
4035e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                        rect.nWidth = videoDef->nFrameWidth;
4036e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                        rect.nHeight = videoDef->nFrameHeight;
4037e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    }
403831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
4039777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (rect.nLeft < 0 ||
4040777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        rect.nTop < 0 ||
4041777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        rect.nLeft + rect.nWidth > videoDef->nFrameWidth ||
4042777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        rect.nTop + rect.nHeight > videoDef->nFrameHeight) {
4043777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        ALOGE("Wrong cropped rect (%d, %d) - (%u, %u) vs. frame (%u, %u)",
4044777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                rect.nLeft, rect.nTop,
4045777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                rect.nLeft + rect.nWidth, rect.nTop + rect.nHeight,
4046777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                videoDef->nFrameWidth, videoDef->nFrameHeight);
4047777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return BAD_VALUE;
4048777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4049e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4050e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setRect(
4051577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar                            "crop",
4052577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar                            rect.nLeft,
4053577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar                            rect.nTop,
4054e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                            rect.nLeft + rect.nWidth - 1,
4055e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                            rect.nTop + rect.nHeight - 1);
4056e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4057e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    break;
4058e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                }
40594730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia
40604730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                case OMX_VIDEO_CodingVP8:
40614730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                case OMX_VIDEO_CodingVP9:
40624730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                {
40634730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                    OMX_VIDEO_PARAM_ANDROID_VP8ENCODERTYPE vp8type;
40644730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                    InitOMXParams(&vp8type);
40654730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                    vp8type.nPortIndex = kPortIndexOutput;
40664730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                    status_t err = mOMX->getParameter(
40674730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                            mNode,
40684730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                            (OMX_INDEXTYPE)OMX_IndexParamVideoAndroidVp8Encoder,
40694730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                            &vp8type,
40704730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                            sizeof(vp8type));
40714730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia
40724730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                    if (err == OK) {
40734730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                        AString tsSchema = "none";
40744730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                        if (vp8type.eTemporalPattern
40754730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                == OMX_VIDEO_VPXTemporalLayerPatternWebRTC) {
40764730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                            switch (vp8type.nTemporalLayerCount) {
40774730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                case 1:
40784730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                {
40794730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                    tsSchema = "webrtc.vp8.1-layer";
40804730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                    break;
40814730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                }
40824730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                case 2:
40834730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                {
40844730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                    tsSchema = "webrtc.vp8.2-layer";
40854730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                    break;
40864730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                }
40874730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                case 3:
40884730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                {
40894730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                    tsSchema = "webrtc.vp8.3-layer";
40904730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                    break;
40914730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                }
40924730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                default:
40934730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                {
40944730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                    break;
40954730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                                }
40964730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                            }
40974730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                        }
40984730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                        notify->setString("ts-schema", tsSchema);
40994730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                    }
41004730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                    // Fall through to set up mime.
41014730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia                }
41024730fa07a3290cc3a904dcb16dbb92fcf46c36acWei Jia
4103e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                default:
4104e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                {
4105777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (mIsEncoder ^ (portIndex == kPortIndexOutput)) {
4106777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        // should be CodingUnused
4107777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        ALOGE("Raw port video compression format is %s(%d)",
4108777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                asString(videoDef->eCompressionFormat),
4109777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                videoDef->eCompressionFormat);
4110777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return BAD_VALUE;
4111777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4112e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    AString mime;
4113e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    if (GetMimeTypeForVideoCoding(
4114e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                        videoDef->eCompressionFormat, &mime) != OK) {
4115e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                        notify->setString("mime", "application/octet-stream");
4116e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    } else {
4117e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                        notify->setString("mime", mime.c_str());
4118e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    }
4119e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    break;
4120e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                }
412131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            }
4122e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar            notify->setInt32("width", videoDef->nFrameWidth);
4123e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar            notify->setInt32("height", videoDef->nFrameHeight);
41245a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar            ALOGV("[%s] %s format is %s", mComponentName.c_str(),
41255a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                    portIndex == kPortIndexInput ? "input" : "output",
41265a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar                    notify->debugString().c_str());
41275a52a060fcbe4804bcf4f61b8a457fe0e18a9014Lajos Molnar
412831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            break;
412931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        }
413031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
413131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        case OMX_PortDomainAudio:
413231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        {
413331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_AUDIO_PORTDEFINITIONTYPE *audioDef = &def.format.audio;
413431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
413597358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu            switch ((int)audioDef->eEncoding) {
4136e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                case OMX_AUDIO_CodingPCM:
4137e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                {
4138e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    OMX_AUDIO_PARAM_PCMMODETYPE params;
4139e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    InitOMXParams(&params);
4140e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    params.nPortIndex = portIndex;
4141e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4142777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4143777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, OMX_IndexParamAudioPcm, &params, sizeof(params));
4144777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4145777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4146777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
414714f7672b5d450ed26a06fd3bb3ce045ea78b11b2Andreas Huber
4148777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (params.nChannels <= 0
4149777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            || (params.nChannels != 1 && !params.bInterleaved)
4150777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            || params.nBitPerSample != 16u
4151777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            || params.eNumData != OMX_NumericalDataSigned
4152777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            || params.ePCMMode != OMX_AUDIO_PCMModeLinear) {
4153777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        ALOGE("unsupported PCM port: %u channels%s, %u-bit, %s(%d), %s(%d) mode ",
4154777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                params.nChannels,
4155777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                params.bInterleaved ? " interleaved" : "",
4156777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                params.nBitPerSample,
4157777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                asString(params.eNumData), params.eNumData,
4158777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                asString(params.ePCMMode), params.ePCMMode);
4159777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return FAILED_TRANSACTION;
4160777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4161e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4162e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_RAW);
4163e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setInt32("channel-count", params.nChannels);
4164e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setInt32("sample-rate", params.nSamplingRate);
4165e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4166e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    if (mChannelMaskPresent) {
4167e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                        notify->setInt32("channel-mask", mChannelMask);
41688b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen                    }
4169e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    break;
41708b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen                }
41719806555d3930be43e11106281dee354820ac1c88Andreas Huber
4172e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                case OMX_AUDIO_CodingAAC:
4173e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                {
4174e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    OMX_AUDIO_PARAM_AACPROFILETYPE params;
4175e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    InitOMXParams(&params);
4176e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    params.nPortIndex = portIndex;
4177e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4178777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4179777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, OMX_IndexParamAudioAac, &params, sizeof(params));
4180777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4181777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4182777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4183e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4184e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_AAC);
4185e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setInt32("channel-count", params.nChannels);
4186e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setInt32("sample-rate", params.nSampleRate);
4187e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    break;
4188e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                }
4189e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4190e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                case OMX_AUDIO_CodingAMR:
4191e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                {
4192e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    OMX_AUDIO_PARAM_AMRTYPE params;
4193e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    InitOMXParams(&params);
4194e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    params.nPortIndex = portIndex;
41959806555d3930be43e11106281dee354820ac1c88Andreas Huber
4196777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4197777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, OMX_IndexParamAudioAmr, &params, sizeof(params));
4198777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4199777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4200777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4201e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4202e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setInt32("channel-count", 1);
4203e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    if (params.eAMRBandMode >= OMX_AUDIO_AMRBandModeWB0) {
42040806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        notify->setString("mime", MEDIA_MIMETYPE_AUDIO_AMR_WB);
4205e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                        notify->setInt32("sample-rate", 16000);
4206e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    } else {
42070806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        notify->setString("mime", MEDIA_MIMETYPE_AUDIO_AMR_NB);
4208e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                        notify->setInt32("sample-rate", 8000);
4209e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    }
4210e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    break;
4211e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                }
4212e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4213e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                case OMX_AUDIO_CodingFLAC:
4214e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                {
4215e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    OMX_AUDIO_PARAM_FLACTYPE params;
4216e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    InitOMXParams(&params);
4217e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    params.nPortIndex = portIndex;
4218e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4219777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4220777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, OMX_IndexParamAudioFlac, &params, sizeof(params));
4221777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4222777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4223777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4224e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4225e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_FLAC);
4226e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setInt32("channel-count", params.nChannels);
4227e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    notify->setInt32("sample-rate", params.nSampleRate);
4228e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                    break;
4229e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                }
4230e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
4231e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                case OMX_AUDIO_CodingMP3:
4232e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                {
4233e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    OMX_AUDIO_PARAM_MP3TYPE params;
4234e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    InitOMXParams(&params);
4235e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    params.nPortIndex = portIndex;
4236e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4237777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4238777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, OMX_IndexParamAudioMp3, &params, sizeof(params));
4239777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4240777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4241777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4242e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4243e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_MPEG);
4244e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setInt32("channel-count", params.nChannels);
4245e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setInt32("sample-rate", params.nSampleRate);
4246e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    break;
4247e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                }
4248e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4249e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                case OMX_AUDIO_CodingVORBIS:
4250e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                {
4251e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    OMX_AUDIO_PARAM_VORBISTYPE params;
4252e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    InitOMXParams(&params);
4253e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    params.nPortIndex = portIndex;
4254e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4255777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4256777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, OMX_IndexParamAudioVorbis, &params, sizeof(params));
4257777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4258777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4259777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
4260e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4261e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_VORBIS);
4262e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setInt32("channel-count", params.nChannels);
4263e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    notify->setInt32("sample-rate", params.nSampleRate);
4264e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    break;
4265e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                }
4266e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
426797358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                case OMX_AUDIO_CodingAndroidAC3:
426897358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                {
426997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                    OMX_AUDIO_PARAM_ANDROID_AC3TYPE params;
427097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                    InitOMXParams(&params);
4271e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                    params.nPortIndex = portIndex;
427297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
4273777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4274777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidAc3,
4275777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            &params, sizeof(params));
4276777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4277777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4278777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
427997358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu
428097358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_AC3);
428197358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                    notify->setInt32("channel-count", params.nChannels);
428297358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                    notify->setInt32("sample-rate", params.nSampleRate);
428397358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                    break;
428497358c3e1adaf4a744cad78891a16d12e3e9c88eChangwan Ryu                }
4285e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber
42868a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                case OMX_AUDIO_CodingAndroidEAC3:
42878a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                {
42888a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                    OMX_AUDIO_PARAM_ANDROID_EAC3TYPE params;
42898a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                    InitOMXParams(&params);
42908a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                    params.nPortIndex = portIndex;
42918a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
4292777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4293777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidEac3,
4294777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            &params, sizeof(params));
4295777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4296777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4297777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
42988a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
42998a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_EAC3);
43008a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                    notify->setInt32("channel-count", params.nChannels);
43018a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                    notify->setInt32("sample-rate", params.nSampleRate);
43028a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                    break;
43038a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad                }
43048a4728966dc9c78e21c3c93a927707e93c05e5e0Rachad
43058c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                case OMX_AUDIO_CodingAndroidOPUS:
43068c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                {
43078c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                    OMX_AUDIO_PARAM_ANDROID_OPUSTYPE params;
43088c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                    InitOMXParams(&params);
43098c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                    params.nPortIndex = portIndex;
43108c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian
4311777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4312777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, (OMX_INDEXTYPE)OMX_IndexParamAudioAndroidOpus,
4313777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            &params, sizeof(params));
4314777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4315777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4316777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
43178c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian
43188c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_OPUS);
43198c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                    notify->setInt32("channel-count", params.nChannels);
43208c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                    notify->setInt32("sample-rate", params.nSampleRate);
43218c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                    break;
43228c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian                }
43238c6693b73cb4aed9d1dcb8d5d0828a77b7471a55Vignesh Venkatasubramanian
432410d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                case OMX_AUDIO_CodingG711:
432510d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                {
432610d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    OMX_AUDIO_PARAM_PCMMODETYPE params;
432710d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    InitOMXParams(&params);
432810d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    params.nPortIndex = portIndex;
432910d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang
4330777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4331777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mNode, (OMX_INDEXTYPE)OMX_IndexParamAudioPcm, &params, sizeof(params));
4332777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4333777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4334777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
433510d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang
433610d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    const char *mime = NULL;
433710d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    if (params.ePCMMode == OMX_AUDIO_PCMModeMULaw) {
433810d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                        mime = MEDIA_MIMETYPE_AUDIO_G711_MLAW;
433910d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    } else if (params.ePCMMode == OMX_AUDIO_PCMModeALaw) {
434010d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                        mime = MEDIA_MIMETYPE_AUDIO_G711_ALAW;
434110d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    } else { // params.ePCMMode == OMX_AUDIO_PCMModeLinear
434210d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                        mime = MEDIA_MIMETYPE_AUDIO_RAW;
434310d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    }
434410d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    notify->setString("mime", mime);
434510d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    notify->setInt32("channel-count", params.nChannels);
434610d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    notify->setInt32("sample-rate", params.nSamplingRate);
434710d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                    break;
434810d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                }
434941d3f579d2c166984958263533284209b90c87d5Marco Nelissen
435041d3f579d2c166984958263533284209b90c87d5Marco Nelissen                case OMX_AUDIO_CodingGSMFR:
435141d3f579d2c166984958263533284209b90c87d5Marco Nelissen                {
43520806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    OMX_AUDIO_PARAM_PCMMODETYPE params;
435341d3f579d2c166984958263533284209b90c87d5Marco Nelissen                    InitOMXParams(&params);
435441d3f579d2c166984958263533284209b90c87d5Marco Nelissen                    params.nPortIndex = portIndex;
435541d3f579d2c166984958263533284209b90c87d5Marco Nelissen
4356777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    err = mOMX->getParameter(
4357777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                                mNode, OMX_IndexParamAudioPcm, &params, sizeof(params));
4358777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    if (err != OK) {
4359777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                        return err;
4360777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    }
436141d3f579d2c166984958263533284209b90c87d5Marco Nelissen
436241d3f579d2c166984958263533284209b90c87d5Marco Nelissen                    notify->setString("mime", MEDIA_MIMETYPE_AUDIO_MSGSM);
436341d3f579d2c166984958263533284209b90c87d5Marco Nelissen                    notify->setInt32("channel-count", params.nChannels);
43640806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    notify->setInt32("sample-rate", params.nSamplingRate);
436541d3f579d2c166984958263533284209b90c87d5Marco Nelissen                    break;
436610d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang                }
436710d023beb0544591d0f5ff556f7f67356f0d9189Chong Zhang
4368e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber                default:
4369777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    ALOGE("Unsupported audio coding: %s(%d)\n",
4370777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            asString(audioDef->eEncoding), audioDef->eEncoding);
4371777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    return BAD_TYPE;
4372e96ee699aca0f711d41e6c0833e5de2341c4a36dAndreas Huber            }
437331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            break;
437431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        }
437531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
437631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        default:
4377777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            ALOGE("Unsupported domain: %s(%d)", asString(def.eDomain), def.eDomain);
4378777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            return BAD_TYPE;
437931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    }
438031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
4381e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    return OK;
4382e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar}
4383e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4384e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnarvoid ACodec::sendFormatChange(const sp<AMessage> &reply) {
43854e865a3cfe4c955e0890321a6b488cf661808b63Lajos Molnar    sp<AMessage> notify = mBaseOutputFormat->dup();
4386e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    notify->setInt32("what", kWhatOutputFormatChanged);
4387e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4388777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (getPortFormat(kPortIndexOutput, notify) != OK) {
4389777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        ALOGE("[%s] Failed to get port format to send format change", mComponentName.c_str());
4390777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return;
4391777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
4392e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4393e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    AString mime;
4394e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    CHECK(notify->findString("mime", &mime));
4395e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
4396e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    int32_t left, top, right, bottom;
4397e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    if (mime == MEDIA_MIMETYPE_VIDEO_RAW &&
4398e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        mNativeWindow != NULL &&
4399e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        notify->findRect("crop", &left, &top, &right, &bottom)) {
4400e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        // notify renderer of the crop change
4401e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        // NOTE: native window uses extended right-bottom coordinate
4402e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        reply->setRect("crop", left, top, right + 1, bottom + 1);
4403e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    } else if (mime == MEDIA_MIMETYPE_AUDIO_RAW &&
4404e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar               (mEncoderDelay || mEncoderPadding)) {
4405e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        int32_t channelCount;
4406e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        CHECK(notify->findInt32("channel-count", &channelCount));
4407e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        size_t frameSize = channelCount * sizeof(int16_t);
4408e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        if (mSkipCutBuffer != NULL) {
4409e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar            size_t prevbufsize = mSkipCutBuffer->size();
4410e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar            if (prevbufsize != 0) {
4411ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                ALOGW("Replacing SkipCutBuffer holding %zu bytes", prevbufsize);
4412e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar            }
4413e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        }
4414e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        mSkipCutBuffer = new SkipCutBuffer(
4415e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                mEncoderDelay * frameSize,
4416e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar                mEncoderPadding * frameSize);
4417e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    }
4418e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar
441931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    notify->post();
442031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
442131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    mSentFormat = true;
442231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber}
442331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
44245778822d86b0337407514b9372562b86edfa91cdAndreas Hubervoid ACodec::signalError(OMX_ERRORTYPE error, status_t internalError) {
4425cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber    sp<AMessage> notify = mNotify->dup();
4426d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar    notify->setInt32("what", CodecBase::kWhatError);
4427251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    ALOGE("signalError(omxError %#x, internalError %d)", error, internalError);
4428251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung
4429251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    if (internalError == UNKNOWN_ERROR) { // find better error code
4430251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        const status_t omxStatus = statusFromOMXError(error);
4431251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        if (omxStatus != 0) {
4432251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung            internalError = omxStatus;
4433251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        } else {
4434251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung            ALOGW("Invalid OMX error %#x", error);
4435251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        }
4436251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    }
44375778822d86b0337407514b9372562b86edfa91cdAndreas Huber    notify->setInt32("err", internalError);
4438251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    notify->setInt32("actionCode", ACTION_CODE_FATAL); // could translate from OMX error.
4439cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber    notify->post();
4440cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber}
4441cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber
4442f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
4443f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4444eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas HuberACodec::PortDescription::PortDescription() {
4445eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber}
4446eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
4447496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huberstatus_t ACodec::requestIDRFrame() {
4448496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber    if (!mIsEncoder) {
4449496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber        return ERROR_UNSUPPORTED;
4450496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber    }
4451496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
4452496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber    OMX_CONFIG_INTRAREFRESHVOPTYPE params;
4453496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber    InitOMXParams(&params);
4454496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
4455496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber    params.nPortIndex = kPortIndexOutput;
4456496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber    params.IntraRefreshVOP = OMX_TRUE;
4457496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
4458496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber    return mOMX->setConfig(
4459496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            mNode,
4460496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            OMX_IndexConfigVideoIntraVOPRefresh,
4461496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            &params,
4462496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            sizeof(params));
4463496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber}
4464496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
4465eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Hubervoid ACodec::PortDescription::addBuffer(
4466eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber        IOMX::buffer_id id, const sp<ABuffer> &buffer) {
4467eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber    mBufferIDs.push_back(id);
4468eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber    mBuffers.push_back(buffer);
4469eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber}
4470eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
4471eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Hubersize_t ACodec::PortDescription::countBuffers() {
4472eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber    return mBufferIDs.size();
4473eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber}
4474eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
4475eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas HuberIOMX::buffer_id ACodec::PortDescription::bufferIDAt(size_t index) const {
4476eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber    return mBufferIDs.itemAt(index);
4477eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber}
4478eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
4479eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Hubersp<ABuffer> ACodec::PortDescription::bufferAt(size_t index) const {
4480eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber    return mBuffers.itemAt(index);
4481eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber}
4482eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
4483eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber////////////////////////////////////////////////////////////////////////////////
4484eb61431af13741aa8b7e57a39f69bba5a6c190dcAndreas Huber
4485f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::BaseState(ACodec *codec, const sp<AState> &parentState)
4486f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : AState(parentState),
4487f933441648ef6a71dee783d733aac17b9508b452Andreas Huber      mCodec(codec) {
4488f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4489f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4490ccb067b1d8424ba610cbd3de83368bd55b532b5bAndreas HuberACodec::BaseState::PortMode ACodec::BaseState::getPortMode(
4491ccb067b1d8424ba610cbd3de83368bd55b532b5bAndreas Huber        OMX_U32 /* portIndex */) {
4492f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return KEEP_BUFFERS;
4493f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4494f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4495f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onMessageReceived(const sp<AMessage> &msg) {
4496f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
4497f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatInputBufferFilled:
4498f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
4499f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onInputBufferFilled(msg);
4500f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
4501f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4502f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4503f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatOutputBufferDrained:
4504f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
4505f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onOutputBufferDrained(msg);
4506f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
4507f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4508f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
450926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        case ACodec::kWhatOMXMessageList:
451026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        {
451126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            return checkOMXMessage(msg) ? onOMXMessageList(msg) : true;
451226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        }
451326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
4514f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatOMXMessage:
4515f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
451626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar            return checkOMXMessage(msg) ? onOMXMessage(msg) : true;
4517f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
45191dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        case ACodec::kWhatSetSurface:
45201dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        {
45211dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            sp<AReplyToken> replyID;
45221dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            CHECK(msg->senderAwaitsResponse(&replyID));
45231dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
45241dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            sp<RefBase> obj;
45251dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            CHECK(msg->findObject("surface", &obj));
45261dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
4527011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar            status_t err =
4528011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                ADebug::isExperimentEnabled("legacy-setsurface") ? BAD_VALUE :
4529011778fd4cb9606b68bfea1ed483d73c04fc6dcdLajos Molnar                        mCodec->handleSetSurface(static_cast<Surface *>(obj.get()));
45301dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
45311dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            sp<AMessage> response = new AMessage;
45321dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            response->setInt32("err", err);
45331dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            response->postReply(replyID);
45341dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar            break;
45351dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar        }
45361dcdfead2971c1fa7c02f24ba86f706890c9f99eLajos Molnar
45377cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        case ACodec::kWhatCreateInputSurface:
45388f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang        case ACodec::kWhatSetInputSurface:
45397cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        case ACodec::kWhatSignalEndOfInputStream:
45407cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        {
4541251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung            // This may result in an app illegal state exception.
45427cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            ALOGE("Message 0x%x was not handled", msg->what());
45437cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            mCodec->signalError(OMX_ErrorUndefined, INVALID_OPERATION);
45447cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            return true;
45457cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        }
45467cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden
4547ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        case ACodec::kWhatOMXDied:
4548ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        {
4549251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung            // This will result in kFlagSawMediaServerDie handling in MediaCodec.
4550ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber            ALOGE("OMX/mediaserver died, signalling error!");
4551ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber            mCodec->signalError(OMX_ErrorResourcesLost, DEAD_OBJECT);
4552ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber            break;
4553ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        }
4554ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
455530358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar        case ACodec::kWhatReleaseCodecInstance:
455630358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar        {
455730358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            ALOGI("[%s] forcing the release of codec",
455830358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar                    mCodec->mComponentName.c_str());
455930358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            status_t err = mCodec->mOMX->freeNode(mCodec->mNode);
456030358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            ALOGE_IF("[%s] failed to release codec instance: err=%d",
456130358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar                       mCodec->mComponentName.c_str(), err);
456230358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            sp<AMessage> notify = mCodec->mNotify->dup();
456330358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            notify->setInt32("what", CodecBase::kWhatShutdownCompleted);
456430358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            notify->post();
456530358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            break;
456630358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar        }
456730358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar
4568f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
4569f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
4570f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4571f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4572f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
4573f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4574f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
457526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnarbool ACodec::BaseState::checkOMXMessage(const sp<AMessage> &msg) {
45765e184b0d12f1ec436246a391da8d9355cc21ee08Lajos Molnar    // there is a possibility that this is an outstanding message for a
45775e184b0d12f1ec436246a391da8d9355cc21ee08Lajos Molnar    // codec that we have already destroyed
4578ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    if (mCodec->mNode == 0) {
45795e184b0d12f1ec436246a391da8d9355cc21ee08Lajos Molnar        ALOGI("ignoring message as already freed component: %s",
45805e184b0d12f1ec436246a391da8d9355cc21ee08Lajos Molnar                msg->debugString().c_str());
458126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        return false;
45825e184b0d12f1ec436246a391da8d9355cc21ee08Lajos Molnar    }
45835e184b0d12f1ec436246a391da8d9355cc21ee08Lajos Molnar
4584f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::node_id nodeID;
4585609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    CHECK(msg->findInt32("node", (int32_t*)&nodeID));
4586777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (nodeID != mCodec->mNode) {
4587777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        ALOGE("Unexpected message for nodeID: %u, should have been %u", nodeID, mCodec->mNode);
4588777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        return false;
4589777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
459026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    return true;
459126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar}
459226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
459326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnarbool ACodec::BaseState::onOMXMessageList(const sp<AMessage> &msg) {
459426a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    sp<RefBase> obj;
459526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    CHECK(msg->findObject("messages", &obj));
459626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    sp<MessageList> msgList = static_cast<MessageList *>(obj.get());
459726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
459890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    bool receivedRenderedEvents = false;
459926a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    for (std::list<sp<AMessage>>::const_iterator it = msgList->getList().cbegin();
460026a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar          it != msgList->getList().cend(); ++it) {
460126a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar        onOMXMessage(*it);
460290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        int32_t type;
460390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        CHECK((*it)->findInt32("type", &type));
460490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        if (type == omx_message::FRAME_RENDERED) {
460590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar            receivedRenderedEvents = true;
460690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        }
460790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    }
460890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
460990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    if (receivedRenderedEvents) {
461090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        // NOTE: all buffers are rendered in this case
461190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        mCodec->notifyOfRenderedFrames();
461226a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    }
461326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    return true;
461426a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar}
461526a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar
461626a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnarbool ACodec::BaseState::onOMXMessage(const sp<AMessage> &msg) {
461726a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    int32_t type;
461826a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    CHECK(msg->findInt32("type", &type));
4619f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4620f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (type) {
4621f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::EVENT:
4622f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
4623f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int32_t event, data1, data2;
4624f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("event", &event));
4625f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("data1", &data1));
4626f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("data2", &data2));
4627f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
46280af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber            if (event == OMX_EventCmdComplete
46290af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                    && data1 == OMX_CommandFlush
46300af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                    && data2 == (int32_t)OMX_ALL) {
46310af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // Use of this notification is not consistent across
46320af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // implementations. We'll drop this notification and rely
46330af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // on flush-complete notifications on the individual port
46340af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // indices instead.
46350af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
46360af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                return true;
46370af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber            }
46380af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
4639f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXEvent(
4640f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_EVENTTYPE>(event),
4641f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_U32>(data1),
4642f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_U32>(data2));
4643f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4644f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4645f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::EMPTY_BUFFER_DONE:
4646f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
4647f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID;
464815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            int32_t fenceFd;
464915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
4650609b815a3131d22da38b2f452faa9f89daad4039Andy Hung            CHECK(msg->findInt32("buffer", (int32_t*)&bufferID));
465115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            CHECK(msg->findInt32("fence_fd", &fenceFd));
4652f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
465315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            return onOMXEmptyBufferDone(bufferID, fenceFd);
4654f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4655f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4656f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::FILL_BUFFER_DONE:
4657f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
4658f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID;
4659609b815a3131d22da38b2f452faa9f89daad4039Andy Hung            CHECK(msg->findInt32("buffer", (int32_t*)&bufferID));
4660f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
466115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            int32_t rangeOffset, rangeLength, flags, fenceFd;
4662f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int64_t timeUs;
4663f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("range_offset", &rangeOffset));
4665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("range_length", &rangeLength));
4666f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("flags", &flags));
4667f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt64("timestamp", &timeUs));
466815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            CHECK(msg->findInt32("fence_fd", &fenceFd));
4669f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4670f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXFillBufferDone(
4671f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    bufferID,
4672f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    (size_t)rangeOffset, (size_t)rangeLength,
4673f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    (OMX_U32)flags,
467415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    timeUs,
467515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    fenceFd);
4676f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
467890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        case omx_message::FRAME_RENDERED:
467990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        {
468090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar            int64_t mediaTimeUs, systemNano;
468190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
468290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar            CHECK(msg->findInt64("media_time_us", &mediaTimeUs));
468390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar            CHECK(msg->findInt64("system_nano", &systemNano));
468490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
468590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar            return onOMXFrameRendered(
468690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                    mediaTimeUs, systemNano);
468790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        }
468890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
4689f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
4690777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            ALOGE("Unexpected message type: %d", type);
4691777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            return false;
4692f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4693f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4694f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
469590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnarbool ACodec::BaseState::onOMXFrameRendered(
469690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        int64_t mediaTimeUs __unused, nsecs_t systemNano __unused) {
469790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    // ignore outside of Executing and PortSettingsChanged states
469890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    return true;
469990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar}
470090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
4701f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXEvent(
4702f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
4703f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (event != OMX_EventError) {
4704ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar        ALOGV("[%s] EVENT(%d, 0x%08x, 0x%08x)",
4705f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             mCodec->mComponentName.c_str(), event, data1, data2);
4706f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4707f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return false;
4708f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4709f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4710ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    ALOGE("[%s] ERROR(0x%08x)", mCodec->mComponentName.c_str(), data1);
4711f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4712251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    // verify OMX component sends back an error we expect.
4713251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    OMX_ERRORTYPE omxError = (OMX_ERRORTYPE)data1;
4714251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    if (!isOMXError(omxError)) {
4715251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        ALOGW("Invalid OMX error %#x", omxError);
4716251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        omxError = OMX_ErrorUndefined;
4717251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    }
4718251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung    mCodec->signalError(omxError);
4719f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4720f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
4721f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4722f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
472315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnarbool ACodec::BaseState::onOMXEmptyBufferDone(IOMX::buffer_id bufferID, int fenceFd) {
4724ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    ALOGV("[%s] onOMXEmptyBufferDone %u",
4725349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber         mCodec->mComponentName.c_str(), bufferID);
4726349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
47270806340688c937e7b78c2d89db3809274130df4eLajos Molnar    BufferInfo *info = mCodec->findBufferByID(kPortIndexInput, bufferID);
47280806340688c937e7b78c2d89db3809274130df4eLajos Molnar    BufferInfo::Status status = BufferInfo::getSafeStatus(info);
47290806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (status != BufferInfo::OWNED_BY_COMPONENT) {
47300806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGE("Wrong ownership in EBD: %s(%d) buffer #%u", _asString(status), status, bufferID);
47310806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->dumpBuffers(kPortIndexInput);
473215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        if (fenceFd >= 0) {
473315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            ::close(fenceFd);
473415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        }
47350806340688c937e7b78c2d89db3809274130df4eLajos Molnar        return false;
47360806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
4737f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
4738f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
473915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    // input buffers cannot take fences, so wait for any fence now
474015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    (void)mCodec->waitForFence(fenceFd, "onOMXEmptyBufferDone");
474115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    fenceFd = -1;
474215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
474315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    // still save fence for completeness
474415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    info->setWriteFence(fenceFd, "onOMXEmptyBufferDone");
474515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
474696e92b58b7e9647b4c7c2f54b62a1b357ab06b66Wei Jia    // We're in "store-metadata-in-buffers" mode, the underlying
474796e92b58b7e9647b4c7c2f54b62a1b357ab06b66Wei Jia    // OMX component had access to data that's implicitly refcounted
474896e92b58b7e9647b4c7c2f54b62a1b357ab06b66Wei Jia    // by this "MediaBuffer" object. Now that the OMX component has
474996e92b58b7e9647b4c7c2f54b62a1b357ab06b66Wei Jia    // told us that it's done with the input buffer, we can decrement
475096e92b58b7e9647b4c7c2f54b62a1b357ab06b66Wei Jia    // the mediaBuffer's reference count.
475196e92b58b7e9647b4c7c2f54b62a1b357ab06b66Wei Jia    info->mData->setMediaBufferBase(NULL);
4752d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
4753f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexInput);
4754f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4755f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
4756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
4757f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
4758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4759f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
4760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            postFillThisBuffer(info);
4761f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
4762f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4763777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        case FREE_BUFFERS:
4764f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
4765777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            ALOGE("SHOULD NOT REACH HERE: cannot free empty output buffers");
4766777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            return false;
4767f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4768f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4769f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
4770f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4771f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4772f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::postFillThisBuffer(BufferInfo *info) {
4773f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mPortEOS[kPortIndexInput]) {
4774f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
4775f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4776f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4777f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
4778f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4779f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> notify = mCodec->mNotify->dup();
4780d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar    notify->setInt32("what", CodecBase::kWhatFillThisBuffer);
4781609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    notify->setInt32("buffer-id", info->mBufferID);
4782f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4783f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mData->meta()->clear();
47842d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber    notify->setBuffer("buffer", info->mData);
4785f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
47861d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    sp<AMessage> reply = new AMessage(kWhatInputBufferFilled, mCodec);
4787609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    reply->setInt32("buffer-id", info->mBufferID);
4788f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4789f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setMessage("reply", reply);
4790f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4791f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->post();
4792f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4793f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_UPSTREAM;
4794f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4795f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4796f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::onInputBufferFilled(const sp<AMessage> &msg) {
4797f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::buffer_id bufferID;
4798609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    CHECK(msg->findInt32("buffer-id", (int32_t*)&bufferID));
47992d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber    sp<ABuffer> buffer;
4800f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t err = OK;
48015778822d86b0337407514b9372562b86edfa91cdAndreas Huber    bool eos = false;
4802a73c954d947748a3b6f630cf2c160fe55ec596e3Lajos Molnar    PortMode mode = getPortMode(kPortIndexInput);
48035778822d86b0337407514b9372562b86edfa91cdAndreas Huber
48042d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber    if (!msg->findBuffer("buffer", &buffer)) {
4805a73c954d947748a3b6f630cf2c160fe55ec596e3Lajos Molnar        /* these are unfilled buffers returned by client */
4806f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("err", &err));
4807f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
48087fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar        if (err == OK) {
48097fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar            /* buffers with no errors are returned on MediaCodec.flush */
48107fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar            mode = KEEP_BUFFERS;
48117fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar        } else {
48127fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar            ALOGV("[%s] saw error %d instead of an input buffer",
48137fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar                 mCodec->mComponentName.c_str(), err);
48147fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar            eos = true;
48157fa015217e618265ff5a844cf6961ecc316c81a4Lajos Molnar        }
48163831a066bcf1019864a94d2bc7b4c9241efc5c22Andreas Huber
48172d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber        buffer.clear();
4818f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4819f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
48205778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t tmp;
48215778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (buffer != NULL && buffer->meta()->findInt32("eos", &tmp) && tmp) {
48225778822d86b0337407514b9372562b86edfa91cdAndreas Huber        eos = true;
48235778822d86b0337407514b9372562b86edfa91cdAndreas Huber        err = ERROR_END_OF_STREAM;
48245778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
48255778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4826f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = mCodec->findBufferByID(kPortIndexInput, bufferID);
48270806340688c937e7b78c2d89db3809274130df4eLajos Molnar    BufferInfo::Status status = BufferInfo::getSafeStatus(info);
48280806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (status != BufferInfo::OWNED_BY_UPSTREAM) {
48290806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGE("Wrong ownership in IBF: %s(%d) buffer #%u", _asString(status), status, bufferID);
48300806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->dumpBuffers(kPortIndexInput);
48310806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
48320806340688c937e7b78c2d89db3809274130df4eLajos Molnar        return;
48330806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
4834f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4835f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
4836f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4837f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
4838f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
4839f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
48405778822d86b0337407514b9372562b86edfa91cdAndreas Huber            if (eos) {
4841dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                if (!mCodec->mPortEOS[kPortIndexInput]) {
4842dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                    mCodec->mPortEOS[kPortIndexInput] = true;
4843dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                    mCodec->mInputEOSResult = err;
4844dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                }
4845f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
4846f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
4847f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4848f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4849f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
4850f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
48515778822d86b0337407514b9372562b86edfa91cdAndreas Huber            if (buffer != NULL && !mCodec->mPortEOS[kPortIndexInput]) {
4852f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                int64_t timeUs;
4853f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(buffer->meta()->findInt64("timeUs", &timeUs));
4854f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4855f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                OMX_U32 flags = OMX_BUFFERFLAG_ENDOFFRAME;
4856f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4857f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                int32_t isCSD;
4858f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (buffer->meta()->findInt32("csd", &isCSD) && isCSD != 0) {
4859f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    flags |= OMX_BUFFERFLAG_CODECCONFIG;
4860f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
4861f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
48625778822d86b0337407514b9372562b86edfa91cdAndreas Huber                if (eos) {
48635778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    flags |= OMX_BUFFERFLAG_EOS;
48645778822d86b0337407514b9372562b86edfa91cdAndreas Huber                }
48655778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4866f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (buffer != info->mData) {
4867ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                    ALOGV("[%s] Needs to copy input data for buffer %u. (%p != %p)",
4868d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                         mCodec->mComponentName.c_str(),
4869d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                         bufferID,
4870d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                         buffer.get(), info->mData.get());
4871f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
48720806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    if (buffer->size() > info->mData->capacity()) {
48730806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        ALOGE("data size (%zu) is greated than buffer capacity (%zu)",
48740806340688c937e7b78c2d89db3809274130df4eLajos Molnar                                buffer->size(),           // this is the data received
48750806340688c937e7b78c2d89db3809274130df4eLajos Molnar                                info->mData->capacity()); // this is out buffer size
48760806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
48770806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        return;
48780806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    }
4879f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    memcpy(info->mData->data(), buffer->data(), buffer->size());
4880f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
4881f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4882078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                if (flags & OMX_BUFFERFLAG_CODECCONFIG) {
4883ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                    ALOGV("[%s] calling emptyBuffer %u w/ codec specific data",
4884078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                         mCodec->mComponentName.c_str(), bufferID);
48855778822d86b0337407514b9372562b86edfa91cdAndreas Huber                } else if (flags & OMX_BUFFERFLAG_EOS) {
4886ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                    ALOGV("[%s] calling emptyBuffer %u w/ EOS",
48875778822d86b0337407514b9372562b86edfa91cdAndreas Huber                         mCodec->mComponentName.c_str(), bufferID);
4888078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                } else {
4889d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#if TRACK_BUFFER_TIMING
4890ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                    ALOGI("[%s] calling emptyBuffer %u w/ time %lld us",
4891ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                         mCodec->mComponentName.c_str(), bufferID, (long long)timeUs);
4892d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#else
4893ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                    ALOGV("[%s] calling emptyBuffer %u w/ time %lld us",
4894ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                         mCodec->mComponentName.c_str(), bufferID, (long long)timeUs);
4895d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#endif
4896078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                }
4897349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
4898d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#if TRACK_BUFFER_TIMING
4899d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                ACodec::BufferStats stats;
4900d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                stats.mEmptyBufferTimeUs = ALooper::GetNowUs();
4901d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                stats.mFillBufferDoneTimeUs = -1ll;
4902d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                mCodec->mBufferStats.add(timeUs, stats);
4903d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#endif
4904d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
4905054219874873b41f1c815552987c10465c34ba2bLajos Molnar                if (mCodec->storingMetadataInDecodedBuffers()) {
4906054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                    // try to submit an output buffer for each input buffer
4907054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                    PortMode outputMode = getPortMode(kPortIndexOutput);
4908054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
4909054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    ALOGV("MetadataBuffersToSubmit=%u portMode=%s",
4910054219874873b41f1c815552987c10465c34ba2bLajos Molnar                            mCodec->mMetadataBuffersToSubmit,
4911054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                            (outputMode == FREE_BUFFERS ? "FREE" :
4912054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                             outputMode == KEEP_BUFFERS ? "KEEP" : "RESUBMIT"));
4913054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                    if (outputMode == RESUBMIT_BUFFERS) {
4914054219874873b41f1c815552987c10465c34ba2bLajos Molnar                        mCodec->submitOutputMetadataBuffer();
4915054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                    }
4916054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                }
491715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info->checkReadFence("onInputBufferFilled");
49180806340688c937e7b78c2d89db3809274130df4eLajos Molnar                status_t err2 = mCodec->mOMX->emptyBuffer(
49190806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->mNode,
49200806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    bufferID,
49210806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    0,
49220806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    buffer->size(),
49230806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    flags,
492415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    timeUs,
492515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    info->mFenceFd);
492615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info->mFenceFd = -1;
49270806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (err2 != OK) {
49280806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err2));
49290806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    return;
49300806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
4931f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
4932f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
49330806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (!eos && err == OK) {
49345778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    getMoreInputDataIfPossible();
49355778822d86b0337407514b9372562b86edfa91cdAndreas Huber                } else {
49360806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    ALOGV("[%s] Signalled EOS (%d) on the input port",
49370806340688c937e7b78c2d89db3809274130df4eLajos Molnar                         mCodec->mComponentName.c_str(), err);
49385778822d86b0337407514b9372562b86edfa91cdAndreas Huber
49395778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    mCodec->mPortEOS[kPortIndexInput] = true;
49405778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    mCodec->mInputEOSResult = err;
49415778822d86b0337407514b9372562b86edfa91cdAndreas Huber                }
4942f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else if (!mCodec->mPortEOS[kPortIndexInput]) {
49430806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (err != OK && err != ERROR_END_OF_STREAM) {
49440806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    ALOGV("[%s] Signalling EOS on the input port due to error %d",
4945dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                         mCodec->mComponentName.c_str(), err);
4946dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                } else {
49473856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                    ALOGV("[%s] Signalling EOS on the input port",
4948dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                         mCodec->mComponentName.c_str());
4949dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                }
4950f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4951ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                ALOGV("[%s] calling emptyBuffer %u signalling EOS",
4952349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                     mCodec->mComponentName.c_str(), bufferID);
4953349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
495415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info->checkReadFence("onInputBufferFilled");
49550806340688c937e7b78c2d89db3809274130df4eLajos Molnar                status_t err2 = mCodec->mOMX->emptyBuffer(
49560806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        mCodec->mNode,
49570806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        bufferID,
49580806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        0,
49590806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        0,
49600806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        OMX_BUFFERFLAG_EOS,
496115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                        0,
496215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                        info->mFenceFd);
496315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info->mFenceFd = -1;
49640806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (err2 != OK) {
49650806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err2));
49660806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    return;
49670806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
4968f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
4969f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4970f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mPortEOS[kPortIndexInput] = true;
4971dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                mCodec->mInputEOSResult = err;
4972f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
4973f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
4974f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
4975625b93f1971039a547b239f87a2dc8a8d5716589Robert Shih
4976777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        case FREE_BUFFERS:
4977777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            break;
4978777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim
4979625b93f1971039a547b239f87a2dc8a8d5716589Robert Shih        default:
4980777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            ALOGE("invalid port mode: %d", mode);
4981625b93f1971039a547b239f87a2dc8a8d5716589Robert Shih            break;
4982f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4983f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
4984f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4985f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::getMoreInputDataIfPossible() {
4986f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mPortEOS[kPortIndexInput]) {
4987f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
4988f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
4989f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4990f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *eligible = NULL;
4991f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4992f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexInput].size(); ++i) {
4993f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mCodec->mBuffers[kPortIndexInput].editItemAt(i);
4994f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4995f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#if 0
4996f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus == BufferInfo::OWNED_BY_UPSTREAM) {
4997f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // There's already a "read" pending.
4998f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return;
4999f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5000f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#endif
5001f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5002f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus == BufferInfo::OWNED_BY_US) {
5003f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            eligible = info;
5004f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5005f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5006f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5007f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (eligible == NULL) {
5008f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
5009f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5010f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5011f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    postFillThisBuffer(eligible);
5012f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5013f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5014f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXFillBufferDone(
5015f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id bufferID,
5016f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        size_t rangeOffset, size_t rangeLength,
5017f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 flags,
501815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        int64_t timeUs,
501915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        int fenceFd) {
5020609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    ALOGV("[%s] onOMXFillBufferDone %u time %" PRId64 " us, flags = 0x%08x",
50215778822d86b0337407514b9372562b86edfa91cdAndreas Huber         mCodec->mComponentName.c_str(), bufferID, timeUs, flags);
5022349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
5023f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ssize_t index;
50240806340688c937e7b78c2d89db3809274130df4eLajos Molnar    status_t err= OK;
5025d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
5026d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#if TRACK_BUFFER_TIMING
5027d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    index = mCodec->mBufferStats.indexOfKey(timeUs);
5028d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (index >= 0) {
5029d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        ACodec::BufferStats *stats = &mCodec->mBufferStats.editValueAt(index);
5030d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        stats->mFillBufferDoneTimeUs = ALooper::GetNowUs();
5031d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
5032d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        ALOGI("frame PTS %lld: %lld",
5033d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                timeUs,
5034d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                stats->mFillBufferDoneTimeUs - stats->mEmptyBufferTimeUs);
5035d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
5036d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mCodec->mBufferStats.removeItemsAt(index);
5037d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        stats = NULL;
5038d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
5039d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#endif
5040d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
5041f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info =
5042f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->findBufferByID(kPortIndexOutput, bufferID, &index);
50430806340688c937e7b78c2d89db3809274130df4eLajos Molnar    BufferInfo::Status status = BufferInfo::getSafeStatus(info);
50440806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (status != BufferInfo::OWNED_BY_COMPONENT) {
50450806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGE("Wrong ownership in FBD: %s(%d) buffer #%u", _asString(status), status, bufferID);
50460806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->dumpBuffers(kPortIndexOutput);
50470806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
504815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        if (fenceFd >= 0) {
504915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            ::close(fenceFd);
505015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        }
50510806340688c937e7b78c2d89db3809274130df4eLajos Molnar        return true;
50520806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
5053f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5054054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    info->mDequeuedAt = ++mCodec->mDequeueCounter;
5055f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
5056f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
505790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    if (info->mRenderInfo != NULL) {
505890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        // The fence for an emptied buffer must have signaled, but there still could be queued
505990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        // or out-of-order dequeued buffers in the render queue prior to this buffer. Drop these,
506090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        // as we will soon requeue this buffer to the surface. While in theory we could still keep
506190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        // track of buffers that are requeued to the surface, it is better to add support to the
506290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        // buffer-queue to notify us of released buffers and their fences (in the future).
506390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        mCodec->notifyOfRenderedFrames(true /* dropIncomplete */);
506490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    }
506590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
506615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    // byte buffers cannot take fences, so wait for any fence now
506715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    if (mCodec->mNativeWindow == NULL) {
506815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        (void)mCodec->waitForFence(fenceFd, "onOMXFillBufferDone");
506915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        fenceFd = -1;
507015ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    }
507115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar    info->setReadFence(fenceFd, "onOMXFillBufferDone");
507215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar
5073f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexOutput);
5074f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5075f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
5076f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
5077f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
5078f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5079f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
5080f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5081a0940a569f2bc24b00dc10ce0fa7658b1dc3a3a5Lajos Molnar            if (rangeLength == 0 && (!(flags & OMX_BUFFERFLAG_EOS)
5082a0940a569f2bc24b00dc10ce0fa7658b1dc3a3a5Lajos Molnar                    || mCodec->mPortEOS[kPortIndexOutput])) {
5083609b815a3131d22da38b2f452faa9f89daad4039Andy Hung                ALOGV("[%s] calling fillBuffer %u",
50845778822d86b0337407514b9372562b86edfa91cdAndreas Huber                     mCodec->mComponentName.c_str(), info->mBufferID);
5085349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
508615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                err = mCodec->mOMX->fillBuffer(mCodec->mNode, info->mBufferID, info->mFenceFd);
508715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                info->mFenceFd = -1;
50880806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (err != OK) {
50890806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
50900806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    return true;
50910806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
5092f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
50935778822d86b0337407514b9372562b86edfa91cdAndreas Huber                info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
50945778822d86b0337407514b9372562b86edfa91cdAndreas Huber                break;
50955778822d86b0337407514b9372562b86edfa91cdAndreas Huber            }
509631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
5097577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar            sp<AMessage> reply =
50981d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar                new AMessage(kWhatOutputBufferDrained, mCodec);
5099577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar
51004bdda35319d5f46efea2089b865c8a64816389cdMarco Nelissen            if (!mCodec->mSentFormat && rangeLength > 0) {
5101577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar                mCodec->sendFormatChange(reply);
51025778822d86b0337407514b9372562b86edfa91cdAndreas Huber            }
5103054219874873b41f1c815552987c10465c34ba2bLajos Molnar            if (mCodec->usingMetadataOnEncoderOutput()) {
5104054219874873b41f1c815552987c10465c34ba2bLajos Molnar                native_handle_t *handle = NULL;
5105054219874873b41f1c815552987c10465c34ba2bLajos Molnar                VideoGrallocMetadata &grallocMeta = *(VideoGrallocMetadata *)info->mData->data();
5106054219874873b41f1c815552987c10465c34ba2bLajos Molnar                VideoNativeMetadata &nativeMeta = *(VideoNativeMetadata *)info->mData->data();
5107054219874873b41f1c815552987c10465c34ba2bLajos Molnar                if (info->mData->size() >= sizeof(grallocMeta)
5108054219874873b41f1c815552987c10465c34ba2bLajos Molnar                        && grallocMeta.eType == kMetadataBufferTypeGrallocSource) {
51099847fcefb183e1cb09eb48e17a09577392b0e8f4Lajos Molnar                    handle = (native_handle_t *)grallocMeta.pHandle;
5110054219874873b41f1c815552987c10465c34ba2bLajos Molnar                } else if (info->mData->size() >= sizeof(nativeMeta)
5111054219874873b41f1c815552987c10465c34ba2bLajos Molnar                        && nativeMeta.eType == kMetadataBufferTypeANWBuffer) {
5112054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    handle = (native_handle_t *)nativeMeta.pBuffer->handle;
5113054219874873b41f1c815552987c10465c34ba2bLajos Molnar                }
5114308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                info->mData->meta()->setPointer("handle", handle);
5115308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                info->mData->meta()->setInt32("rangeOffset", rangeOffset);
5116308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                info->mData->meta()->setInt32("rangeLength", rangeLength);
5117308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang            } else {
5118308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang                info->mData->setRange(rangeOffset, rangeLength);
5119308bcaa44e578279e61be32b572fdb0b11b1e4c7Chong Zhang            }
5120496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber#if 0
512121ad778dcfcddb8f8fd9dc3fe4992fbef246c511Marco Nelissen            if (mCodec->mNativeWindow == NULL) {
5122496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber                if (IsIDR(info->mData)) {
5123496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber                    ALOGI("IDR frame");
5124496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber                }
51255778822d86b0337407514b9372562b86edfa91cdAndreas Huber            }
512621ad778dcfcddb8f8fd9dc3fe4992fbef246c511Marco Nelissen#endif
5127f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
51288b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen            if (mCodec->mSkipCutBuffer != NULL) {
51298b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen                mCodec->mSkipCutBuffer->submit(info->mData);
51308b71241ce7353731ab75322c46e090ee35014a33Marco Nelissen            }
51315778822d86b0337407514b9372562b86edfa91cdAndreas Huber            info->mData->meta()->setInt64("timeUs", timeUs);
5132f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
51335778822d86b0337407514b9372562b86edfa91cdAndreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
5134d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar            notify->setInt32("what", CodecBase::kWhatDrainThisBuffer);
5135609b815a3131d22da38b2f452faa9f89daad4039Andy Hung            notify->setInt32("buffer-id", info->mBufferID);
51362d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber            notify->setBuffer("buffer", info->mData);
51375778822d86b0337407514b9372562b86edfa91cdAndreas Huber            notify->setInt32("flags", flags);
5138f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5139609b815a3131d22da38b2f452faa9f89daad4039Andy Hung            reply->setInt32("buffer-id", info->mBufferID);
5140f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
51415778822d86b0337407514b9372562b86edfa91cdAndreas Huber            notify->setMessage("reply", reply);
5142f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
51435778822d86b0337407514b9372562b86edfa91cdAndreas Huber            notify->post();
51445778822d86b0337407514b9372562b86edfa91cdAndreas Huber
51455778822d86b0337407514b9372562b86edfa91cdAndreas Huber            info->mStatus = BufferInfo::OWNED_BY_DOWNSTREAM;
5146f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5147f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (flags & OMX_BUFFERFLAG_EOS) {
51485778822d86b0337407514b9372562b86edfa91cdAndreas Huber                ALOGV("[%s] saw output EOS", mCodec->mComponentName.c_str());
51495778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                sp<AMessage> notify = mCodec->mNotify->dup();
5151d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar                notify->setInt32("what", CodecBase::kWhatEOS);
5152dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                notify->setInt32("err", mCodec->mInputEOSResult);
5153f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->post();
5154f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5155f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mPortEOS[kPortIndexOutput] = true;
5156f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
5157f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
5158f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5159f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5160777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        case FREE_BUFFERS:
51610806340688c937e7b78c2d89db3809274130df4eLajos Molnar            err = mCodec->freeBuffer(kPortIndexOutput, index);
51620806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (err != OK) {
51630806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
51640806340688c937e7b78c2d89db3809274130df4eLajos Molnar                return true;
51650806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
5166f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
5167777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim
5168777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        default:
5169777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            ALOGE("Invalid port mode: %d", mode);
5170777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            return false;
5171f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5172f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5173f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
5174f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5175f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5176f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::onOutputBufferDrained(const sp<AMessage> &msg) {
5177f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::buffer_id bufferID;
5178609b815a3131d22da38b2f452faa9f89daad4039Andy Hung    CHECK(msg->findInt32("buffer-id", (int32_t*)&bufferID));
5179f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ssize_t index;
51800806340688c937e7b78c2d89db3809274130df4eLajos Molnar    BufferInfo *info = mCodec->findBufferByID(kPortIndexOutput, bufferID, &index);
51810806340688c937e7b78c2d89db3809274130df4eLajos Molnar    BufferInfo::Status status = BufferInfo::getSafeStatus(info);
51820806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (status != BufferInfo::OWNED_BY_DOWNSTREAM) {
51830806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGE("Wrong ownership in OBD: %s(%d) buffer #%u", _asString(status), status, bufferID);
51840806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->dumpBuffers(kPortIndexOutput);
51850806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
51860806340688c937e7b78c2d89db3809274130df4eLajos Molnar        return;
51870806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
5188f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5189577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar    android_native_rect_t crop;
5190777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (msg->findRect("crop", &crop.left, &crop.top, &crop.right, &crop.bottom)) {
5191777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        status_t err = native_window_set_crop(mCodec->mNativeWindow.get(), &crop);
5192777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        ALOGW_IF(err != NO_ERROR, "failed to set crop: %d", err);
5193577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar    }
5194577db6d42b8d83b3578c7e56b1391bdaca9cb705Lajos Molnar
5195f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t render;
5196f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mNativeWindow != NULL
51976aade6058521b0dbd35a9a4620f4d04f02f90444Marco Nelissen            && msg->findInt32("render", &render) && render != 0
519879ee2399b67c7a11042c5904dc1309712a76f8cbJianzheng Zhou            && info->mData != NULL && info->mData->size() != 0) {
51996fbc1afd36f514f8b8d36ce67f5e69776c690bd2Lajos Molnar        ATRACE_NAME("render");
5200f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // The client wants this buffer to be rendered.
5201f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
520290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        // save buffers sent to the surface so we can get render time when they return
520390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        int64_t mediaTimeUs = -1;
520490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        info->mData->meta()->findInt64("timeUs", &mediaTimeUs);
520590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        if (mediaTimeUs >= 0) {
520690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar            mCodec->mRenderTracker.onFrameQueued(
520790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar                    mediaTimeUs, info->mGraphicBuffer, new Fence(::dup(info->mFenceFd)));
520890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        }
520990fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
5210fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar        int64_t timestampNs = 0;
5211fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar        if (!msg->findInt64("timestampNs", &timestampNs)) {
5212c8edf5af010ac24a99b302a18e7b84e8b4b2b783Lajos Molnar            // use media timestamp if client did not request a specific render timestamp
5213fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar            if (info->mData->meta()->findInt64("timeUs", &timestampNs)) {
5214c8edf5af010ac24a99b302a18e7b84e8b4b2b783Lajos Molnar                ALOGV("using buffer PTS of %lld", (long long)timestampNs);
5215fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar                timestampNs *= 1000;
5216fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar            }
5217fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar        }
5218fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar
52195778822d86b0337407514b9372562b86edfa91cdAndreas Huber        status_t err;
5220fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar        err = native_window_set_buffers_timestamp(mCodec->mNativeWindow.get(), timestampNs);
52210806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGW_IF(err != NO_ERROR, "failed to set buffer timestamp: %d", err);
5222fc7fca77caa12993dd938d5ff43797d781291027Lajos Molnar
522315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        info->checkReadFence("onOutputBufferDrained before queueBuffer");
52240806340688c937e7b78c2d89db3809274130df4eLajos Molnar        err = mCodec->mNativeWindow->queueBuffer(
522515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    mCodec->mNativeWindow.get(), info->mGraphicBuffer.get(), info->mFenceFd);
522615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        info->mFenceFd = -1;
52270806340688c937e7b78c2d89db3809274130df4eLajos Molnar        if (err == OK) {
5228cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber            info->mStatus = BufferInfo::OWNED_BY_NATIVE_WINDOW;
5229cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber        } else {
5230251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung            mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
5231cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber            info->mStatus = BufferInfo::OWNED_BY_US;
523215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            // keeping read fence as write fence to avoid clobbering
523315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            info->mIsReadFence = false;
5234cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber        }
5235f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
52366fbc1afd36f514f8b8d36ce67f5e69776c690bd2Lajos Molnar        if (mCodec->mNativeWindow != NULL &&
52376fbc1afd36f514f8b8d36ce67f5e69776c690bd2Lajos Molnar            (info->mData == NULL || info->mData->size() != 0)) {
523815ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            // move read fence into write fence to avoid clobbering
523915ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar            info->mIsReadFence = false;
52406fbc1afd36f514f8b8d36ce67f5e69776c690bd2Lajos Molnar            ATRACE_NAME("frame-drop");
52416fbc1afd36f514f8b8d36ce67f5e69776c690bd2Lajos Molnar        }
5242f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info->mStatus = BufferInfo::OWNED_BY_US;
5243f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5244f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5245f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexOutput);
5246f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5247f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
5248f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
5249f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5250f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // XXX fishy, revisit!!! What about the FREE_BUFFERS case below?
5251f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5252f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
5253f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // We cannot resubmit the buffer we just rendered, dequeue
5254f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // the spare instead.
5255f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5256f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info = mCodec->dequeueBufferFromNativeWindow();
5257f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
5258f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
5259f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5260f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5261f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
5262f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5263f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (!mCodec->mPortEOS[kPortIndexOutput]) {
5264f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
5265f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    // We cannot resubmit the buffer we just rendered, dequeue
5266f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    // the spare instead.
5267f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5268f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info = mCodec->dequeueBufferFromNativeWindow();
5269f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
5270f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5271c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                if (info != NULL) {
5272609b815a3131d22da38b2f452faa9f89daad4039Andy Hung                    ALOGV("[%s] calling fillBuffer %u",
5273c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                         mCodec->mComponentName.c_str(), info->mBufferID);
527415ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    info->checkWriteFence("onOutputBufferDrained::RESUBMIT_BUFFERS");
527515ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    status_t err = mCodec->mOMX->fillBuffer(
527615ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                            mCodec->mNode, info->mBufferID, info->mFenceFd);
527715ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar                    info->mFenceFd = -1;
52780806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    if (err == OK) {
52790806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
52800806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    } else {
52810806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
52820806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    }
5283c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                }
5284f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
5285f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
5286f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5287f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5288777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        case FREE_BUFFERS:
5289f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
52900806340688c937e7b78c2d89db3809274130df4eLajos Molnar            status_t err = mCodec->freeBuffer(kPortIndexOutput, index);
52910806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (err != OK) {
52920806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
52930806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
5294f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
5295f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5296777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim
5297777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        default:
5298777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            ALOGE("Invalid port mode: %d", mode);
5299777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            return;
5300f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5301f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5302f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5303f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
5304f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5305f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::UninitializedState::UninitializedState(ACodec *codec)
5306f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
5307f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5308f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5309c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Hubervoid ACodec::UninitializedState::stateEntered() {
5310c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    ALOGV("Now uninitialized");
5311ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
5312ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    if (mDeathNotifier != NULL) {
5313f888020c6e2735624f2b2a30e72aca24e17b8b4dMarco Nelissen        IInterface::asBinder(mCodec->mOMX)->unlinkToDeath(mDeathNotifier);
5314ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        mDeathNotifier.clear();
5315ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    }
5316ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
5317ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    mCodec->mNativeWindow.clear();
5318ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    mCodec->mNode = 0;
5319ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    mCodec->mOMX.clear();
5320ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    mCodec->mQuirks = 0;
5321ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    mCodec->mFlags = 0;
5322054219874873b41f1c815552987c10465c34ba2bLajos Molnar    mCodec->mInputMetadataType = kMetadataBufferTypeInvalid;
5323054219874873b41f1c815552987c10465c34ba2bLajos Molnar    mCodec->mOutputMetadataType = kMetadataBufferTypeInvalid;
5324ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    mCodec->mComponentName.clear();
5325c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber}
5326c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5327f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::UninitializedState::onMessageReceived(const sp<AMessage> &msg) {
5328f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
5329f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5330f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
5331f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatSetup:
5332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5333f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onSetup(msg);
5334f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
5336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
5337f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5338f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
53395778822d86b0337407514b9372562b86edfa91cdAndreas Huber        case ACodec::kWhatAllocateComponent:
53405778822d86b0337407514b9372562b86edfa91cdAndreas Huber        {
53415778822d86b0337407514b9372562b86edfa91cdAndreas Huber            onAllocateComponent(msg);
53425778822d86b0337407514b9372562b86edfa91cdAndreas Huber            handled = true;
53435778822d86b0337407514b9372562b86edfa91cdAndreas Huber            break;
53445778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
53455778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5346f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatShutdown:
5347f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5348c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            int32_t keepComponentAllocated;
5349c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            CHECK(msg->findInt32(
5350c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber                        "keepComponentAllocated", &keepComponentAllocated));
535154b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar            ALOGW_IF(keepComponentAllocated,
535254b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar                     "cannot keep component allocated on shutdown in Uninitialized state");
5353c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5354f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
5355d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar            notify->setInt32("what", CodecBase::kWhatShutdownCompleted);
5356f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
5357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5358f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
5359c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber            break;
5360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatFlush:
5363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
5365d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar            notify->setInt32("what", CodecBase::kWhatFlushCompleted);
5366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
5367f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5368f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
5369c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber            break;
5370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
537230358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar        case ACodec::kWhatReleaseCodecInstance:
537330358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar        {
537430358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            // nothing to do, as we have already signaled shutdown
537530358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            handled = true;
537630358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar            break;
537730358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar        }
537830358faf33fb9b638257b017fadb4c5f7352d903Lajos Molnar
5379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
5380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
5381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
5384f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5385f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5386f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::UninitializedState::onSetup(
5387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
5388c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    if (onAllocateComponent(msg)
5389c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            && mCodec->mLoadedState->onConfigureComponent(msg)) {
5390c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        mCodec->mLoadedState->onStart();
5391c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    }
53925778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
53935778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5394c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huberbool ACodec::UninitializedState::onAllocateComponent(const sp<AMessage> &msg) {
53955778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ALOGV("onAllocateComponent");
53965778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5397ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    CHECK(mCodec->mNode == 0);
53985778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5399f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMXClient client;
540048a31bf3f1c1ed5953a4e64f71cdf528f3a38ee5Andy Hung    if (client.connect() != OK) {
540148a31bf3f1c1ed5953a4e64f71cdf528f3a38ee5Andy Hung        mCodec->signalError(OMX_ErrorUndefined, NO_INIT);
540248a31bf3f1c1ed5953a4e64f71cdf528f3a38ee5Andy Hung        return false;
540348a31bf3f1c1ed5953a4e64f71cdf528f3a38ee5Andy Hung    }
5404f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5405f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<IOMX> omx = client.interface();
5406f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
54071d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar    sp<AMessage> notify = new AMessage(kWhatOMXDied, mCodec);
5408ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
5409ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    mDeathNotifier = new DeathNotifier(notify);
5410f888020c6e2735624f2b2a30e72aca24e17b8b4dMarco Nelissen    if (IInterface::asBinder(omx)->linkToDeath(mDeathNotifier) != OK) {
5411ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        // This was a local binder, if it dies so do we, we won't care
5412ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        // about any notifications in the afterlife.
5413ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber        mDeathNotifier.clear();
5414ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber    }
5415ac0230da14a3d223c2144b165a3a163e8519d239Andreas Huber
5416e671207115fac3914134c61b336d5fa0242c68caAndreas Huber    Vector<OMXCodec::CodecNameAndQuirks> matchingCodecs;
54175778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5418f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    AString mime;
5419f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
54205778822d86b0337407514b9372562b86edfa91cdAndreas Huber    AString componentName;
5421d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    uint32_t quirks = 0;
54227791cf11186a22b3f84d98cfde67393bee748cb0Marco Nelissen    int32_t encoder = false;
54235778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (msg->findString("componentName", &componentName)) {
5424e671207115fac3914134c61b336d5fa0242c68caAndreas Huber        ssize_t index = matchingCodecs.add();
5425e671207115fac3914134c61b336d5fa0242c68caAndreas Huber        OMXCodec::CodecNameAndQuirks *entry = &matchingCodecs.editItemAt(index);
5426e671207115fac3914134c61b336d5fa0242c68caAndreas Huber        entry->mName = String8(componentName.c_str());
5427afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber
5428e671207115fac3914134c61b336d5fa0242c68caAndreas Huber        if (!OMXCodec::findCodecQuirks(
5429e671207115fac3914134c61b336d5fa0242c68caAndreas Huber                    componentName.c_str(), &entry->mQuirks)) {
5430e671207115fac3914134c61b336d5fa0242c68caAndreas Huber            entry->mQuirks = 0;
5431afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber        }
54325778822d86b0337407514b9372562b86edfa91cdAndreas Huber    } else {
54335778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(msg->findString("mime", &mime));
54345778822d86b0337407514b9372562b86edfa91cdAndreas Huber
54355778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!msg->findInt32("encoder", &encoder)) {
54365778822d86b0337407514b9372562b86edfa91cdAndreas Huber            encoder = false;
54375778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
54385778822d86b0337407514b9372562b86edfa91cdAndreas Huber
54395778822d86b0337407514b9372562b86edfa91cdAndreas Huber        OMXCodec::findMatchingCodecs(
54405778822d86b0337407514b9372562b86edfa91cdAndreas Huber                mime.c_str(),
54415778822d86b0337407514b9372562b86edfa91cdAndreas Huber                encoder, // createEncoder
54425778822d86b0337407514b9372562b86edfa91cdAndreas Huber                NULL,  // matchComponentName
54435778822d86b0337407514b9372562b86edfa91cdAndreas Huber                0,     // flags
5444e671207115fac3914134c61b336d5fa0242c68caAndreas Huber                &matchingCodecs);
54455778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
54461065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
54471065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    sp<CodecObserver> observer = new CodecObserver;
5448ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    IOMX::node_id node = 0;
54491065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
5450201d8d400eb037547f4f476a838475b13a446007Wei Jia    status_t err = NAME_NOT_FOUND;
54511065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    for (size_t matchIndex = 0; matchIndex < matchingCodecs.size();
54521065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            ++matchIndex) {
5453e671207115fac3914134c61b336d5fa0242c68caAndreas Huber        componentName = matchingCodecs.itemAt(matchIndex).mName.string();
5454e671207115fac3914134c61b336d5fa0242c68caAndreas Huber        quirks = matchingCodecs.itemAt(matchIndex).mQuirks;
5455f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5456da153975581fb3161a30452348a5b26ee72d9255Elliott Hughes        pid_t tid = gettid();
5457078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber        int prevPriority = androidGetThreadPriority(tid);
5458078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber        androidSetThreadPriority(tid, ANDROID_PRIORITY_FOREGROUND);
54599f41ee5a4dd73093d107bc80df5cebb9a273cc72Ronghua Wu        err = omx->allocateNode(componentName.c_str(), observer, &node);
5460078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber        androidSetThreadPriority(tid, prevPriority);
5461f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
54621065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        if (err == OK) {
54631065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            break;
54647a727021b86c2fb2c9f60af2c999154dce1766ebZhijun He        } else {
54657a727021b86c2fb2c9f60af2c999154dce1766ebZhijun He            ALOGW("Allocating component '%s' failed, try next one.", componentName.c_str());
54661065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        }
54671065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
5468ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar        node = 0;
54691065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    }
54701065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
5471ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    if (node == 0) {
54725778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!mime.empty()) {
54739f41ee5a4dd73093d107bc80df5cebb9a273cc72Ronghua Wu            ALOGE("Unable to instantiate a %scoder for type '%s' with err %#x.",
54749f41ee5a4dd73093d107bc80df5cebb9a273cc72Ronghua Wu                    encoder ? "en" : "de", mime.c_str(), err);
54755778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else {
54769f41ee5a4dd73093d107bc80df5cebb9a273cc72Ronghua Wu            ALOGE("Unable to instantiate codec '%s' with err %#x.", componentName.c_str(), err);
54775778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
5478c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber
547952e88b2986536e83a7a6da63461556b8734a85f3Ronghua Wu        mCodec->signalError((OMX_ERRORTYPE)err, makeNoSideEffectStatus(err));
5480c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        return false;
5481c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    }
5482f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
548326a48f304a8754d655e554178ffb6d7ba4c5aac3Lajos Molnar    notify = new AMessage(kWhatOMXMessageList, mCodec);
5484f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    observer->setNotificationMessage(notify);
5485f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5486f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mComponentName = componentName;
548790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    mCodec->mRenderTracker.setComponentName(componentName);
5488ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    mCodec->mFlags = 0;
5489ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
5490ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    if (componentName.endsWith(".secure")) {
5491ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        mCodec->mFlags |= kFlagIsSecure;
54921713460104b86f6be3a5d9993d9ace864d889b2dChong Zhang        mCodec->mFlags |= kFlagIsGrallocUsageProtected;
54930167414e261f88a96b5e4bf6cb592e6ca11e5a95Andreas Huber        mCodec->mFlags |= kFlagPushBlankBuffersToNativeWindowOnShutdown;
5494ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    }
5495ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
5496afc16d667afa23f5aa00154ccad62f8c45cf5419Andreas Huber    mCodec->mQuirks = quirks;
5497f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mOMX = omx;
5498f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mNode = node;
5499f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
55005778822d86b0337407514b9372562b86edfa91cdAndreas Huber    {
55015778822d86b0337407514b9372562b86edfa91cdAndreas Huber        sp<AMessage> notify = mCodec->mNotify->dup();
5502d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar        notify->setInt32("what", CodecBase::kWhatComponentAllocated);
55035778822d86b0337407514b9372562b86edfa91cdAndreas Huber        notify->setString("componentName", mCodec->mComponentName.c_str());
55045778822d86b0337407514b9372562b86edfa91cdAndreas Huber        notify->post();
55055778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
5506c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5507c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    mCodec->changeState(mCodec->mLoadedState);
5508c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5509c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    return true;
55105778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
55115778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5512c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber////////////////////////////////////////////////////////////////////////////////
5513c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5514c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas HuberACodec::LoadedState::LoadedState(ACodec *codec)
5515c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    : BaseState(codec) {
5516c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber}
5517c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5518c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Hubervoid ACodec::LoadedState::stateEntered() {
5519c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    ALOGV("[%s] Now Loaded", mCodec->mComponentName.c_str());
5520c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5521f6f38287b97ec69b169387add6458f859b770e65Andreas Huber    mCodec->mPortEOS[kPortIndexInput] =
5522f6f38287b97ec69b169387add6458f859b770e65Andreas Huber        mCodec->mPortEOS[kPortIndexOutput] = false;
5523f6f38287b97ec69b169387add6458f859b770e65Andreas Huber
5524f6f38287b97ec69b169387add6458f859b770e65Andreas Huber    mCodec->mInputEOSResult = OK;
5525f6f38287b97ec69b169387add6458f859b770e65Andreas Huber
5526054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    mCodec->mDequeueCounter = 0;
5527054219874873b41f1c815552987c10465c34ba2bLajos Molnar    mCodec->mMetadataBuffersToSubmit = 0;
5528a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber    mCodec->mRepeatFrameDelayUs = -1ll;
5529e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    mCodec->mInputFormat.clear();
5530e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar    mCodec->mOutputFormat.clear();
55314e865a3cfe4c955e0890321a6b488cf661808b63Lajos Molnar    mCodec->mBaseOutputFormat.clear();
5532054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
5533c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    if (mCodec->mShutdownInProgress) {
5534c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        bool keepComponentAllocated = mCodec->mKeepComponentAllocated;
5535c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5536c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        mCodec->mShutdownInProgress = false;
5537c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        mCodec->mKeepComponentAllocated = false;
5538c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5539c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        onShutdown(keepComponentAllocated);
5540c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    }
554154b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar    mCodec->mExplicitShutdown = false;
5542f393c33e65ae51a7cf168b4e15771b152f996b32Wei Jia
5543f393c33e65ae51a7cf168b4e15771b152f996b32Wei Jia    mCodec->processDeferredMessages();
5544c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber}
5545c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5546c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Hubervoid ACodec::LoadedState::onShutdown(bool keepComponentAllocated) {
5547c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    if (!keepComponentAllocated) {
55480806340688c937e7b78c2d89db3809274130df4eLajos Molnar        (void)mCodec->mOMX->freeNode(mCodec->mNode);
5549c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5550c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        mCodec->changeState(mCodec->mUninitializedState);
5551c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    }
5552c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
555354b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar    if (mCodec->mExplicitShutdown) {
555454b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar        sp<AMessage> notify = mCodec->mNotify->dup();
5555d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar        notify->setInt32("what", CodecBase::kWhatShutdownCompleted);
555654b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar        notify->post();
555754b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar        mCodec->mExplicitShutdown = false;
555854b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar    }
5559c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber}
5560c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5561c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huberbool ACodec::LoadedState::onMessageReceived(const sp<AMessage> &msg) {
5562c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    bool handled = false;
5563c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5564c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    switch (msg->what()) {
5565c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        case ACodec::kWhatConfigureComponent:
5566c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        {
5567c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            onConfigureComponent(msg);
5568c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            handled = true;
5569c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            break;
5570c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        }
5571c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
55727cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        case ACodec::kWhatCreateInputSurface:
55737cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        {
55747cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            onCreateInputSurface(msg);
55757cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            handled = true;
55767cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            break;
55777cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        }
55787cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden
55798f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang        case ACodec::kWhatSetInputSurface:
5580d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        {
55818f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang            onSetInputSurface(msg);
5582d291c222357303b9611cab89d0c3b047584ef377Chong Zhang            handled = true;
5583d291c222357303b9611cab89d0c3b047584ef377Chong Zhang            break;
5584d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        }
5585d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5586c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        case ACodec::kWhatStart:
5587c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        {
5588c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            onStart();
5589c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            handled = true;
5590c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            break;
5591c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        }
5592c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5593c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        case ACodec::kWhatShutdown:
5594c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        {
5595c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            int32_t keepComponentAllocated;
5596c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            CHECK(msg->findInt32(
5597c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber                        "keepComponentAllocated", &keepComponentAllocated));
5598c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
559954b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar            mCodec->mExplicitShutdown = true;
5600c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            onShutdown(keepComponentAllocated);
5601c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5602c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            handled = true;
5603c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            break;
5604c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        }
5605c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5606c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        case ACodec::kWhatFlush:
5607c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        {
5608c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
5609d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar            notify->setInt32("what", CodecBase::kWhatFlushCompleted);
5610c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            notify->post();
5611c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5612c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            handled = true;
5613c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            break;
5614c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        }
5615c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5616c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        default:
5617c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            return BaseState::onMessageReceived(msg);
5618c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    }
5619c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5620c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    return handled;
5621c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber}
5622c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5623c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huberbool ACodec::LoadedState::onConfigureComponent(
56245778822d86b0337407514b9372562b86edfa91cdAndreas Huber        const sp<AMessage> &msg) {
56255778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ALOGV("onConfigureComponent");
56265778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5627ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    CHECK(mCodec->mNode != 0);
56285778822d86b0337407514b9372562b86edfa91cdAndreas Huber
56290806340688c937e7b78c2d89db3809274130df4eLajos Molnar    status_t err = OK;
56305778822d86b0337407514b9372562b86edfa91cdAndreas Huber    AString mime;
56310806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (!msg->findString("mime", &mime)) {
56320806340688c937e7b78c2d89db3809274130df4eLajos Molnar        err = BAD_VALUE;
56330806340688c937e7b78c2d89db3809274130df4eLajos Molnar    } else {
56340806340688c937e7b78c2d89db3809274130df4eLajos Molnar        err = mCodec->configureCodec(mime.c_str(), msg);
56350806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
56365778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (err != OK) {
5637c5255ac5b1bd313dcb50159566163b24dce7483fAndreas Huber        ALOGE("[%s] configureCodec returning error %d",
5638c5255ac5b1bd313dcb50159566163b24dce7483fAndreas Huber              mCodec->mComponentName.c_str(), err);
5639c5255ac5b1bd313dcb50159566163b24dce7483fAndreas Huber
5640251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
5641c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber        return false;
56425778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
5643f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
56445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    {
56455778822d86b0337407514b9372562b86edfa91cdAndreas Huber        sp<AMessage> notify = mCodec->mNotify->dup();
5646d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar        notify->setInt32("what", CodecBase::kWhatComponentConfigured);
5647e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        notify->setMessage("input-format", mCodec->mInputFormat);
5648e0381245dff04aa823a59aa8b85869eddab0f39fLajos Molnar        notify->setMessage("output-format", mCodec->mOutputFormat);
56495778822d86b0337407514b9372562b86edfa91cdAndreas Huber        notify->post();
56505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
5651c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
5652c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber    return true;
56535778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
56545778822d86b0337407514b9372562b86edfa91cdAndreas Huber
5655d291c222357303b9611cab89d0c3b047584ef377Chong Zhangstatus_t ACodec::LoadedState::setupInputSurface() {
5656d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    status_t err = OK;
5657a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber
5658d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (mCodec->mRepeatFrameDelayUs > 0ll) {
5659a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber        err = mCodec->mOMX->setInternalOption(
5660a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                mCodec->mNode,
5661a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                kPortIndexInput,
5662a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                IOMX::INTERNAL_OPTION_REPEAT_PREVIOUS_FRAME_DELAY,
5663a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                &mCodec->mRepeatFrameDelayUs,
5664a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                sizeof(mCodec->mRepeatFrameDelayUs));
5665a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber
5666a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber        if (err != OK) {
5667a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber            ALOGE("[%s] Unable to configure option to repeat previous "
5668a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                  "frames (err %d)",
5669a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                  mCodec->mComponentName.c_str(),
5670a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                  err);
5671d291c222357303b9611cab89d0c3b047584ef377Chong Zhang            return err;
5672a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber        }
5673a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber    }
5674a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber
5675d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (mCodec->mMaxPtsGapUs > 0ll) {
567694ee4b708acfa941581160b267afb79192b1d816Chong Zhang        err = mCodec->mOMX->setInternalOption(
567794ee4b708acfa941581160b267afb79192b1d816Chong Zhang                mCodec->mNode,
567894ee4b708acfa941581160b267afb79192b1d816Chong Zhang                kPortIndexInput,
567994ee4b708acfa941581160b267afb79192b1d816Chong Zhang                IOMX::INTERNAL_OPTION_MAX_TIMESTAMP_GAP,
568094ee4b708acfa941581160b267afb79192b1d816Chong Zhang                &mCodec->mMaxPtsGapUs,
568194ee4b708acfa941581160b267afb79192b1d816Chong Zhang                sizeof(mCodec->mMaxPtsGapUs));
568294ee4b708acfa941581160b267afb79192b1d816Chong Zhang
568394ee4b708acfa941581160b267afb79192b1d816Chong Zhang        if (err != OK) {
568494ee4b708acfa941581160b267afb79192b1d816Chong Zhang            ALOGE("[%s] Unable to configure max timestamp gap (err %d)",
568572cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                    mCodec->mComponentName.c_str(),
568672cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                    err);
5687d291c222357303b9611cab89d0c3b047584ef377Chong Zhang            return err;
56882c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        }
56892c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang    }
56902c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang
5691d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (mCodec->mMaxFps > 0) {
569237b2b389139ed638831e49708c947863eef631efRonghua Wu        err = mCodec->mOMX->setInternalOption(
569337b2b389139ed638831e49708c947863eef631efRonghua Wu                mCodec->mNode,
569437b2b389139ed638831e49708c947863eef631efRonghua Wu                kPortIndexInput,
569537b2b389139ed638831e49708c947863eef631efRonghua Wu                IOMX::INTERNAL_OPTION_MAX_FPS,
569637b2b389139ed638831e49708c947863eef631efRonghua Wu                &mCodec->mMaxFps,
569737b2b389139ed638831e49708c947863eef631efRonghua Wu                sizeof(mCodec->mMaxFps));
569837b2b389139ed638831e49708c947863eef631efRonghua Wu
569937b2b389139ed638831e49708c947863eef631efRonghua Wu        if (err != OK) {
570037b2b389139ed638831e49708c947863eef631efRonghua Wu            ALOGE("[%s] Unable to configure max fps (err %d)",
570137b2b389139ed638831e49708c947863eef631efRonghua Wu                    mCodec->mComponentName.c_str(),
570237b2b389139ed638831e49708c947863eef631efRonghua Wu                    err);
5703d291c222357303b9611cab89d0c3b047584ef377Chong Zhang            return err;
570437b2b389139ed638831e49708c947863eef631efRonghua Wu        }
570537b2b389139ed638831e49708c947863eef631efRonghua Wu    }
570637b2b389139ed638831e49708c947863eef631efRonghua Wu
5707d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (mCodec->mTimePerCaptureUs > 0ll
57082c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang            && mCodec->mTimePerFrameUs > 0ll) {
57092c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        int64_t timeLapse[2];
57102c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        timeLapse[0] = mCodec->mTimePerFrameUs;
57112c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        timeLapse[1] = mCodec->mTimePerCaptureUs;
57122c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        err = mCodec->mOMX->setInternalOption(
57132c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang                mCodec->mNode,
57142c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang                kPortIndexInput,
57152c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang                IOMX::INTERNAL_OPTION_TIME_LAPSE,
57162c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang                &timeLapse[0],
57172c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang                sizeof(timeLapse));
57182c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang
57192c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        if (err != OK) {
57202c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang            ALOGE("[%s] Unable to configure time lapse (err %d)",
57212c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang                    mCodec->mComponentName.c_str(),
57222c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang                    err);
5723d291c222357303b9611cab89d0c3b047584ef377Chong Zhang            return err;
57242c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang        }
57252c9c8cba8562cc3a27532e4cd348912cc78d8d98Chong Zhang    }
572672cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang
5727d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (mCodec->mCreateInputBuffersSuspended) {
572872cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        bool suspend = true;
572972cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        err = mCodec->mOMX->setInternalOption(
573072cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                mCodec->mNode,
573172cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                kPortIndexInput,
573272cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                IOMX::INTERNAL_OPTION_SUSPEND,
573372cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                &suspend,
573472cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                sizeof(suspend));
573572cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang
573672cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        if (err != OK) {
573772cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang            ALOGE("[%s] Unable to configure option to suspend (err %d)",
573894ee4b708acfa941581160b267afb79192b1d816Chong Zhang                  mCodec->mComponentName.c_str(),
573994ee4b708acfa941581160b267afb79192b1d816Chong Zhang                  err);
5740d291c222357303b9611cab89d0c3b047584ef377Chong Zhang            return err;
574194ee4b708acfa941581160b267afb79192b1d816Chong Zhang        }
574294ee4b708acfa941581160b267afb79192b1d816Chong Zhang    }
574394ee4b708acfa941581160b267afb79192b1d816Chong Zhang
5744d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    return OK;
5745d291c222357303b9611cab89d0c3b047584ef377Chong Zhang}
5746d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5747d291c222357303b9611cab89d0c3b047584ef377Chong Zhangvoid ACodec::LoadedState::onCreateInputSurface(
5748d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        const sp<AMessage> & /* msg */) {
5749d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    ALOGV("onCreateInputSurface");
5750d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5751d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    sp<AMessage> notify = mCodec->mNotify->dup();
5752d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    notify->setInt32("what", CodecBase::kWhatInputSurfaceCreated);
5753d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5754d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    sp<IGraphicBufferProducer> bufferProducer;
5755d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    status_t err = mCodec->mOMX->createInputSurface(
5756054219874873b41f1c815552987c10465c34ba2bLajos Molnar            mCodec->mNode, kPortIndexInput, &bufferProducer, &mCodec->mInputMetadataType);
5757d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5758d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (err == OK) {
5759d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        err = setupInputSurface();
5760d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    }
5761d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
57627cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden    if (err == OK) {
57637cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        notify->setObject("input-surface",
57647cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden                new BufferProducerWrapper(bufferProducer));
57657cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden    } else {
57667cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        // Can't use mCodec->signalError() here -- MediaCodec won't forward
57677cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        // the error through because it's in the "configured" state.  We
57687cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        // send a kWhatInputSurfaceCreated with an error value instead.
57697cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        ALOGE("[%s] onCreateInputSurface returning error %d",
57707cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden                mCodec->mComponentName.c_str(), err);
57717cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        notify->setInt32("err", err);
57727cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden    }
57737cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden    notify->post();
57747cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden}
57757cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden
57768f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhangvoid ACodec::LoadedState::onSetInputSurface(
5777d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        const sp<AMessage> &msg) {
57788f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang    ALOGV("onSetInputSurface");
5779d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5780d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    sp<AMessage> notify = mCodec->mNotify->dup();
5781d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    notify->setInt32("what", CodecBase::kWhatInputSurfaceAccepted);
5782d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5783d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    sp<RefBase> obj;
5784d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    CHECK(msg->findObject("input-surface", &obj));
5785d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    sp<PersistentSurface> surface = static_cast<PersistentSurface *>(obj.get());
5786d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
57878f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang    status_t err = mCodec->mOMX->setInputSurface(
5788054219874873b41f1c815552987c10465c34ba2bLajos Molnar            mCodec->mNode, kPortIndexInput, surface->getBufferConsumer(),
5789054219874873b41f1c815552987c10465c34ba2bLajos Molnar            &mCodec->mInputMetadataType);
5790d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5791d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (err == OK) {
5792d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        err = setupInputSurface();
5793d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    }
5794d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5795d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    if (err != OK) {
5796d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        // Can't use mCodec->signalError() here -- MediaCodec won't forward
5797d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        // the error through because it's in the "configured" state.  We
5798d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        // send a kWhatInputSurfaceAccepted with an error value instead.
57998f469e18c307cb9dc0d16ed9225972aa8be4516fChong Zhang        ALOGE("[%s] onSetInputSurface returning error %d",
5800d291c222357303b9611cab89d0c3b047584ef377Chong Zhang                mCodec->mComponentName.c_str(), err);
5801d291c222357303b9611cab89d0c3b047584ef377Chong Zhang        notify->setInt32("err", err);
5802d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    }
5803d291c222357303b9611cab89d0c3b047584ef377Chong Zhang    notify->post();
5804d291c222357303b9611cab89d0c3b047584ef377Chong Zhang}
5805d291c222357303b9611cab89d0c3b047584ef377Chong Zhang
5806c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Hubervoid ACodec::LoadedState::onStart() {
58075778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ALOGV("onStart");
58085778822d86b0337407514b9372562b86edfa91cdAndreas Huber
58090806340688c937e7b78c2d89db3809274130df4eLajos Molnar    status_t err = mCodec->mOMX->sendCommand(mCodec->mNode, OMX_CommandStateSet, OMX_StateIdle);
58100806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (err != OK) {
58110806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
58120806340688c937e7b78c2d89db3809274130df4eLajos Molnar    } else {
58130806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->changeState(mCodec->mLoadedToIdleState);
58140806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
5815f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5816f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5817f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
5818f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5819f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::LoadedToIdleState::LoadedToIdleState(ACodec *codec)
5820f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
5821f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5822f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5823f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::LoadedToIdleState::stateEntered() {
58243856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%s] Now Loaded->Idle", mCodec->mComponentName.c_str());
5825f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5826cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    status_t err;
5827cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    if ((err = allocateBuffers()) != OK) {
582829357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Failed to allocate buffers after transitioning to IDLE state "
5829cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber             "(error 0x%08x)",
5830cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber             err);
5831cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber
5832251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung        mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
583391bfadb4aa75444c60e9949232de3c184cfe060aAndreas Huber
583491bfadb4aa75444c60e9949232de3c184cfe060aAndreas Huber        mCodec->changeState(mCodec->mLoadedState);
5835cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    }
5836f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5837f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5838f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::LoadedToIdleState::allocateBuffers() {
5839f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mCodec->allocateBuffersOnPort(kPortIndexInput);
5840f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5841f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
5842f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
5843f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5844f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5845f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return mCodec->allocateBuffersOnPort(kPortIndexOutput);
5846f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5847f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5848f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::LoadedToIdleState::onMessageReceived(const sp<AMessage> &msg) {
5849f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
585072cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        case kWhatSetParameters:
5851f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
5852f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5853f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
5854f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
5855f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5856f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
58576507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden        case kWhatSignalEndOfInputStream:
58586507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden        {
58596507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden            mCodec->onSignalEndOfInputStream();
58606507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden            return true;
58616507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden        }
58626507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden
58636463e76d41430f9b03a79b221de84255f2475658Marco Nelissen        case kWhatResume:
58646463e76d41430f9b03a79b221de84255f2475658Marco Nelissen        {
58656463e76d41430f9b03a79b221de84255f2475658Marco Nelissen            // We'll be active soon enough.
58666463e76d41430f9b03a79b221de84255f2475658Marco Nelissen            return true;
58676463e76d41430f9b03a79b221de84255f2475658Marco Nelissen        }
58686463e76d41430f9b03a79b221de84255f2475658Marco Nelissen
58696463e76d41430f9b03a79b221de84255f2475658Marco Nelissen        case kWhatFlush:
58706463e76d41430f9b03a79b221de84255f2475658Marco Nelissen        {
58716463e76d41430f9b03a79b221de84255f2475658Marco Nelissen            // We haven't even started yet, so we're flushed alright...
58726463e76d41430f9b03a79b221de84255f2475658Marco Nelissen            sp<AMessage> notify = mCodec->mNotify->dup();
5873d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar            notify->setInt32("what", CodecBase::kWhatFlushCompleted);
58746463e76d41430f9b03a79b221de84255f2475658Marco Nelissen            notify->post();
58756463e76d41430f9b03a79b221de84255f2475658Marco Nelissen            return true;
58766463e76d41430f9b03a79b221de84255f2475658Marco Nelissen        }
58776463e76d41430f9b03a79b221de84255f2475658Marco Nelissen
5878f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
5879f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
5880f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5881f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5882f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5883f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::LoadedToIdleState::onOMXEvent(
5884f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
5885f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
5886f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
5887f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
58880806340688c937e7b78c2d89db3809274130df4eLajos Molnar            status_t err = OK;
58890806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (data1 != (OMX_U32)OMX_CommandStateSet
58900806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    || data2 != (OMX_U32)OMX_StateIdle) {
58910806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGE("Unexpected command completion in LoadedToIdleState: %s(%u) %s(%u)",
58920806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_COMMANDTYPE)data1), data1,
58930806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_STATETYPE)data2), data2);
58940806340688c937e7b78c2d89db3809274130df4eLajos Molnar                err = FAILED_TRANSACTION;
58950806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
5896f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
58970806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (err == OK) {
58980806340688c937e7b78c2d89db3809274130df4eLajos Molnar                err = mCodec->mOMX->sendCommand(
58990806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->mNode, OMX_CommandStateSet, OMX_StateExecuting);
59000806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
5901f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
59020806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (err != OK) {
59030806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
59040806340688c937e7b78c2d89db3809274130df4eLajos Molnar            } else {
59050806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->changeState(mCodec->mIdleToExecutingState);
59060806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
5907f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5908f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
5909f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5910f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5911f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
5912f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
5913f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5914f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5915f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5916f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
5917f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5918f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::IdleToExecutingState::IdleToExecutingState(ACodec *codec)
5919f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
5920f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5921f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5922f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::IdleToExecutingState::stateEntered() {
59233856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%s] Now Idle->Executing", mCodec->mComponentName.c_str());
5924f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5925f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5926f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToExecutingState::onMessageReceived(const sp<AMessage> &msg) {
5927f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
592872cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        case kWhatSetParameters:
5929f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
5930f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
5931f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
5932f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
5933f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5934f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5935d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber        case kWhatResume:
5936d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber        {
5937d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber            // We'll be active soon enough.
5938d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber            return true;
5939d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber        }
5940d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber
5941d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber        case kWhatFlush:
5942d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber        {
5943d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber            // We haven't even started yet, so we're flushed alright...
5944d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
5945d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar            notify->setInt32("what", CodecBase::kWhatFlushCompleted);
5946d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber            notify->post();
5947d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber
5948d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber            return true;
5949d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber        }
5950d477b8d071826c0768620f7ac302f31d8b12b1caAndreas Huber
59516507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden        case kWhatSignalEndOfInputStream:
59526507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden        {
59536507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden            mCodec->onSignalEndOfInputStream();
59546507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden            return true;
59556507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden        }
59566507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden
5957f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
5958f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
5959f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5960f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5961f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5962f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToExecutingState::onOMXEvent(
5963f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
5964f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
5965f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
5966f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
59670806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (data1 != (OMX_U32)OMX_CommandStateSet
59680806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    || data2 != (OMX_U32)OMX_StateExecuting) {
59690806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGE("Unexpected command completion in IdleToExecutingState: %s(%u) %s(%u)",
59700806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_COMMANDTYPE)data1), data1,
59710806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_STATETYPE)data2), data2);
59720806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
59730806340688c937e7b78c2d89db3809274130df4eLajos Molnar                return true;
59740806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
5975f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5976f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mExecutingState->resume();
5977f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mExecutingState);
5978f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5979f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
5980f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
5981f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5982f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
5983f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
5984f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
5985f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5986f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5987f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
5988f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5989f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ExecutingState::ExecutingState(ACodec *codec)
5990349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    : BaseState(codec),
5991349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber      mActive(false) {
5992f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5993f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5994f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::ExecutingState::getPortMode(
5995ccb067b1d8424ba610cbd3de83368bd55b532b5bAndreas Huber        OMX_U32 /* portIndex */) {
5996f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return RESUBMIT_BUFFERS;
5997f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
5998f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
5999054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnarvoid ACodec::ExecutingState::submitOutputMetaBuffers() {
6000054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    // submit as many buffers as there are input buffers with the codec
6001054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    // in case we are in port reconfiguring
6002054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexInput].size(); ++i) {
6003054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        BufferInfo *info = &mCodec->mBuffers[kPortIndexInput].editItemAt(i);
6004054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
6005054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        if (info->mStatus == BufferInfo::OWNED_BY_COMPONENT) {
6006054219874873b41f1c815552987c10465c34ba2bLajos Molnar            if (mCodec->submitOutputMetadataBuffer() != OK)
6007054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar                break;
6008054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        }
6009054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    }
60104dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar
60114dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar    // *** NOTE: THE FOLLOWING WORKAROUND WILL BE REMOVED ***
6012054219874873b41f1c815552987c10465c34ba2bLajos Molnar    mCodec->signalSubmitOutputMetadataBufferIfEOS_workaround();
6013054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar}
6014054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
6015054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnarvoid ACodec::ExecutingState::submitRegularOutputBuffers() {
60160806340688c937e7b78c2d89db3809274130df4eLajos Molnar    bool failed = false;
6017f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexOutput].size(); ++i) {
6018f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mCodec->mBuffers[kPortIndexOutput].editItemAt(i);
6019f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6020f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (mCodec->mNativeWindow != NULL) {
60210806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (info->mStatus != BufferInfo::OWNED_BY_US
60220806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    && info->mStatus != BufferInfo::OWNED_BY_NATIVE_WINDOW) {
60230806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGE("buffers should be owned by us or the surface");
60240806340688c937e7b78c2d89db3809274130df4eLajos Molnar                failed = true;
60250806340688c937e7b78c2d89db3809274130df4eLajos Molnar                break;
60260806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
6027f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6028f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
6029f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                continue;
6030f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
6031f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        } else {
60320806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (info->mStatus != BufferInfo::OWNED_BY_US) {
60330806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGE("buffers should be owned by us");
60340806340688c937e7b78c2d89db3809274130df4eLajos Molnar                failed = true;
60350806340688c937e7b78c2d89db3809274130df4eLajos Molnar                break;
60360806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
6037f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6038f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
60390806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGV("[%s] calling fillBuffer %u", mCodec->mComponentName.c_str(), info->mBufferID);
6040349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
604115ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        info->checkWriteFence("submitRegularOutputBuffers");
604215ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        status_t err = mCodec->mOMX->fillBuffer(mCodec->mNode, info->mBufferID, info->mFenceFd);
604315ab4996019387f27a48b81cb4774c21502bc0e5Lajos Molnar        info->mFenceFd = -1;
60440806340688c937e7b78c2d89db3809274130df4eLajos Molnar        if (err != OK) {
60450806340688c937e7b78c2d89db3809274130df4eLajos Molnar            failed = true;
60460806340688c937e7b78c2d89db3809274130df4eLajos Molnar            break;
60470806340688c937e7b78c2d89db3809274130df4eLajos Molnar        }
6048f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6049f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
6050f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
60510806340688c937e7b78c2d89db3809274130df4eLajos Molnar
60520806340688c937e7b78c2d89db3809274130df4eLajos Molnar    if (failed) {
60530806340688c937e7b78c2d89db3809274130df4eLajos Molnar        mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
60540806340688c937e7b78c2d89db3809274130df4eLajos Molnar    }
6055f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6056f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6057054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnarvoid ACodec::ExecutingState::submitOutputBuffers() {
6058c38fcfba95f711e5738e4c72bd5499317a2f30d9Lajos Molnar    submitRegularOutputBuffers();
6059054219874873b41f1c815552987c10465c34ba2bLajos Molnar    if (mCodec->storingMetadataInDecodedBuffers()) {
6060054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar        submitOutputMetaBuffers();
6061054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar    }
6062054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar}
6063054e7347cc60ad4b9dd2e8f456406f122f9f5879Lajos Molnar
6064f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::resume() {
6065349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    if (mActive) {
60660806340688c937e7b78c2d89db3809274130df4eLajos Molnar        ALOGV("[%s] We're already active, no need to resume.", mCodec->mComponentName.c_str());
6067349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        return;
6068349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    }
6069349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6070f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    submitOutputBuffers();
6071f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
60723cb576166020bfdc1522b33919fe5ff8d211d0a3Lajos Molnar    // Post all available input buffers
6073777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    if (mCodec->mBuffers[kPortIndexInput].size() == 0u) {
6074777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim        ALOGW("[%s] we don't have any input buffers to resume", mCodec->mComponentName.c_str());
6075777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim    }
6076777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim
60773cb576166020bfdc1522b33919fe5ff8d211d0a3Lajos Molnar    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexInput].size(); i++) {
60783cb576166020bfdc1522b33919fe5ff8d211d0a3Lajos Molnar        BufferInfo *info = &mCodec->mBuffers[kPortIndexInput].editItemAt(i);
60793cb576166020bfdc1522b33919fe5ff8d211d0a3Lajos Molnar        if (info->mStatus == BufferInfo::OWNED_BY_US) {
60803cb576166020bfdc1522b33919fe5ff8d211d0a3Lajos Molnar            postFillThisBuffer(info);
60813cb576166020bfdc1522b33919fe5ff8d211d0a3Lajos Molnar        }
60823cb576166020bfdc1522b33919fe5ff8d211d0a3Lajos Molnar    }
6083349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6084349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    mActive = true;
6085f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6086f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6087f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::stateEntered() {
60883856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%s] Now Executing", mCodec->mComponentName.c_str());
6089f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
609090fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    mCodec->mRenderTracker.clear(systemTime(CLOCK_MONOTONIC));
6091f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->processDeferredMessages();
6092f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6093f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6094f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingState::onMessageReceived(const sp<AMessage> &msg) {
6095f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
6096f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6097f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
6098f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
6099f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6100c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            int32_t keepComponentAllocated;
6101c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            CHECK(msg->findInt32(
6102c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber                        "keepComponentAllocated", &keepComponentAllocated));
6103c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
6104c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            mCodec->mShutdownInProgress = true;
610554b0bc74e051bd7df10cd657b481d88db6cdaa27Lajos Molnar            mCodec->mExplicitShutdown = true;
6106c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            mCodec->mKeepComponentAllocated = keepComponentAllocated;
6107c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber
6108349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mActive = false;
6109349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
61100806340688c937e7b78c2d89db3809274130df4eLajos Molnar            status_t err = mCodec->mOMX->sendCommand(
61110806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->mNode, OMX_CommandStateSet, OMX_StateIdle);
61120806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (err != OK) {
61130806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (keepComponentAllocated) {
61140806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
61150806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
61160806340688c937e7b78c2d89db3809274130df4eLajos Molnar                // TODO: do some recovery here.
61170806340688c937e7b78c2d89db3809274130df4eLajos Molnar            } else {
61180806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->changeState(mCodec->mExecutingToIdleState);
61190806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
6120f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6121f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
6122f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6123f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6124f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6125f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
6126f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
61277a3a2b2f9bb9421dcf83fbd47276e57917078aefJames Dong            ALOGV("[%s] ExecutingState flushing now "
6128ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                 "(codec owns %zu/%zu input, %zu/%zu output).",
6129d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber                    mCodec->mComponentName.c_str(),
6130d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber                    mCodec->countBuffersOwnedByComponent(kPortIndexInput),
6131d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber                    mCodec->mBuffers[kPortIndexInput].size(),
6132d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber                    mCodec->countBuffersOwnedByComponent(kPortIndexOutput),
6133d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber                    mCodec->mBuffers[kPortIndexOutput].size());
6134d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber
6135349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mActive = false;
6136349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
61370806340688c937e7b78c2d89db3809274130df4eLajos Molnar            status_t err = mCodec->mOMX->sendCommand(mCodec->mNode, OMX_CommandFlush, OMX_ALL);
61380806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (err != OK) {
61390806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
61400806340688c937e7b78c2d89db3809274130df4eLajos Molnar            } else {
61410806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->changeState(mCodec->mFlushingState);
61420806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
6143f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6144f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
6145f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6146f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6147f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6148f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatResume:
6149f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            resume();
6151f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6152f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
6153f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6154f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6155f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6156496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber        case kWhatRequestIDRFrame:
6157496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber        {
6158496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            status_t err = mCodec->requestIDRFrame();
6159496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            if (err != OK) {
6160496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber                ALOGW("Requesting an IDR frame failed.");
6161496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            }
6162496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
6163496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            handled = true;
6164496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber            break;
6165496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber        }
6166496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
6167a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        case kWhatSetParameters:
6168a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        {
6169a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            sp<AMessage> params;
6170a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            CHECK(msg->findMessage("params", &params));
6171a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
6172a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            status_t err = mCodec->setParameters(params);
6173a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
6174a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            sp<AMessage> reply;
6175a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            if (msg->findMessage("reply", &reply)) {
6176a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber                reply->setInt32("err", err);
6177a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber                reply->post();
6178a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            }
6179a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
6180a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            handled = true;
6181a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            break;
6182a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        }
6183a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
61847cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        case ACodec::kWhatSignalEndOfInputStream:
61857cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        {
61866507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden            mCodec->onSignalEndOfInputStream();
61877cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            handled = true;
61887cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden            break;
61897cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden        }
61907cd58537932ef6f481f68be0b9c597a89cebdfecAndy McFadden
61914dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar        // *** NOTE: THE FOLLOWING WORKAROUND WILL BE REMOVED ***
6192054219874873b41f1c815552987c10465c34ba2bLajos Molnar        case kWhatSubmitOutputMetadataBufferIfEOS:
61934dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar        {
61944dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar            if (mCodec->mPortEOS[kPortIndexInput] &&
61954dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar                    !mCodec->mPortEOS[kPortIndexOutput]) {
6196054219874873b41f1c815552987c10465c34ba2bLajos Molnar                status_t err = mCodec->submitOutputMetadataBuffer();
61974dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar                if (err == OK) {
6198054219874873b41f1c815552987c10465c34ba2bLajos Molnar                    mCodec->signalSubmitOutputMetadataBufferIfEOS_workaround();
61994dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar                }
62004dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar            }
62014dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar            return true;
62024dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar        }
62034dd0a8a3d66c2853faf2834565b3c5df4f68734dLajos Molnar
6204f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6205f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
6206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6207f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
6210f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6211f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6212a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huberstatus_t ACodec::setParameters(const sp<AMessage> &params) {
6213a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber    int32_t videoBitrate;
6214530fdbdc1b5491f3fbf172752834d1515701e142Lajos Molnar    if (params->findInt32("video-bitrate", &videoBitrate)) {
6215a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        OMX_VIDEO_CONFIG_BITRATETYPE configParams;
6216a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        InitOMXParams(&configParams);
6217a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        configParams.nPortIndex = kPortIndexOutput;
6218a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        configParams.nEncodeBitrate = videoBitrate;
6219a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
6220a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        status_t err = mOMX->setConfig(
6221a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber                mNode,
6222a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber                OMX_IndexConfigVideoBitrate,
6223a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber                &configParams,
6224a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber                sizeof(configParams));
6225a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
6226a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        if (err != OK) {
6227a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            ALOGE("setConfig(OMX_IndexConfigVideoBitrate, %d) failed w/ err %d",
6228a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber                   videoBitrate, err);
6229a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
6230a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber            return err;
6231a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber        }
6232a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber    }
6233a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
623472cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang    int64_t skipFramesBeforeUs;
623572cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang    if (params->findInt64("skip-frames-before", &skipFramesBeforeUs)) {
623672cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        status_t err =
623772cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang            mOMX->setInternalOption(
623872cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                     mNode,
623972cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                     kPortIndexInput,
624072cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                     IOMX::INTERNAL_OPTION_START_TIME,
624172cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                     &skipFramesBeforeUs,
624272cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang                     sizeof(skipFramesBeforeUs));
624372cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang
624472cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        if (err != OK) {
624572cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang            ALOGE("Failed to set parameter 'skip-frames-before' (err %d)", err);
624672cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang            return err;
624772cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang        }
624872cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang    }
624972cecca17d735db6532c45f0a7e10c47ee6f065aChong Zhang
6250e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    int32_t dropInputFrames;
6251e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    if (params->findInt32("drop-input-frames", &dropInputFrames)) {
6252e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        bool suspend = dropInputFrames != 0;
6253e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
6254b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber        status_t err =
6255b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber            mOMX->setInternalOption(
6256e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber                     mNode,
6257e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber                     kPortIndexInput,
6258e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber                     IOMX::INTERNAL_OPTION_SUSPEND,
6259e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber                     &suspend,
6260b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber                     sizeof(suspend));
6261b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber
6262b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber        if (err != OK) {
6263b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber            ALOGE("Failed to set parameter 'drop-input-frames' (err %d)", err);
6264b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber            return err;
6265b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber        }
6266b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber    }
6267b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber
6268b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber    int32_t dummy;
6269b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber    if (params->findInt32("request-sync", &dummy)) {
6270b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber        status_t err = requestIDRFrame();
6271b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber
6272b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber        if (err != OK) {
6273b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber            ALOGE("Requesting a sync frame failed w/ err %d", err);
6274b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber            return err;
6275b8689b31813f55cbc1bf8e9ca0d46a9ee113e38aAndreas Huber        }
6276e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    }
6277e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
62788db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu    float rate;
62798db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu    if (params->findFloat("operating-rate", &rate) && rate > 0) {
62808db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu        status_t err = setOperatingRate(rate, mIsVideo);
62818db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu        if (err != OK) {
62828db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu            ALOGE("Failed to set parameter 'operating-rate' (err %d)", err);
62838db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu            return err;
62848db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu        }
62858db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu    }
62868db8813d39e3c8b5fbd580dfc3062830744afd63Ronghua Wu
6287a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber    return OK;
6288a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber}
6289a2eb22c1de262aa3fa7c356537ac2fe165afdf3dAndreas Huber
62906507d14c6d10f93d390de62b9eed267f9b544985Andy McFaddenvoid ACodec::onSignalEndOfInputStream() {
62916507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden    sp<AMessage> notify = mNotify->dup();
6292d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar    notify->setInt32("what", CodecBase::kWhatSignaledInputEOS);
62936507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden
62946507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden    status_t err = mOMX->signalEndOfInputStream(mNode);
62956507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden    if (err != OK) {
62966507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden        notify->setInt32("err", err);
62976507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden    }
62986507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden    notify->post();
62996507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden}
63006507d14c6d10f93d390de62b9eed267f9b544985Andy McFadden
630190fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnarbool ACodec::ExecutingState::onOMXFrameRendered(int64_t mediaTimeUs, nsecs_t systemNano) {
630290fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    mCodec->onFrameRendered(mediaTimeUs, systemNano);
630390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    return true;
630490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar}
630590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
6306f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingState::onOMXEvent(
6307f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
6308f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
6309f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventPortSettingsChanged:
6310f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6311f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)kPortIndexOutput);
6312f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
631331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            if (data2 == 0 || data2 == OMX_IndexParamPortDefinition) {
6314054219874873b41f1c815552987c10465c34ba2bLajos Molnar                mCodec->mMetadataBuffersToSubmit = 0;
6315f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->sendCommand(
6316f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode,
6317f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            OMX_CommandPortDisable, kPortIndexOutput),
6318f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
6319f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6320349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                mCodec->freeOutputBuffersNotOwnedByComponent();
6321f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6322f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->changeState(mCodec->mOutputPortSettingsChangedState);
632331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            } else if (data2 == OMX_IndexConfigCommonOutputCrop) {
632431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mCodec->mSentFormat = false;
6325f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else {
6326ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar                ALOGV("[%s] OMX_EventPortSettingsChanged 0x%08x",
6327f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->mComponentName.c_str(), data2);
6328f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
6329f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6330f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
6331f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6333f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventBufferFlag:
6334f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
6336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6337f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6338f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
6340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6341f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6342f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6343f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
6344f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6345f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::OutputPortSettingsChangedState::OutputPortSettingsChangedState(
6346f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ACodec *codec)
6347f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
6348f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6349f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6350f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::OutputPortSettingsChangedState::getPortMode(
6351f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
6352f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexOutput) {
6353f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return FREE_BUFFERS;
6354f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6355f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6356f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(portIndex, (OMX_U32)kPortIndexInput);
6357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6358f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return RESUBMIT_BUFFERS;
6359f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6361f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::OutputPortSettingsChangedState::onMessageReceived(
6362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
6363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
6364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6365f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
6366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
6367f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
6368349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case kWhatResume:
6369e6eea3a6b6e0cf92287ec1ceb8350f201d17e1acPer Persson        case kWhatSetParameters:
6370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6371349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            if (msg->what() == kWhatResume) {
63723856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("[%s] Deferring resume", mCodec->mComponentName.c_str());
6373349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            }
6374349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6375f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
6376f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
6377f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6378f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
6382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6384f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6385f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
6386f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6388f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::OutputPortSettingsChangedState::stateEntered() {
63893856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%s] Now handling output port settings change",
6390f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mCodec->mComponentName.c_str());
6391f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6392f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
639390fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnarbool ACodec::OutputPortSettingsChangedState::onOMXFrameRendered(
639490fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        int64_t mediaTimeUs, nsecs_t systemNano) {
639590fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    mCodec->onFrameRendered(mediaTimeUs, systemNano);
639690fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar    return true;
639790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar}
639890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
6399f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::OutputPortSettingsChangedState::onOMXEvent(
6400f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
6401f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
6402f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
6403f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6404f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (data1 == (OMX_U32)OMX_CommandPortDisable) {
64050806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (data2 != (OMX_U32)kPortIndexOutput) {
64060806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    ALOGW("ignoring EventCmdComplete CommandPortDisable for port %u", data2);
64070806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    return false;
64080806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
6409f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
64100806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGV("[%s] Output port now disabled.", mCodec->mComponentName.c_str());
6411f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
64120806340688c937e7b78c2d89db3809274130df4eLajos Molnar                status_t err = OK;
64130806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (!mCodec->mBuffers[kPortIndexOutput].isEmpty()) {
64140806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    ALOGE("disabled port should be empty, but has %zu buffers",
64150806340688c937e7b78c2d89db3809274130df4eLajos Molnar                            mCodec->mBuffers[kPortIndexOutput].size());
64160806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    err = FAILED_TRANSACTION;
64170806340688c937e7b78c2d89db3809274130df4eLajos Molnar                } else {
64180806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    mCodec->mDealer[kPortIndexOutput].clear();
64190806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
6420f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
64210806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (err == OK) {
64220806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    err = mCodec->mOMX->sendCommand(
64230806340688c937e7b78c2d89db3809274130df4eLajos Molnar                            mCodec->mNode, OMX_CommandPortEnable, kPortIndexOutput);
64240806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
6425f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
64260806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (err == OK) {
64270806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    err = mCodec->allocateBuffersOnPort(kPortIndexOutput);
64280806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    ALOGE_IF(err != OK, "Failed to allocate output port buffers after port "
64290806340688c937e7b78c2d89db3809274130df4eLajos Molnar                            "reconfiguration: (%d)", err);
64300806340688c937e7b78c2d89db3809274130df4eLajos Molnar                }
6431cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber
64320806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (err != OK) {
6433251d4be8aa5ab80bc915a82a2420233bdc62018eAndy Hung                    mCodec->signalError(OMX_ErrorUndefined, makeNoSideEffectStatus(err));
6434d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber
6435755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    // This is technically not correct, but appears to be
6436755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    // the only way to free the component instance.
6437755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    // Controlled transitioning from excecuting->idle
6438755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    // and idle->loaded seem impossible probably because
6439755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    // the output port never finishes re-enabling.
6440755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    mCodec->mShutdownInProgress = true;
6441755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    mCodec->mKeepComponentAllocated = false;
6442755b9861197f1566f916785ba7d35cf86fd5b2d5Andreas Huber                    mCodec->changeState(mCodec->mLoadedState);
6443cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                }
6444f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6445f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                return true;
6446f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else if (data1 == (OMX_U32)OMX_CommandPortEnable) {
6447777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                if (data2 != (OMX_U32)kPortIndexOutput) {
6448777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    ALOGW("ignoring EventCmdComplete OMX_CommandPortEnable for port %u", data2);
6449777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    return false;
6450777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                }
6451f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
645231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mCodec->mSentFormat = false;
645331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
64540806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGV("[%s] Output port now reenabled.", mCodec->mComponentName.c_str());
6455f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6456349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                if (mCodec->mExecutingState->active()) {
6457349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                    mCodec->mExecutingState->submitOutputBuffers();
6458349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                }
6459349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6460f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->changeState(mCodec->mExecutingState);
6461f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6462f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                return true;
6463f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
6464f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6465f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
6466f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6467f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6468f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6469f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
6470f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6471f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6472f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6473f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
6474f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6475f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ExecutingToIdleState::ExecutingToIdleState(ACodec *codec)
64765778822d86b0337407514b9372562b86edfa91cdAndreas Huber    : BaseState(codec),
64775778822d86b0337407514b9372562b86edfa91cdAndreas Huber      mComponentNowIdle(false) {
6478f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6479f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6480f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingToIdleState::onMessageReceived(const sp<AMessage> &msg) {
6481f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
6482f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6483f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
6484f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
6485f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6486f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // Don't send me a flush request if you previously wanted me
6487f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // to shutdown.
64880806340688c937e7b78c2d89db3809274130df4eLajos Molnar            ALOGW("Ignoring flush request in ExecutingToIdleState");
6489f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6490f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6491f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6492f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
6493f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6494f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing that...
6495f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6496f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
6497f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6498f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6499f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6500f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6501f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
6502f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6503f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6504f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6505f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
6506f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6507f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6508f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::stateEntered() {
65093856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%s] Now Executing->Idle", mCodec->mComponentName.c_str());
651031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
65115778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mComponentNowIdle = false;
651231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    mCodec->mSentFormat = false;
6513f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6514f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6515f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingToIdleState::onOMXEvent(
6516f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
6517f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
6518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
6519f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
65200806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (data1 != (OMX_U32)OMX_CommandStateSet
65210806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    || data2 != (OMX_U32)OMX_StateIdle) {
65220806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGE("Unexpected command completion in ExecutingToIdleState: %s(%u) %s(%u)",
65230806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_COMMANDTYPE)data1), data1,
65240806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_STATETYPE)data2), data2);
65250806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
65260806340688c937e7b78c2d89db3809274130df4eLajos Molnar                return true;
65270806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
6528f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
65295778822d86b0337407514b9372562b86edfa91cdAndreas Huber            mComponentNowIdle = true;
65305778822d86b0337407514b9372562b86edfa91cdAndreas Huber
6531f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            changeStateIfWeOwnAllBuffers();
6532f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6533f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
6534f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6535f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6536349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventPortSettingsChanged:
6537349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventBufferFlag:
6538349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        {
6539349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            // We're shutting down and don't care about this anymore.
6540349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            return true;
6541349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        }
6542349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6543f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6544f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
6545f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6546f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
65470af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
6548f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::changeStateIfWeOwnAllBuffers() {
65495778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (mComponentNowIdle && mCodec->allYourBuffersAreBelongToUs()) {
65500806340688c937e7b78c2d89db3809274130df4eLajos Molnar        status_t err = mCodec->mOMX->sendCommand(
65510806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->mNode, OMX_CommandStateSet, OMX_StateLoaded);
65520806340688c937e7b78c2d89db3809274130df4eLajos Molnar        if (err == OK) {
65530806340688c937e7b78c2d89db3809274130df4eLajos Molnar            err = mCodec->freeBuffersOnPort(kPortIndexInput);
65540806340688c937e7b78c2d89db3809274130df4eLajos Molnar            status_t err2 = mCodec->freeBuffersOnPort(kPortIndexOutput);
65550806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (err == OK) {
65560806340688c937e7b78c2d89db3809274130df4eLajos Molnar                err = err2;
65570806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
65580806340688c937e7b78c2d89db3809274130df4eLajos Molnar        }
6559f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
65600167414e261f88a96b5e4bf6cb592e6ca11e5a95Andreas Huber        if ((mCodec->mFlags & kFlagPushBlankBuffersToNativeWindowOnShutdown)
65610167414e261f88a96b5e4bf6cb592e6ca11e5a95Andreas Huber                && mCodec->mNativeWindow != NULL) {
6562bc098410be55f9d96f394b3981a0c482b83859b6Andreas Huber            // We push enough 1x1 blank buffers to ensure that one of
6563bc098410be55f9d96f394b3981a0c482b83859b6Andreas Huber            // them has made it to the display.  This allows the OMX
6564bc098410be55f9d96f394b3981a0c482b83859b6Andreas Huber            // component teardown to zero out any protected buffers
6565bc098410be55f9d96f394b3981a0c482b83859b6Andreas Huber            // without the risk of scanning out one of those buffers.
6566b2d0b487efd40700199852c9a18b369b1651f15bLajos Molnar            pushBlankBuffersToNativeWindow(mCodec->mNativeWindow.get());
6567bc098410be55f9d96f394b3981a0c482b83859b6Andreas Huber        }
6568bc098410be55f9d96f394b3981a0c482b83859b6Andreas Huber
65690806340688c937e7b78c2d89db3809274130df4eLajos Molnar        if (err != OK) {
65700806340688c937e7b78c2d89db3809274130df4eLajos Molnar            mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
65710806340688c937e7b78c2d89db3809274130df4eLajos Molnar            return;
65720806340688c937e7b78c2d89db3809274130df4eLajos Molnar        }
65730806340688c937e7b78c2d89db3809274130df4eLajos Molnar
6574f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->changeState(mCodec->mIdleToLoadedState);
6575f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6576f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6577f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6578f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::onInputBufferFilled(
6579f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
6580f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onInputBufferFilled(msg);
6581f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6582f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
6583f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6584f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6585f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::onOutputBufferDrained(
6586f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
6587f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onOutputBufferDrained(msg);
6588f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6589f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
6590f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6591f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6592f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
6593f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6594f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::IdleToLoadedState::IdleToLoadedState(ACodec *codec)
6595f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
6596f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6597f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6598f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToLoadedState::onMessageReceived(const sp<AMessage> &msg) {
6599f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
6600f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6601f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
6602f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
6603f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6604f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing that...
6605f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6606f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
6607f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6608f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6609f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6610f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
6611f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6612f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // Don't send me a flush request if you previously wanted me
6613f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // to shutdown.
66140806340688c937e7b78c2d89db3809274130df4eLajos Molnar            ALOGE("Got flush request in IdleToLoadedState");
6615f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6616f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6617f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6618f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6619f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
6620f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6621f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6622f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6623f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
6624f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6625f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6626f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::IdleToLoadedState::stateEntered() {
66273856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%s] Now Idle->Loaded", mCodec->mComponentName.c_str());
6628f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6629f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6630f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToLoadedState::onOMXEvent(
6631f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
6632f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
6633f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
6634f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
66350806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (data1 != (OMX_U32)OMX_CommandStateSet
66360806340688c937e7b78c2d89db3809274130df4eLajos Molnar                    || data2 != (OMX_U32)OMX_StateLoaded) {
66370806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGE("Unexpected command completion in IdleToLoadedState: %s(%u) %s(%u)",
66380806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_COMMANDTYPE)data1), data1,
66390806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_STATETYPE)data2), data2);
66400806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
66410806340688c937e7b78c2d89db3809274130df4eLajos Molnar                return true;
66420806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
6643f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6644c95c2ddcdfc974f42408a377fbe2de51b94a8c94Andreas Huber            mCodec->changeState(mCodec->mLoadedState);
6645f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6646f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
6647f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6648f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6649f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6650f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
6651f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6652f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6653f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6654f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
6655f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6656f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::FlushingState::FlushingState(ACodec *codec)
6657f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
6658f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6659f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6660f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::stateEntered() {
66613856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%s] Now Flushing", mCodec->mComponentName.c_str());
6662f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6663f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mFlushComplete[kPortIndexInput] = mFlushComplete[kPortIndexOutput] = false;
6664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6666f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::FlushingState::onMessageReceived(const sp<AMessage> &msg) {
6667f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
6668f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6669f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
6670f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
6671f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6672f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
6673f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6674f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6675f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6676f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
6677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
6678f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing this right now.
6679f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
6680f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6681f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6682f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6683f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6684f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
6685f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
6686f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6687f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6688f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
6689f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6690f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6691f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::FlushingState::onOMXEvent(
6692f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
6693ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar    ALOGV("[%s] FlushingState onOMXEvent(%u,%d)",
6694ee4e1b1a63758941460ae79a064249d3a5189443Lajos Molnar            mCodec->mComponentName.c_str(), event, (OMX_S32)data1);
6695d3d822204249eaca4259bdf46f6f0357b96e8e21Andreas Huber
6696f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
6697f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
6698f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
66990806340688c937e7b78c2d89db3809274130df4eLajos Molnar            if (data1 != (OMX_U32)OMX_CommandFlush) {
67000806340688c937e7b78c2d89db3809274130df4eLajos Molnar                ALOGE("unexpected EventCmdComplete %s(%d) data2:%d in FlushingState",
67010806340688c937e7b78c2d89db3809274130df4eLajos Molnar                        asString((OMX_COMMANDTYPE)data1), data1, data2);
67020806340688c937e7b78c2d89db3809274130df4eLajos Molnar                mCodec->signalError(OMX_ErrorUndefined, FAILED_TRANSACTION);
67030806340688c937e7b78c2d89db3809274130df4eLajos Molnar                return true;
67040806340688c937e7b78c2d89db3809274130df4eLajos Molnar            }
6705f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6706f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (data2 == kPortIndexInput || data2 == kPortIndexOutput) {
6707777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                if (mFlushComplete[data2]) {
6708777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    ALOGW("Flush already completed for %s port",
6709777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            data2 == kPortIndexInput ? "input" : "output");
6710777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    return true;
6711777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                }
6712f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mFlushComplete[data2] = true;
6713e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber
67140806340688c937e7b78c2d89db3809274130df4eLajos Molnar                if (mFlushComplete[kPortIndexInput] && mFlushComplete[kPortIndexOutput]) {
6715e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                    changeStateIfWeOwnAllBuffers();
6716e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                }
6717777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            } else if (data2 == OMX_ALL) {
6718777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                if (!mFlushComplete[kPortIndexInput] || !mFlushComplete[kPortIndexOutput]) {
6719777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    ALOGW("received flush complete event for OMX_ALL before ports have been"
6720777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            "flushed (%d/%d)",
6721777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                            mFlushComplete[kPortIndexInput], mFlushComplete[kPortIndexOutput]);
6722777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                    return false;
6723777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                }
6724f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6725f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                changeStateIfWeOwnAllBuffers();
6726777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim            } else {
6727777cf6d2788b12bbd68f620cda94a128462a208dJinsuk Kim                ALOGW("data2 not OMX_ALL but %u in EventCmdComplete CommandFlush", data2);
6728f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
6729f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6730f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
6731f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
6732f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6733349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventPortSettingsChanged:
6734349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        {
67351d15ab58bf8239069ef343de6cb21aabf3ef7d78Lajos Molnar            sp<AMessage> msg = new AMessage(kWhatOMXMessage, mCodec);
6736349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("type", omx_message::EVENT);
6737609b815a3131d22da38b2f452faa9f89daad4039Andy Hung            msg->setInt32("node", mCodec->mNode);
6738349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("event", event);
6739349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("data1", data1);
6740349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("data2", data2);
6741349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
67423856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV("[%s] Deferring OMX_EventPortSettingsChanged",
6743349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                 mCodec->mComponentName.c_str());
6744349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6745349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mCodec->deferMessage(msg);
6746349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6747349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            return true;
6748349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        }
6749349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
6750f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
6751f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
6752f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6753f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6754f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
6755f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6757f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::onOutputBufferDrained(const sp<AMessage> &msg) {
6758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onOutputBufferDrained(msg);
6759f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
6761f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6762f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6763f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::onInputBufferFilled(const sp<AMessage> &msg) {
6764f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onInputBufferFilled(msg);
6765f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6766f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
6767f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6768f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6769f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::changeStateIfWeOwnAllBuffers() {
6770f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mFlushComplete[kPortIndexInput]
6771f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && mFlushComplete[kPortIndexOutput]
6772f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && mCodec->allYourBuffersAreBelongToUs()) {
67737e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber        // We now own all buffers except possibly those still queued with
67747e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber        // the native window for rendering. Let's get those back as well.
67757e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber        mCodec->waitUntilAllPossibleNativeWindowBuffersAreReturnedToUs();
67767e7013392e302a28364df1dcee79b82ad90978b4Andreas Huber
677790fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar        mCodec->mRenderTracker.clear(systemTime(CLOCK_MONOTONIC));
677890fcf68fd29f3cb695bd53a830ad984cb7d430c0Lajos Molnar
6779f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<AMessage> notify = mCodec->mNotify->dup();
6780d8cbe4a024ef54adf043b6ea31fa22271b8b2c51Lajos Molnar        notify->setInt32("what", CodecBase::kWhatFlushCompleted);
6781f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        notify->post();
6782f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6783f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->mPortEOS[kPortIndexInput] =
6784f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mPortEOS[kPortIndexOutput] = false;
6785f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6786dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber        mCodec->mInputEOSResult = OK;
6787dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber
6788f3bd1972e039c6ded5154db715e5a32f1813a239Marco Nelissen        if (mCodec->mSkipCutBuffer != NULL) {
6789f3bd1972e039c6ded5154db715e5a32f1813a239Marco Nelissen            mCodec->mSkipCutBuffer->clear();
6790f3bd1972e039c6ded5154db715e5a32f1813a239Marco Nelissen        }
6791f3bd1972e039c6ded5154db715e5a32f1813a239Marco Nelissen
6792f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->changeState(mCodec->mExecutingState);
6793f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
6794f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
6795f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
6796f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}  // namespace android
6797