ACodec.cpp revision dc9bacd838442a524585887e6ea6696836be2eda
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
20f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/ACodec.h>
21f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
22f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <binder/MemoryDealer.h>
23f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
24f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/hexdump.h>
25f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/ABuffer.h>
26f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/ADebug.h>
27f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/foundation/AMessage.h>
28f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
29f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/MediaDefs.h>
301173118eace0e9e347cb007f0da817cee87579edGlenn Kasten#include <media/stagefright/NativeWindowWrapper.h>
31f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <media/stagefright/OMXClient.h>
321065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber#include <media/stagefright/OMXCodec.h>
33f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
34f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <surfaceflinger/Surface.h>
351173118eace0e9e347cb007f0da817cee87579edGlenn Kasten#include <gui/SurfaceTextureClient.h>
36f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
37f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include <OMX_Component.h>
38f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
39f933441648ef6a71dee783d733aac17b9508b452Andreas Hubernamespace android {
40f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
41f933441648ef6a71dee783d733aac17b9508b452Andreas Hubertemplate<class T>
42f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatic void InitOMXParams(T *params) {
43f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nSize = sizeof(T);
44f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nVersionMajor = 1;
45f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nVersionMinor = 0;
46f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nRevision = 0;
47f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    params->nVersion.s.nStep = 0;
48f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
49f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
50f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct CodecObserver : public BnOMXObserver {
51f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CodecObserver() {}
52f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
53f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void setNotificationMessage(const sp<AMessage> &msg) {
54f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mNotify = msg;
55f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
56f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
57f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // from IOMXObserver
58f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onMessage(const omx_message &omx_msg) {
59f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<AMessage> msg = mNotify->dup();
60f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
61f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        msg->setInt32("type", omx_msg.type);
62f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        msg->setPointer("node", omx_msg.node);
63f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
64f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        switch (omx_msg.type) {
65f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            case omx_message::EVENT:
66f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            {
67f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setInt32("event", omx_msg.u.event_data.event);
68f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setInt32("data1", omx_msg.u.event_data.data1);
69f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setInt32("data2", omx_msg.u.event_data.data2);
70f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
71f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
72f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
73f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            case omx_message::EMPTY_BUFFER_DONE:
74f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            {
75f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setPointer("buffer", omx_msg.u.buffer_data.buffer);
76f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
77f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
78f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
79f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            case omx_message::FILL_BUFFER_DONE:
80f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            {
81f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setPointer(
82f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        "buffer", omx_msg.u.extended_buffer_data.buffer);
83f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setInt32(
84f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        "range_offset",
85f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        omx_msg.u.extended_buffer_data.range_offset);
86f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setInt32(
87f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        "range_length",
88f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        omx_msg.u.extended_buffer_data.range_length);
89f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setInt32(
90f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        "flags",
91f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        omx_msg.u.extended_buffer_data.flags);
92f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setInt64(
93f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        "timestamp",
94f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        omx_msg.u.extended_buffer_data.timestamp);
95f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setPointer(
96f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        "platform_private",
97f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        omx_msg.u.extended_buffer_data.platform_private);
98f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                msg->setPointer(
99f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        "data_ptr",
100f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        omx_msg.u.extended_buffer_data.data_ptr);
101f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
102f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
103f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
104f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            default:
105f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                TRESPASS();
106f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
107f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
108f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
109f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        msg->post();
110f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
111f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
112f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
113f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual ~CodecObserver() {}
114f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
115f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
116f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> mNotify;
117f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
118f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(CodecObserver);
119f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
120f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
121f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
122f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
123f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::BaseState : public AState {
124f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState(ACodec *codec, const sp<AState> &parentState = NULL);
125f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
126f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
127f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    enum PortMode {
128f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        KEEP_BUFFERS,
129f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        RESUBMIT_BUFFERS,
130f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        FREE_BUFFERS,
131f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
132f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
133f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ACodec *mCodec;
134f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
135f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual PortMode getPortMode(OMX_U32 portIndex);
136f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
137f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
138f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
139f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
140f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
141f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onOutputBufferDrained(const sp<AMessage> &msg);
142f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onInputBufferFilled(const sp<AMessage> &msg);
143f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
144f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void postFillThisBuffer(BufferInfo *info);
145f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
146f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
147f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool onOMXMessage(const sp<AMessage> &msg);
148f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
149f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool onOMXEmptyBufferDone(IOMX::buffer_id bufferID);
150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
151f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool onOMXFillBufferDone(
152f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID,
153f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            size_t rangeOffset, size_t rangeLength,
154f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            OMX_U32 flags,
155f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int64_t timeUs,
156f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            void *platformPrivate,
157f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            void *dataPtr);
158f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
159f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void getMoreInputDataIfPossible();
160f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
161f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(BaseState);
162f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
163f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
164f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
165f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
166f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::UninitializedState : public ACodec::BaseState {
167f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    UninitializedState(ACodec *codec);
168f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
169f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
170f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
171f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
172f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
173f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void onSetup(const sp<AMessage> &msg);
174f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
175f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(UninitializedState);
176f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
177f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
178f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
179f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
180f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::LoadedToIdleState : public ACodec::BaseState {
181f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LoadedToIdleState(ACodec *codec);
182f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
183f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
184f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
185f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
186f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
187f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
188f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
189f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t allocateBuffers();
190f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
191f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(LoadedToIdleState);
192f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
193f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
194f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
195f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
196f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::IdleToExecutingState : public ACodec::BaseState {
197f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IdleToExecutingState(ACodec *codec);
198f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
199f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
200f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
201f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
202f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
203f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
204f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
205f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(IdleToExecutingState);
206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
207f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
210f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::ExecutingState : public ACodec::BaseState {
211f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ExecutingState(ACodec *codec);
212f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
213f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void submitOutputBuffers();
214f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
215f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Submit output buffers to the decoder, submit input buffers to client
216f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // to fill with data.
217f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void resume();
218f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
219349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    // Returns true iff input and output buffers are in play.
220349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    bool active() const { return mActive; }
221349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
222f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
223f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual PortMode getPortMode(OMX_U32 portIndex);
224f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
225f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
226f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
227f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
228f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
229f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
230349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    bool mActive;
231349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
232f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(ExecutingState);
233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
235f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
236f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
237f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::OutputPortSettingsChangedState : public ACodec::BaseState {
238f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OutputPortSettingsChangedState(ACodec *codec);
239f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
240f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
241f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual PortMode getPortMode(OMX_U32 portIndex);
242f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
243f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
244f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
245f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
246f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
247f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
248f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(OutputPortSettingsChangedState);
249f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
250f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
251f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
252f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
253f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::ExecutingToIdleState : public ACodec::BaseState {
254f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ExecutingToIdleState(ACodec *codec);
255f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
256f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
257f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
258f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
259f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
260f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
261f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
262f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onOutputBufferDrained(const sp<AMessage> &msg);
263f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onInputBufferFilled(const sp<AMessage> &msg);
264f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
265f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
266f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void changeStateIfWeOwnAllBuffers();
267f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
268f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(ExecutingToIdleState);
269f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
270f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
271f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
272f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
273f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::IdleToLoadedState : public ACodec::BaseState {
274f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IdleToLoadedState(ACodec *codec);
275f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
276f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
277f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
278f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
279f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
280f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
281f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
282f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
283f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(IdleToLoadedState);
284f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
285f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
286f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
287f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
288f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstruct ACodec::FlushingState : public ACodec::BaseState {
289f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    FlushingState(ACodec *codec);
290f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
291f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprotected:
292f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onMessageReceived(const sp<AMessage> &msg);
293f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void stateEntered();
294f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
295f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual bool onOMXEvent(OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2);
296f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
297f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onOutputBufferDrained(const sp<AMessage> &msg);
298f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    virtual void onInputBufferFilled(const sp<AMessage> &msg);
299f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
300f933441648ef6a71dee783d733aac17b9508b452Andreas Huberprivate:
301f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool mFlushComplete[2];
302f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
303f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    void changeStateIfWeOwnAllBuffers();
304f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
305f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    DISALLOW_EVIL_CONSTRUCTORS(FlushingState);
306f933441648ef6a71dee783d733aac17b9508b452Andreas Huber};
307f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
308f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
309f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
310f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ACodec()
31131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    : mNode(NULL),
31231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber      mSentFormat(false) {
313f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mUninitializedState = new UninitializedState(this);
314f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mLoadedToIdleState = new LoadedToIdleState(this);
315f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mIdleToExecutingState = new IdleToExecutingState(this);
316f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mExecutingState = new ExecutingState(this);
317f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
318f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mOutputPortSettingsChangedState =
319f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        new OutputPortSettingsChangedState(this);
320f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
321f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mExecutingToIdleState = new ExecutingToIdleState(this);
322f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mIdleToLoadedState = new IdleToLoadedState(this);
323f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mFlushingState = new FlushingState(this);
324f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
325f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mPortEOS[kPortIndexInput] = mPortEOS[kPortIndexOutput] = false;
326dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber    mInputEOSResult = OK;
327f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
328f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeState(mUninitializedState);
329f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
330f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
331f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::~ACodec() {
332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
333f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
334f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::setNotificationMessage(const sp<AMessage> &msg) {
335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mNotify = msg;
336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
337f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
338f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::initiateSetup(const sp<AMessage> &msg) {
339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->setWhat(kWhatSetup);
340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->setTarget(id());
341f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->post();
342f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
343f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
344f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::signalFlush() {
345f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    (new AMessage(kWhatFlush, id()))->post();
346f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
347f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
348f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::signalResume() {
349f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    (new AMessage(kWhatResume, id()))->post();
350f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
351f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
352f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::initiateShutdown() {
353f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    (new AMessage(kWhatShutdown, id()))->post();
354f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
355f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
356f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::allocateBuffersOnPort(OMX_U32 portIndex) {
357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(portIndex == kPortIndexInput || portIndex == kPortIndexOutput);
358f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
359f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(mDealer[portIndex] == NULL);
360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(mBuffers[portIndex].isEmpty());
361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mNativeWindow != NULL && portIndex == kPortIndexOutput) {
363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return allocateOutputBuffersFromNativeWindow();
364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
365f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
367f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
368f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
369f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
373f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
374f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
375f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
376f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
377f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LOGV("[%s] Allocating %lu buffers of size %lu on %s port",
378f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mComponentName.c_str(),
379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.nBufferCountActual, def.nBufferSize,
380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            portIndex == kPortIndexInput ? "input" : "output");
381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    size_t totalSize = def.nBufferCountActual * def.nBufferSize;
383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDealer[portIndex] = new MemoryDealer(totalSize, "OMXCodec");
384f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
385f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (OMX_U32 i = 0; i < def.nBufferCountActual; ++i) {
386f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<IMemory> mem = mDealer[portIndex]->allocate(def.nBufferSize);
387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(mem.get() != NULL);
388f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
389f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id buffer;
3901065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
3911065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        if (!strcasecmp(
3921065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                    mComponentName.c_str(), "OMX.TI.DUCATI1.VIDEO.DECODER")) {
3931065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            if (portIndex == kPortIndexInput && i == 0) {
3941065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                // Only log this warning once per allocation round.
3951065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
3961065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                LOGW("OMX.TI.DUCATI1.VIDEO.DECODER requires the use of "
3971065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                     "OMX_AllocateBuffer instead of the preferred "
3981065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                     "OMX_UseBuffer. Vendor must fix this.");
3991065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            }
4001065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
4011065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            err = mOMX->allocateBufferWithBackup(
4021065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                    mNode, portIndex, mem, &buffer);
4031065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        } else {
4041065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            err = mOMX->useBuffer(mNode, portIndex, mem, &buffer);
4051065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        }
406f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
407f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
408f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return err;
409f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
410f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
411f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo info;
412f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info.mBufferID = buffer;
413f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info.mStatus = BufferInfo::OWNED_BY_US;
414f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info.mData = new ABuffer(mem->pointer(), def.nBufferSize);
415f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mBuffers[portIndex].push(info);
416f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
417f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
418f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
419f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
420f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
421f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::allocateOutputBuffersFromNativeWindow() {
422f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
423f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
424f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = kPortIndexOutput;
425f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
426f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
427f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
428f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
429f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
430f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
431f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
432f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4339bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    err = native_window_set_scaling_mode(mNativeWindow.get(),
4349bc7af17974f448291a44912566ec7472a0d798bMathias Agopian            NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW);
4359bc7af17974f448291a44912566ec7472a0d798bMathias Agopian
4369bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    if (err != OK) {
4379bc7af17974f448291a44912566ec7472a0d798bMathias Agopian        return err;
4389bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    }
4399bc7af17974f448291a44912566ec7472a0d798bMathias Agopian
440f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = native_window_set_buffers_geometry(
441f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNativeWindow.get(),
442f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.format.video.nFrameWidth,
443f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.format.video.nFrameHeight,
444f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.format.video.eColorFormat);
445f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
446f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
447f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGE("native_window_set_buffers_geometry failed: %s (%d)",
448f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                strerror(-err), -err);
449f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
450f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
451f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
452f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Set up the native window.
4533c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    OMX_U32 usage = 0;
4543c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    err = mOMX->getGraphicBufferUsage(mNode, kPortIndexOutput, &usage);
4553c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    if (err != 0) {
4563c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        LOGW("querying usage flags from OMX IL component failed: %d", err);
4573c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        // XXX: Currently this error is logged, but not fatal.
4583c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        usage = 0;
4593c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    }
4603c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis
461f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = native_window_set_usage(
462f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNativeWindow.get(),
4633c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis            usage | GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_EXTERNAL_DISP);
464f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
465f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
466f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGE("native_window_set_usage failed: %s (%d)", strerror(-err), -err);
467f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
468f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
469f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
470258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    int minUndequeuedBufs = 0;
471258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    err = mNativeWindow->query(
472258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            mNativeWindow.get(), NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS,
473258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            &minUndequeuedBufs);
474258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
475258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    if (err != 0) {
476258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        LOGE("NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS query failed: %s (%d)",
477258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                strerror(-err), -err);
478258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        return err;
479258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
480258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
481258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // XXX: Is this the right logic to use?  It's not clear to me what the OMX
482258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // buffer counts refer to - how do they account for the renderer holding on
483258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // to buffers?
484258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    if (def.nBufferCountActual < def.nBufferCountMin + minUndequeuedBufs) {
485258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        OMX_U32 newBufferCount = def.nBufferCountMin + minUndequeuedBufs;
486258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        def.nBufferCountActual = newBufferCount;
487258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        err = mOMX->setParameter(
488258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
489258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
490258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        if (err != OK) {
491258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            LOGE("[%s] setting nBufferCountActual to %lu failed: %d",
492258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                    mComponentName.c_str(), newBufferCount, err);
493258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            return err;
494258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        }
495258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
496258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
497f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = native_window_set_buffer_count(
498f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNativeWindow.get(), def.nBufferCountActual);
499f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
500f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
501f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGE("native_window_set_buffer_count failed: %s (%d)", strerror(-err),
502f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                -err);
503f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
504f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
505f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
506f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LOGV("[%s] Allocating %lu buffers from a native window of size %lu on "
507f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         "output port",
508f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mComponentName.c_str(), def.nBufferCountActual, def.nBufferSize);
509f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
510f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Dequeue buffers and send them to OMX
51174006804065941841883c4b46ee785070164023fJamie Gennis    for (OMX_U32 i = 0; i < def.nBufferCountActual; i++) {
5128ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev        ANativeWindowBuffer *buf;
513f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf);
514f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != 0) {
515f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            LOGE("dequeueBuffer failed: %s (%d)", strerror(-err), -err);
516f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
517f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
519f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(buf, false));
52074006804065941841883c4b46ee785070164023fJamie Gennis        BufferInfo info;
52174006804065941841883c4b46ee785070164023fJamie Gennis        info.mStatus = BufferInfo::OWNED_BY_US;
52274006804065941841883c4b46ee785070164023fJamie Gennis        info.mData = new ABuffer(0);
52374006804065941841883c4b46ee785070164023fJamie Gennis        info.mGraphicBuffer = graphicBuffer;
52474006804065941841883c4b46ee785070164023fJamie Gennis        mBuffers[kPortIndexOutput].push(info);
52574006804065941841883c4b46ee785070164023fJamie Gennis
526f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id bufferId;
527f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        err = mOMX->useGraphicBuffer(mNode, kPortIndexOutput, graphicBuffer,
528f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &bufferId);
529f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != 0) {
53074006804065941841883c4b46ee785070164023fJamie Gennis            LOGE("registering GraphicBuffer %lu with OMX IL component failed: "
53174006804065941841883c4b46ee785070164023fJamie Gennis                 "%d", i, err);
532f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
533f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
534f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
53574006804065941841883c4b46ee785070164023fJamie Gennis        mBuffers[kPortIndexOutput].editItemAt(i).mBufferID = bufferId;
53674006804065941841883c4b46ee785070164023fJamie Gennis
537f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGV("[%s] Registered graphic buffer with ID %p (pointer = %p)",
538f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             mComponentName.c_str(),
539f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             bufferId, graphicBuffer.get());
540f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
541f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
542f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 cancelStart;
543f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 cancelEnd;
544f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
545f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
546f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // If an error occurred while dequeuing we need to cancel any buffers
547f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // that were dequeued.
548f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        cancelStart = 0;
54974006804065941841883c4b46ee785070164023fJamie Gennis        cancelEnd = mBuffers[kPortIndexOutput].size();
550f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
551f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // Return the last two buffers to the native window.
552258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        cancelStart = def.nBufferCountActual - minUndequeuedBufs;
553f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        cancelEnd = def.nBufferCountActual;
554f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
555f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
556f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (OMX_U32 i = cancelStart; i < cancelEnd; i++) {
557f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[kPortIndexOutput].editItemAt(i);
558f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        cancelBufferToNativeWindow(info);
559f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
560f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
561f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
562f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
563f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
564f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::cancelBufferToNativeWindow(BufferInfo *info) {
565f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
566f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
567f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LOGV("[%s] Calling cancelBuffer on buffer %p",
568f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mComponentName.c_str(), info->mBufferID);
569f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
570f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int err = mNativeWindow->cancelBuffer(
571f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mNativeWindow.get(), info->mGraphicBuffer.get());
572f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
573f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(err, 0);
574f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
575f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_NATIVE_WINDOW;
576f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
577f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
578f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
579f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
580f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BufferInfo *ACodec::dequeueBufferFromNativeWindow() {
5818ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev    ANativeWindowBuffer *buf;
582c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    if (mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf) != 0) {
583c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        LOGE("dequeueBuffer failed.");
584c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        return NULL;
585c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    }
586f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
587f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = mBuffers[kPortIndexOutput].size(); i-- > 0;) {
588f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info =
589f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &mBuffers[kPortIndexOutput].editItemAt(i);
590f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
591f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mGraphicBuffer->handle == buf->handle) {
592f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)info->mStatus,
593f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (int)BufferInfo::OWNED_BY_NATIVE_WINDOW);
594f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
595f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            info->mStatus = BufferInfo::OWNED_BY_US;
596f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
597f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return info;
598f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
599f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
600f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
601f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    TRESPASS();
602f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
603f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return NULL;
604f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
605f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
606f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::freeBuffersOnPort(OMX_U32 portIndex) {
607f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = mBuffers[portIndex].size(); i-- > 0;) {
608f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ((status_t)OK, freeBuffer(portIndex, i));
609f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
610f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
611f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDealer[portIndex].clear();
612f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
613f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
614f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
615f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
616349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huberstatus_t ACodec::freeOutputBuffersNotOwnedByComponent() {
617f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = mBuffers[kPortIndexOutput].size(); i-- > 0;) {
618f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info =
619f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &mBuffers[kPortIndexOutput].editItemAt(i);
620f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
621349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        if (info->mStatus !=
622349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                BufferInfo::OWNED_BY_COMPONENT) {
623349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            // We shouldn't have sent out any buffers to the client at this
624349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            // point.
625349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            CHECK_NE((int)info->mStatus, (int)BufferInfo::OWNED_BY_DOWNSTREAM);
626349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
627f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((status_t)OK, freeBuffer(kPortIndexOutput, i));
628f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
629f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
630f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
631f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
632f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
633f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
634f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::freeBuffer(OMX_U32 portIndex, size_t i) {
635f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
636f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
637f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(info->mStatus == BufferInfo::OWNED_BY_US
638f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            || info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW);
639f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
640f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexOutput && mNativeWindow != NULL
641f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && info->mStatus == BufferInfo::OWNED_BY_US) {
642f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ((status_t)OK, cancelBufferToNativeWindow(info));
643f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
644f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
645f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(mOMX->freeBuffer(
646f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, portIndex, info->mBufferID),
647f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             (status_t)OK);
648f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
649f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mBuffers[portIndex].removeAt(i);
650f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
651f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
652f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
653f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
654f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BufferInfo *ACodec::findBufferByID(
655f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        uint32_t portIndex, IOMX::buffer_id bufferID,
656f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ssize_t *index) {
657f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
658f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
659f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
660f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mBufferID == bufferID) {
661f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (index != NULL) {
662f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                *index = i;
663f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return info;
665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
666f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
667f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
668f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    TRESPASS();
669f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
670f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return NULL;
671f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
672f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
673f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::setComponentRole(
674f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        bool isEncoder, const char *mime) {
675f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    struct MimeToRole {
676f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime;
677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *decoderRole;
678f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *encoderRole;
679f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
680f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
681f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    static const MimeToRole kMimeToRole[] = {
682f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_MPEG,
683f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.mp3", "audio_encoder.mp3" },
684f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_NB,
685f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.amrnb", "audio_encoder.amrnb" },
686f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_WB,
687f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.amrwb", "audio_encoder.amrwb" },
688f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AAC,
689f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.aac", "audio_encoder.aac" },
690f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_AVC,
691f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.avc", "video_encoder.avc" },
692f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_MPEG4,
693f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.mpeg4", "video_encoder.mpeg4" },
694f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_H263,
695f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.h263", "video_encoder.h263" },
696f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
697f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
698f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    static const size_t kNumMimeToRole =
699f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sizeof(kMimeToRole) / sizeof(kMimeToRole[0]);
700f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
701f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    size_t i;
702f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (i = 0; i < kNumMimeToRole; ++i) {
703f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (!strcasecmp(mime, kMimeToRole[i].mime)) {
704f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
705f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
706f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
707f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
708f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (i == kNumMimeToRole) {
709f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
710f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
711f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
712f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    const char *role =
713f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        isEncoder ? kMimeToRole[i].encoderRole
714f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                  : kMimeToRole[i].decoderRole;
715f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
716f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (role != NULL) {
717f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_PARAM_COMPONENTROLETYPE roleParams;
718f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        InitOMXParams(&roleParams);
719f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
720f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        strncpy((char *)roleParams.cRole,
721f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                role, OMX_MAX_STRINGNAME_SIZE - 1);
722f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
723f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        roleParams.cRole[OMX_MAX_STRINGNAME_SIZE - 1] = '\0';
724f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
725f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        status_t err = mOMX->setParameter(
726f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, OMX_IndexParamStandardComponentRole,
727f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &roleParams, sizeof(roleParams));
728f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
729f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
730f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            LOGW("[%s] Failed to set standard component role '%s'.",
731f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 mComponentName.c_str(), role);
732f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
733f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
734f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
735f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
736f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::configureCodec(
737f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime, const sp<AMessage> &msg) {
738f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    setComponentRole(false /* isEncoder */, mime);
739f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
740f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!strncasecmp(mime, "video/", 6)) {
741f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int32_t width, height;
742f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("width", &width));
743f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("height", &height));
744f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
745f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setupVideoDecoder(mime, width, height),
746f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
747f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_AAC)) {
748f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int32_t numChannels, sampleRate;
749f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("channel-count", &numChannels));
750f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("sample-rate", &sampleRate));
751f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
752f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setupAACDecoder(numChannels, sampleRate), (status_t)OK);
753f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_MPEG)) {
754f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
755f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        TRESPASS();
756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
757f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t maxInputSize;
759f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (msg->findInt32("max-input-size", &maxInputSize)) {
760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setMinBufferSize(kPortIndexInput, (size_t)maxInputSize),
761f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
762f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcmp("OMX.Nvidia.aac.decoder", mComponentName.c_str())) {
763f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setMinBufferSize(kPortIndexInput, 8192),  // XXX
764f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
765f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
766f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
767f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
768f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setMinBufferSize(OMX_U32 portIndex, size_t size) {
769f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
770f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
771f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
772f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
773f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
774f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
775f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
776f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
777f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
778f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
779f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
780f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (def.nBufferSize >= size) {
781f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return OK;
782f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
783f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
784f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nBufferSize = size;
785f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
786f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
787f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
788f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
789f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
790f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
791f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
792f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
793f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->getParameter(
794f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
795f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
796f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
797f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
798f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
799f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
800f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(def.nBufferSize >= size);
801f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
802f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
803f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
804f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
805f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setupAACDecoder(int32_t numChannels, int32_t sampleRate) {
806f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_AUDIO_PARAM_AACPROFILETYPE profile;
807f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&profile);
808f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nPortIndex = kPortIndexInput;
809f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
810f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
811f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
812f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
813f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
814f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
815f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
816f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
817f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nChannels = numChannels;
818f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nSampleRate = sampleRate;
819f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.eAACStreamFormat = OMX_AUDIO_AACStreamFormatMP4ADTS;
820f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
821f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
822f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
823f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
824f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
825f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
826f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
827f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setVideoPortFormatType(
828f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex,
829f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_VIDEO_CODINGTYPE compressionFormat,
830f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_COLOR_FORMATTYPE colorFormat) {
831f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PARAM_PORTFORMATTYPE format;
832f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&format);
833f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nPortIndex = portIndex;
834f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nIndex = 0;
835f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool found = false;
836f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
837f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 index = 0;
838f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (;;) {
839f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        format.nIndex = index;
840f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        status_t err = mOMX->getParameter(
841f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, OMX_IndexParamVideoPortFormat,
842f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &format, sizeof(format));
843f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
844f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
845f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return err;
846f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
847f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
848f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // The following assertion is violated by TI's video decoder.
849f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // CHECK_EQ(format.nIndex, index);
850f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
851f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (!strcmp("OMX.TI.Video.encoder", mComponentName.c_str())) {
852f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (portIndex == kPortIndexInput
853f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    && colorFormat == format.eColorFormat) {
854f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // eCompressionFormat does not seem right.
855f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                found = true;
856f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
857f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
858f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (portIndex == kPortIndexOutput
859f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    && compressionFormat == format.eCompressionFormat) {
860f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // eColorFormat does not seem right.
861f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                found = true;
862f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
863f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
864f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
865f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
866f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (format.eCompressionFormat == compressionFormat
867f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && format.eColorFormat == colorFormat) {
868f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            found = true;
869f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
870f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
871f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
872f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ++index;
873f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
874f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
875f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!found) {
876f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return UNKNOWN_ERROR;
877f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
878f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
879f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->setParameter(
880f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
881f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
882f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
883f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
884f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
885f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
886f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setSupportedOutputFormat() {
887f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PARAM_PORTFORMATTYPE format;
888f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&format);
889f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nPortIndex = kPortIndexOutput;
890f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nIndex = 0;
891f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
892f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
893f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
894f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
895f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(err, (status_t)OK);
896f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)format.eCompressionFormat, (int)OMX_VIDEO_CodingUnused);
897f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
898f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(format.eColorFormat == OMX_COLOR_FormatYUV420Planar
899f933441648ef6a71dee783d733aac17b9508b452Andreas Huber           || format.eColorFormat == OMX_COLOR_FormatYUV420SemiPlanar
900f933441648ef6a71dee783d733aac17b9508b452Andreas Huber           || format.eColorFormat == OMX_COLOR_FormatCbYCrY
9011065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber           || format.eColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar
902f933441648ef6a71dee783d733aac17b9508b452Andreas Huber           || format.eColorFormat == OMX_QCOM_COLOR_FormatYVU420SemiPlanar);
903f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
904f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return mOMX->setParameter(
905f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
906f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
907f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
908f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
909f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setupVideoDecoder(
910f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime, int32_t width, int32_t height) {
911f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_CODINGTYPE compressionFormat = OMX_VIDEO_CodingUnused;
912f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mime)) {
913f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        compressionFormat = OMX_VIDEO_CodingAVC;
914f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_MPEG4, mime)) {
915f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        compressionFormat = OMX_VIDEO_CodingMPEG4;
916f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_H263, mime)) {
917f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        compressionFormat = OMX_VIDEO_CodingH263;
918386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_MPEG2, mime)) {
919386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber        compressionFormat = OMX_VIDEO_CodingMPEG2;
920f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
921f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        TRESPASS();
922f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
923f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
924f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = setVideoPortFormatType(
925f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexInput, compressionFormat, OMX_COLOR_FormatUnused);
926f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
927f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
928f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
929f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
930f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
931f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setSupportedOutputFormat();
932f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
933f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
934f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
935f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
936f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
937f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setVideoFormatOnPort(
938f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexInput, width, height, compressionFormat);
939f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
940f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
941f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
942f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
943f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
944f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setVideoFormatOnPort(
945f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexOutput, width, height, OMX_VIDEO_CodingUnused);
946f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
947f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
948f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
949f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
950f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
951f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
952f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
953f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
954f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setVideoFormatOnPort(
955f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex,
956f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int32_t width, int32_t height, OMX_VIDEO_CODINGTYPE compressionFormat) {
957f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
958f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
959f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
960f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
961f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PORTDEFINITIONTYPE *video_def = &def.format.video;
962f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
963f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
964f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
965f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
966f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(err, (status_t)OK);
967f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
968f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexInput) {
969f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // XXX Need a (much) better heuristic to compute input buffer sizes.
970f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const size_t X = 64 * 1024;
971f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (def.nBufferSize < X) {
972f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.nBufferSize = X;
973f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
974f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
975f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
976f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainVideo);
977f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
978f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    video_def->nFrameWidth = width;
979f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    video_def->nFrameHeight = height;
980f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
981f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexInput) {
982f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        video_def->eCompressionFormat = compressionFormat;
983f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        video_def->eColorFormat = OMX_COLOR_FormatUnused;
984f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
985f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
986f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
987f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
988f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
989f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
990f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
991f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
992f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::initNativeWindow() {
993f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mNativeWindow != NULL) {
994f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_TRUE);
995f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
996f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
997f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_FALSE);
998f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
999f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1000f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1001f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::allYourBuffersAreBelongToUs(
1002f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
1003f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
1004f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
1005f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1006f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus != BufferInfo::OWNED_BY_US
1007f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                && info->mStatus != BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1008f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            LOGV("[%s] Buffer %p on port %ld still has status %d",
1009f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mComponentName.c_str(),
1010f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mBufferID, portIndex, info->mStatus);
1011f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
1012f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1013f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1014f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1015f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1016f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1017f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1018f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::allYourBuffersAreBelongToUs() {
1019f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return allYourBuffersAreBelongToUs(kPortIndexInput)
1020f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        && allYourBuffersAreBelongToUs(kPortIndexOutput);
1021f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1022f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1023f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::deferMessage(const sp<AMessage> &msg) {
1024f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool wasEmptyBefore = mDeferredQueue.empty();
1025f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDeferredQueue.push_back(msg);
1026f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1027f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1028f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::processDeferredMessages() {
1029f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    List<sp<AMessage> > queue = mDeferredQueue;
1030f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDeferredQueue.clear();
1031f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1032f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    List<sp<AMessage> >::iterator it = queue.begin();
1033f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    while (it != queue.end()) {
1034f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        onMessageReceived(*it++);
1035f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1036f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1037f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
103831e2508c75018145a8238925ff1a08cbde4e799aAndreas Hubervoid ACodec::sendFormatChange() {
103931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    sp<AMessage> notify = mNotify->dup();
104031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    notify->setInt32("what", kWhatOutputFormatChanged);
104131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
104231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
104331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    InitOMXParams(&def);
104431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    def.nPortIndex = kPortIndexOutput;
104531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
104631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    CHECK_EQ(mOMX->getParameter(
104731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def)),
104831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber             (status_t)OK);
104931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
105031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    CHECK_EQ((int)def.eDir, (int)OMX_DirOutput);
105131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
105231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    switch (def.eDomain) {
105331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        case OMX_PortDomainVideo:
105431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        {
105531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_VIDEO_PORTDEFINITIONTYPE *videoDef = &def.format.video;
105631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
105731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setString("mime", MEDIA_MIMETYPE_VIDEO_RAW);
105831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("width", videoDef->nFrameWidth);
105931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("height", videoDef->nFrameHeight);
106031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
106131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_CONFIG_RECTTYPE rect;
106231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            InitOMXParams(&rect);
106331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            rect.nPortIndex = kPortIndexOutput;
106431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
106531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            if (mOMX->getConfig(
106631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        mNode, OMX_IndexConfigCommonOutputCrop,
106731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        &rect, sizeof(rect)) != OK) {
106831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nLeft = 0;
106931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nTop = 0;
107031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nWidth = videoDef->nFrameWidth;
107131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nHeight = videoDef->nFrameHeight;
107231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            }
107331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
107431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nLeft, 0);
107531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nTop, 0);
107631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nWidth, 0u);
107731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nHeight, 0u);
107831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_LE(rect.nLeft + rect.nWidth - 1, videoDef->nFrameWidth);
107931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_LE(rect.nTop + rect.nHeight - 1, videoDef->nFrameHeight);
108031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
108131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setRect(
108231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    "crop",
108331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nLeft,
108431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nTop,
108531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nLeft + rect.nWidth - 1,
108631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nTop + rect.nHeight - 1);
108731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
108831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            if (mNativeWindow != NULL) {
108931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                android_native_rect_t crop;
109031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                crop.left = rect.nLeft;
109131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                crop.top = rect.nTop;
109289c120e7adbe09c6283591789594c5e591aa5032Andreas Huber                crop.right = rect.nLeft + rect.nWidth;
109389c120e7adbe09c6283591789594c5e591aa5032Andreas Huber                crop.bottom = rect.nTop + rect.nHeight;
109431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
109531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                CHECK_EQ(0, native_window_set_crop(
109631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                            mNativeWindow.get(), &crop));
109731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            }
109831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            break;
109931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        }
110031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
110131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        case OMX_PortDomainAudio:
110231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        {
110331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_AUDIO_PORTDEFINITIONTYPE *audioDef = &def.format.audio;
110431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ((int)audioDef->eEncoding, (int)OMX_AUDIO_CodingPCM);
110531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
110631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_AUDIO_PARAM_PCMMODETYPE params;
110731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            InitOMXParams(&params);
110831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            params.nPortIndex = kPortIndexOutput;
110931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
111031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ(mOMX->getParameter(
111131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        mNode, OMX_IndexParamAudioPcm,
111231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        &params, sizeof(params)),
111331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                     (status_t)OK);
111431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
111531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK(params.nChannels == 1 || params.bInterleaved);
111631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ(params.nBitPerSample, 16u);
111731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ((int)params.eNumData, (int)OMX_NumericalDataSigned);
111831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ((int)params.ePCMMode, (int)OMX_AUDIO_PCMModeLinear);
111931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
112031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setString("mime", MEDIA_MIMETYPE_AUDIO_RAW);
112131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("channel-count", params.nChannels);
112231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("sample-rate", params.nSamplingRate);
112331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            break;
112431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        }
112531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
112631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        default:
112731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            TRESPASS();
112831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    }
112931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
113031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    notify->post();
113131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
113231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    mSentFormat = true;
113331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber}
113431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
1135cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Hubervoid ACodec::signalError(OMX_ERRORTYPE error) {
1136cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber    sp<AMessage> notify = mNotify->dup();
1137cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber    notify->setInt32("what", ACodec::kWhatError);
1138cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber    notify->setInt32("omx-error", error);
1139cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber    notify->post();
1140cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber}
1141cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber
1142f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1143f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1144f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::BaseState(ACodec *codec, const sp<AState> &parentState)
1145f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : AState(parentState),
1146f933441648ef6a71dee783d733aac17b9508b452Andreas Huber      mCodec(codec) {
1147f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1148f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1149f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::BaseState::getPortMode(OMX_U32 portIndex) {
1150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return KEEP_BUFFERS;
1151f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1152f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1153f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onMessageReceived(const sp<AMessage> &msg) {
1154f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1155f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatInputBufferFilled:
1156f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1157f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onInputBufferFilled(msg);
1158f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1159f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1160f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1161f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatOutputBufferDrained:
1162f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1163f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onOutputBufferDrained(msg);
1164f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1165f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1166f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1167f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatOMXMessage:
1168f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1169f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXMessage(msg);
1170f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1171f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1172f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1173f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
1174f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1175f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1176f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1177f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1178f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1179f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXMessage(const sp<AMessage> &msg) {
1180f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t type;
1181f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findInt32("type", &type));
1182f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1183f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::node_id nodeID;
1184f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findPointer("node", &nodeID));
1185f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(nodeID, mCodec->mNode);
1186f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1187f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (type) {
1188f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::EVENT:
1189f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1190f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int32_t event, data1, data2;
1191f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("event", &event));
1192f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("data1", &data1));
1193f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("data2", &data2));
1194f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
11950af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber            if (event == OMX_EventCmdComplete
11960af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                    && data1 == OMX_CommandFlush
11970af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                    && data2 == (int32_t)OMX_ALL) {
11980af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // Use of this notification is not consistent across
11990af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // implementations. We'll drop this notification and rely
12000af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // on flush-complete notifications on the individual port
12010af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // indices instead.
12020af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
12030af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                return true;
12040af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber            }
12050af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
1206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXEvent(
1207f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_EVENTTYPE>(event),
1208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_U32>(data1),
1209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_U32>(data2));
1210f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1211f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1212f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::EMPTY_BUFFER_DONE:
1213f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1214f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID;
1215f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("buffer", &bufferID));
1216f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1217f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXEmptyBufferDone(bufferID);
1218f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1219f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1220f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::FILL_BUFFER_DONE:
1221f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1222f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID;
1223f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("buffer", &bufferID));
1224f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1225f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int32_t rangeOffset, rangeLength, flags;
1226f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int64_t timeUs;
1227f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            void *platformPrivate;
1228f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            void *dataPtr;
1229f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1230f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("range_offset", &rangeOffset));
1231f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("range_length", &rangeLength));
1232f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("flags", &flags));
1233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt64("timestamp", &timeUs));
1234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("platform_private", &platformPrivate));
1235f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("data_ptr", &dataPtr));
1236f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1237f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXFillBufferDone(
1238f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    bufferID,
1239f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    (size_t)rangeOffset, (size_t)rangeLength,
1240f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    (OMX_U32)flags,
1241f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    timeUs,
1242f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    platformPrivate,
1243f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    dataPtr);
1244f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1245f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1246f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1247f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();
1248f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1249f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1250f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1251f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1252f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXEvent(
1253f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1254f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (event != OMX_EventError) {
12555bc087c573c70c84c6a39946457590b42d392a33Andreas Huber        LOGV("[%s] EVENT(%d, 0x%08lx, 0x%08lx)",
1256f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             mCodec->mComponentName.c_str(), event, data1, data2);
1257f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1258f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return false;
1259f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1260f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1261c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    LOGE("[%s] ERROR(0x%08lx)", mCodec->mComponentName.c_str(), data1);
1262f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1263cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber    mCodec->signalError((OMX_ERRORTYPE)data1);
1264f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1265f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1266f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1267f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1268f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXEmptyBufferDone(IOMX::buffer_id bufferID) {
1269349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    LOGV("[%s] onOMXEmptyBufferDone %p",
1270349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber         mCodec->mComponentName.c_str(), bufferID);
1271349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1272f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info =
1273f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->findBufferByID(kPortIndexInput, bufferID);
1274f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1275f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_COMPONENT);
1276f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
1277f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1278f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexInput);
1279f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1280f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1281f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1282f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1283f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1284f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1285f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            postFillThisBuffer(info);
1286f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1287f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1288f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1289f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1290f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1291f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();  // Not currently used
1292f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1293f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1294f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1295f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1296f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1297f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1298f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1299f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::postFillThisBuffer(BufferInfo *info) {
1300f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mPortEOS[kPortIndexInput]) {
1301f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
1302f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1303f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1304f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
1305f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1306f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> notify = mCodec->mNotify->dup();
1307f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setInt32("what", ACodec::kWhatFillThisBuffer);
1308f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setPointer("buffer-id", info->mBufferID);
1309f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1310f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mData->meta()->clear();
1311f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setObject("buffer", info->mData);
1312f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1313f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> reply = new AMessage(kWhatInputBufferFilled, mCodec->id());
1314f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    reply->setPointer("buffer-id", info->mBufferID);
1315f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1316f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setMessage("reply", reply);
1317f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1318f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->post();
1319f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1320f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_UPSTREAM;
1321f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1322f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1323f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::onInputBufferFilled(const sp<AMessage> &msg) {
1324f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::buffer_id bufferID;
1325f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findPointer("buffer-id", &bufferID));
1326f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1327f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<RefBase> obj;
1328f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t err = OK;
1329f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!msg->findObject("buffer", &obj)) {
1330f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("err", &err));
1331f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
13323831a066bcf1019864a94d2bc7b4c9241efc5c22Andreas Huber        LOGV("[%s] saw error %d instead of an input buffer",
13333831a066bcf1019864a94d2bc7b4c9241efc5c22Andreas Huber             mCodec->mComponentName.c_str(), err);
13343831a066bcf1019864a94d2bc7b4c9241efc5c22Andreas Huber
1335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        obj.clear();
1336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1337f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1338f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<ABuffer> buffer = static_cast<ABuffer *>(obj.get());
1339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = mCodec->findBufferByID(kPortIndexInput, bufferID);
1341f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_UPSTREAM);
1342f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1343f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
1344f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1345f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexInput);
1346f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1347f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1348f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1349f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1350f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (buffer == NULL) {
1351dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                if (!mCodec->mPortEOS[kPortIndexInput]) {
1352dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                    mCodec->mPortEOS[kPortIndexInput] = true;
1353dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                    mCodec->mInputEOSResult = err;
1354dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                }
1355f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1356f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1358f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1359f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (buffer != NULL) {
1362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(!mCodec->mPortEOS[kPortIndexInput]);
1363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                int64_t timeUs;
1365f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(buffer->meta()->findInt64("timeUs", &timeUs));
1366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1367f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                OMX_U32 flags = OMX_BUFFERFLAG_ENDOFFRAME;
1368f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1369f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                int32_t isCSD;
1370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (buffer->meta()->findInt32("csd", &isCSD) && isCSD != 0) {
1371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    flags |= OMX_BUFFERFLAG_CODECCONFIG;
1372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1373f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1374f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (buffer != info->mData) {
1375349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                    if (0 && !(flags & OMX_BUFFERFLAG_CODECCONFIG)) {
1376f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        LOGV("[%s] Needs to copy input data.",
1377f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                             mCodec->mComponentName.c_str());
1378f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    }
1379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    CHECK_LE(buffer->size(), info->mData->capacity());
1381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    memcpy(info->mData->data(), buffer->data(), buffer->size());
1382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1384078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                if (flags & OMX_BUFFERFLAG_CODECCONFIG) {
1385078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                    LOGV("[%s] calling emptyBuffer %p w/ codec specific data",
1386078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                         mCodec->mComponentName.c_str(), bufferID);
1387078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                } else {
1388078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                    LOGV("[%s] calling emptyBuffer %p w/ time %lld us",
1389078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                         mCodec->mComponentName.c_str(), bufferID, timeUs);
1390078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                }
1391349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1392f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->emptyBuffer(
1393f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode,
1394f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            bufferID,
1395f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0,
1396f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            buffer->size(),
1397f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            flags,
1398f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            timeUs),
1399f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
1400f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1401f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1402f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1403f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                getMoreInputDataIfPossible();
1404f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else if (!mCodec->mPortEOS[kPortIndexInput]) {
1405dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                if (err != ERROR_END_OF_STREAM) {
1406dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                    LOGV("[%s] Signalling EOS on the input port "
1407dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                         "due to error %d",
1408dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                         mCodec->mComponentName.c_str(), err);
1409dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                } else {
1410dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                    LOGV("[%s] Signalling EOS on the input port",
1411dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                         mCodec->mComponentName.c_str());
1412dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                }
1413f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1414078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber                LOGV("[%s] calling emptyBuffer %p signalling EOS",
1415349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                     mCodec->mComponentName.c_str(), bufferID);
1416349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1417f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->emptyBuffer(
1418f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode,
1419f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            bufferID,
1420f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0,
1421f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0,
1422f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            OMX_BUFFERFLAG_EOS,
1423f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0),
1424f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
1425f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1426f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1427f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1428f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mPortEOS[kPortIndexInput] = true;
1429dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                mCodec->mInputEOSResult = err;
1430f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1431f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1432f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1433f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            default:
1434f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1435f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
1436f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1437f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1438f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1439f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1440f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::getMoreInputDataIfPossible() {
1441f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mPortEOS[kPortIndexInput]) {
1442f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
1443f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1444f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1445f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *eligible = NULL;
1446f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1447f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexInput].size(); ++i) {
1448f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mCodec->mBuffers[kPortIndexInput].editItemAt(i);
1449f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1450f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#if 0
1451f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus == BufferInfo::OWNED_BY_UPSTREAM) {
1452f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // There's already a "read" pending.
1453f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return;
1454f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1455f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#endif
1456f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1457f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus == BufferInfo::OWNED_BY_US) {
1458f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            eligible = info;
1459f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1460f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1461f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1462f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (eligible == NULL) {
1463f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
1464f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1465f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1466f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    postFillThisBuffer(eligible);
1467f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1468f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1469f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXFillBufferDone(
1470f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id bufferID,
1471f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        size_t rangeOffset, size_t rangeLength,
1472f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 flags,
1473f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int64_t timeUs,
1474f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        void *platformPrivate,
1475f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        void *dataPtr) {
1476078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber    LOGV("[%s] onOMXFillBufferDone %p time %lld us",
1477078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber         mCodec->mComponentName.c_str(), bufferID, timeUs);
1478349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1479f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ssize_t index;
1480f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info =
1481f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->findBufferByID(kPortIndexOutput, bufferID, &index);
1482f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1483f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_COMPONENT);
1484f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1485f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
1486f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1487f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexOutput);
1488f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1489f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1490f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1491f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1492f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1493f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1494f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1495f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (rangeLength == 0) {
1496f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (!(flags & OMX_BUFFERFLAG_EOS)) {
1497349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                    LOGV("[%s] calling fillBuffer %p",
1498349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                         mCodec->mComponentName.c_str(), info->mBufferID);
1499349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1500f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    CHECK_EQ(mCodec->mOMX->fillBuffer(
1501f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                                mCodec->mNode, info->mBufferID),
1502f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                             (status_t)OK);
1503f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1504f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1505f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1506f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else {
150731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                if (!mCodec->mSentFormat) {
150831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    mCodec->sendFormatChange();
150931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                }
151031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
1511f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (mCodec->mNativeWindow == NULL) {
1512f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mData->setRange(rangeOffset, rangeLength);
1513f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1514f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1515f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mData->meta()->setInt64("timeUs", timeUs);
1516f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1517f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                sp<AMessage> notify = mCodec->mNotify->dup();
1518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setInt32("what", ACodec::kWhatDrainThisBuffer);
1519f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setPointer("buffer-id", info->mBufferID);
1520f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setObject("buffer", info->mData);
1521f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1522f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                sp<AMessage> reply =
1523f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    new AMessage(kWhatOutputBufferDrained, mCodec->id());
1524f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1525f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                reply->setPointer("buffer-id", info->mBufferID);
1526f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1527f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setMessage("reply", reply);
1528f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1529f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->post();
1530f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1531f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_DOWNSTREAM;
1532f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1533f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1534f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (flags & OMX_BUFFERFLAG_EOS) {
1535f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                sp<AMessage> notify = mCodec->mNotify->dup();
1536f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setInt32("what", ACodec::kWhatEOS);
1537dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber                notify->setInt32("err", mCodec->mInputEOSResult);
1538f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->post();
1539f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1540f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mPortEOS[kPortIndexOutput] = true;
1541f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1542f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1543f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1544f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1545f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1546f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1547f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1548f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1549f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((status_t)OK,
1550f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->freeBuffer(kPortIndexOutput, index));
1551f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1552f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1553f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1554f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1555f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1556f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1557f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1558f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::onOutputBufferDrained(const sp<AMessage> &msg) {
1559f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::buffer_id bufferID;
1560f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findPointer("buffer-id", &bufferID));
1561f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1562f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ssize_t index;
1563f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info =
1564f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->findBufferByID(kPortIndexOutput, bufferID, &index);
1565f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_DOWNSTREAM);
1566f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1567f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t render;
1568f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mNativeWindow != NULL
1569f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && msg->findInt32("render", &render) && render != 0) {
1570f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // The client wants this buffer to be rendered.
1571f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1572cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber        if (mCodec->mNativeWindow->queueBuffer(
1573f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mCodec->mNativeWindow.get(),
1574cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber                    info->mGraphicBuffer.get()) == OK) {
1575cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber            info->mStatus = BufferInfo::OWNED_BY_NATIVE_WINDOW;
1576cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber        } else {
1577cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber            mCodec->signalError();
1578cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber            info->mStatus = BufferInfo::OWNED_BY_US;
1579cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber        }
1580f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
1581f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info->mStatus = BufferInfo::OWNED_BY_US;
1582f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1583f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1584f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexOutput);
1585f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1586f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1587f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1588f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1589f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // XXX fishy, revisit!!! What about the FREE_BUFFERS case below?
1590f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1591f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1592f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // We cannot resubmit the buffer we just rendered, dequeue
1593f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // the spare instead.
1594f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1595f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info = mCodec->dequeueBufferFromNativeWindow();
1596f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1597f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1598f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1599f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1600f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1601f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1602f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (!mCodec->mPortEOS[kPortIndexOutput]) {
1603f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1604f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    // We cannot resubmit the buffer we just rendered, dequeue
1605f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    // the spare instead.
1606f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1607f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info = mCodec->dequeueBufferFromNativeWindow();
1608f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1609f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1610c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                if (info != NULL) {
1611c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                    LOGV("[%s] calling fillBuffer %p",
1612c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                         mCodec->mComponentName.c_str(), info->mBufferID);
1613349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1614c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                    CHECK_EQ(mCodec->mOMX->fillBuffer(mCodec->mNode, info->mBufferID),
1615c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                             (status_t)OK);
1616f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1617c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                    info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1618c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                }
1619f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1620f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1621f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1622f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1623f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1624f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1625f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1626f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1627f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((status_t)OK,
1628f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->freeBuffer(kPortIndexOutput, index));
1629f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1630f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1631f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1632f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1633f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1634f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1635f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1636f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::UninitializedState::UninitializedState(ACodec *codec)
1637f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
1638f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1639f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1640f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::UninitializedState::onMessageReceived(const sp<AMessage> &msg) {
1641f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
1642f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1643f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1644f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatSetup:
1645f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1646f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onSetup(msg);
1647f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1648f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1649f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1650f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1651f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1652f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatShutdown:
1653f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1654f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
1655f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->setInt32("what", ACodec::kWhatShutdownCompleted);
1656f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
1657f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1658f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1659c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber            break;
1660f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1661f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1662f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatFlush:
1663f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
1665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->setInt32("what", ACodec::kWhatFlushCompleted);
1666f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
1667f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1668f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1669c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber            break;
1670f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1671f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1672f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1673f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
1674f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1675f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1676f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
1677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1678f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1679f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::UninitializedState::onSetup(
1680f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
1681f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMXClient client;
1682f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(client.connect(), (status_t)OK);
1683f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1684f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<IOMX> omx = client.interface();
1685f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1686f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    AString mime;
1687f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findString("mime", &mime));
1688f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
16891065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    Vector<String8> matchingCodecs;
16901065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    OMXCodec::findMatchingCodecs(
16911065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            mime.c_str(),
16921065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            false, // createEncoder
16931065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            NULL,  // matchComponentName
16941065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            0,     // flags
16951065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            &matchingCodecs);
16961065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
16971065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    sp<CodecObserver> observer = new CodecObserver;
16981065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    IOMX::node_id node = NULL;
16991065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
1700f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    AString componentName;
1701f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
17021065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    for (size_t matchIndex = 0; matchIndex < matchingCodecs.size();
17031065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            ++matchIndex) {
17041065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        componentName = matchingCodecs.itemAt(matchIndex).string();
1705f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1706078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber        pid_t tid = androidGetTid();
1707078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber        int prevPriority = androidGetThreadPriority(tid);
1708078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber        androidSetThreadPriority(tid, ANDROID_PRIORITY_FOREGROUND);
17091065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        status_t err = omx->allocateNode(componentName.c_str(), observer, &node);
1710078cfcf7cce9185ec7559910d08b0bc02bfc88a3Andreas Huber        androidSetThreadPriority(tid, prevPriority);
1711f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
17121065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        if (err == OK) {
17131065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            break;
17141065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        }
17151065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
17161065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        node = NULL;
17171065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    }
17181065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
1719c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    if (node == NULL) {
1720c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        LOGE("Unable to instantiate a decoder for type '%s'.", mime.c_str());
1721c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber
1722cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber        mCodec->signalError(OMX_ErrorComponentNotFound);
1723c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        return;
1724c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    }
1725f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1726f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> notify = new AMessage(kWhatOMXMessage, mCodec->id());
1727f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    observer->setNotificationMessage(notify);
1728f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1729f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mComponentName = componentName;
1730f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mOMX = omx;
1731f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mNode = node;
1732f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1733349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    mCodec->mPortEOS[kPortIndexInput] =
1734349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        mCodec->mPortEOS[kPortIndexOutput] = false;
1735349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1736dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber    mCodec->mInputEOSResult = OK;
1737dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber
1738f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->configureCodec(mime.c_str(), msg);
1739f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1740f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<RefBase> obj;
1741386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber    if (msg->findObject("native-window", &obj)
1742386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber            && strncmp("OMX.google.", componentName.c_str(), 11)) {
17431173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        sp<NativeWindowWrapper> nativeWindow(
17441173118eace0e9e347cb007f0da817cee87579edGlenn Kasten                static_cast<NativeWindowWrapper *>(obj.get()));
17451173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        CHECK(nativeWindow != NULL);
17461173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        mCodec->mNativeWindow = nativeWindow->getNativeWindow();
1747f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1748f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1749f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((status_t)OK, mCodec->initNativeWindow());
1750f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1751f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(omx->sendCommand(node, OMX_CommandStateSet, OMX_StateIdle),
1752f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             (status_t)OK);
1753f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1754f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->changeState(mCodec->mLoadedToIdleState);
1755f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1757f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1759f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::LoadedToIdleState::LoadedToIdleState(ACodec *codec)
1760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
1761f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1762f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1763f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::LoadedToIdleState::stateEntered() {
17645bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Loaded->Idle", mCodec->mComponentName.c_str());
1765f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1766cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    status_t err;
1767cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    if ((err = allocateBuffers()) != OK) {
1768cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber        LOGE("Failed to allocate buffers after transitioning to IDLE state "
1769cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber             "(error 0x%08x)",
1770cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber             err);
1771cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber
1772cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber        mCodec->signalError();
1773cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    }
1774f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1775f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1776f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::LoadedToIdleState::allocateBuffers() {
1777f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mCodec->allocateBuffersOnPort(kPortIndexInput);
1778f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1779f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
1780f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
1781f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1782f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1783f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return mCodec->allocateBuffersOnPort(kPortIndexOutput);
1784f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1785f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1786f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::LoadedToIdleState::onMessageReceived(const sp<AMessage> &msg) {
1787f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1788f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
1789f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1790f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
1791f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1792f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1793f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1794f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1795f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
1796f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1797f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1798f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1799f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::LoadedToIdleState::onOMXEvent(
1800f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1801f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
1802f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
1803f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1804f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
1805f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateIdle);
1806f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1807f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->sendCommand(
1808f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mNode, OMX_CommandStateSet, OMX_StateExecuting),
1809f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (status_t)OK);
1810f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1811f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mIdleToExecutingState);
1812f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1813f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1814f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1815f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1816f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1817f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
1818f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1819f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1820f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1821f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1822f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1823f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::IdleToExecutingState::IdleToExecutingState(ACodec *codec)
1824f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
1825f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1826f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1827f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::IdleToExecutingState::stateEntered() {
18285bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Idle->Executing", mCodec->mComponentName.c_str());
1829f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1830f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1831f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToExecutingState::onMessageReceived(const sp<AMessage> &msg) {
1832f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1833f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
1834f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1835f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
1836f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1837f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1838f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1839f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1840f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
1841f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1842f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1843f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1844f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToExecutingState::onOMXEvent(
1845f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1846f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
1847f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
1848f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1849f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
1850f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateExecuting);
1851f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1852f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mExecutingState->resume();
1853f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mExecutingState);
1854f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1855f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1856f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1857f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1858f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1859f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
1860f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1861f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1862f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1863f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1864f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1865f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ExecutingState::ExecutingState(ACodec *codec)
1866349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    : BaseState(codec),
1867349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber      mActive(false) {
1868f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1869f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1870f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::ExecutingState::getPortMode(
1871f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
1872f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return RESUBMIT_BUFFERS;
1873f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1874f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1875f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::submitOutputBuffers() {
1876f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexOutput].size(); ++i) {
1877f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mCodec->mBuffers[kPortIndexOutput].editItemAt(i);
1878f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1879f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (mCodec->mNativeWindow != NULL) {
1880f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(info->mStatus == BufferInfo::OWNED_BY_US
1881f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    || info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW);
1882f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1883f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1884f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                continue;
1885f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1886f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1887f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            status_t err = mCodec->mNativeWindow->lockBuffer(
1888f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mCodec->mNativeWindow.get(),
1889f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mGraphicBuffer.get());
1890f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(err, (status_t)OK);
1891f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        } else {
1892f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
1893f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1894f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1895349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        LOGV("[%s] calling fillBuffer %p",
1896349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber             mCodec->mComponentName.c_str(), info->mBufferID);
1897349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1898f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->mOMX->fillBuffer(mCodec->mNode, info->mBufferID),
1899f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
1900f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1901f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1902f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1903f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1904f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1905f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::resume() {
1906349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    if (mActive) {
1907349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        LOGV("[%s] We're already active, no need to resume.",
1908349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber             mCodec->mComponentName.c_str());
1909349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1910349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        return;
1911349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    }
1912349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1913f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    submitOutputBuffers();
1914f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1915f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Post the first input buffer.
1916f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_GT(mCodec->mBuffers[kPortIndexInput].size(), 0u);
1917f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = &mCodec->mBuffers[kPortIndexInput].editItemAt(0);
1918f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1919f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    postFillThisBuffer(info);
1920349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1921349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    mActive = true;
1922f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1923f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1924f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::stateEntered() {
19255bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Executing", mCodec->mComponentName.c_str());
1926f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1927f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->processDeferredMessages();
1928f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1929f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1930f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingState::onMessageReceived(const sp<AMessage> &msg) {
1931f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
1932f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1933f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1934f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
1935f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1936349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mActive = false;
1937349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1938f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->sendCommand(
1939f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mNode, OMX_CommandStateSet, OMX_StateIdle),
1940f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (status_t)OK);
1941f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1942f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mExecutingToIdleState);
1943f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1944f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1945f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1946f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1947f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1948f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
1949f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1950349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mActive = false;
1951349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1952f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->sendCommand(
1953f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mNode, OMX_CommandFlush, OMX_ALL),
1954f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (status_t)OK);
1955f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1956f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mFlushingState);
1957f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1958f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1959f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1960f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1961f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1962f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatResume:
1963f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1964f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            resume();
1965f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1966f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1967f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1968f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1969f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1970f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1971f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
1972f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1973f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1974f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1975f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
1976f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1977f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1978f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingState::onOMXEvent(
1979f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1980f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
1981f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventPortSettingsChanged:
1982f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1983f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)kPortIndexOutput);
1984f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
198531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            if (data2 == 0 || data2 == OMX_IndexParamPortDefinition) {
1986f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->sendCommand(
1987f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode,
1988f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            OMX_CommandPortDisable, kPortIndexOutput),
1989f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
1990f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1991349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                mCodec->freeOutputBuffersNotOwnedByComponent();
1992f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1993f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->changeState(mCodec->mOutputPortSettingsChangedState);
199431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            } else if (data2 == OMX_IndexConfigCommonOutputCrop) {
199531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mCodec->mSentFormat = false;
1996f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else {
1997f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                LOGV("[%s] OMX_EventPortSettingsChanged 0x%08lx",
1998f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->mComponentName.c_str(), data2);
1999f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
2000f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2001f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2002f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2003f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2004f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventBufferFlag:
2005f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2006f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2007f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2008f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2009f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2010f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
2011f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2012f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2013f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2014f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
2015f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2016f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::OutputPortSettingsChangedState::OutputPortSettingsChangedState(
2017f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ACodec *codec)
2018f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
2019f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2020f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2021f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::OutputPortSettingsChangedState::getPortMode(
2022f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
2023f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexOutput) {
2024f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return FREE_BUFFERS;
2025f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2026f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2027f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(portIndex, (OMX_U32)kPortIndexInput);
2028f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2029f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return RESUBMIT_BUFFERS;
2030f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2031f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2032f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::OutputPortSettingsChangedState::onMessageReceived(
2033f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
2034f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2035f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2036f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2037f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2038f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2039349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case kWhatResume:
2040f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2041349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            if (msg->what() == kWhatResume) {
2042349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                LOGV("[%s] Deferring resume", mCodec->mComponentName.c_str());
2043349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            }
2044349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2045f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
2046f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2047f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2048f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2049f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2050f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2051f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2052f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2053f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2054f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2055f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2056f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2057f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2058f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::OutputPortSettingsChangedState::stateEntered() {
20595bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now handling output port settings change",
2060f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mCodec->mComponentName.c_str());
2061f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2062f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2063f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::OutputPortSettingsChangedState::onOMXEvent(
2064f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2065f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2066f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2067f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2068f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (data1 == (OMX_U32)OMX_CommandPortDisable) {
2069f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(data2, (OMX_U32)kPortIndexOutput);
2070f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2071f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                LOGV("[%s] Output port now disabled.",
2072f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mComponentName.c_str());
2073f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2074f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(mCodec->mBuffers[kPortIndexOutput].isEmpty());
2075f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mDealer[kPortIndexOutput].clear();
2076f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2077f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->sendCommand(
2078f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode, OMX_CommandPortEnable, kPortIndexOutput),
2079f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
2080f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2081cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                status_t err;
2082cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                if ((err = mCodec->allocateBuffersOnPort(
2083cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                                kPortIndexOutput)) != OK) {
2084cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                    LOGE("Failed to allocate output port buffers after "
2085cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                         "port reconfiguration (error 0x%08x)",
2086cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                         err);
2087cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber
2088cc54fbaa69c0b69929467449d2c19192f15b5039Andreas Huber                    mCodec->signalError();
2089cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                }
2090f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2091f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                return true;
2092f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else if (data1 == (OMX_U32)OMX_CommandPortEnable) {
2093f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(data2, (OMX_U32)kPortIndexOutput);
2094f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
209531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mCodec->mSentFormat = false;
209631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
2097f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                LOGV("[%s] Output port now reenabled.",
2098f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mComponentName.c_str());
2099f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2100349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                if (mCodec->mExecutingState->active()) {
2101349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                    mCodec->mExecutingState->submitOutputBuffers();
2102349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                }
2103349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2104f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->changeState(mCodec->mExecutingState);
2105f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2106f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                return true;
2107f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
2108f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2109f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
2110f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2111f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2112f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2113f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
2114f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2115f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2116f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2117f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
2118f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2119f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ExecutingToIdleState::ExecutingToIdleState(ACodec *codec)
2120f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
2121f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2122f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2123f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingToIdleState::onMessageReceived(const sp<AMessage> &msg) {
2124f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2125f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2126f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2127f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2128f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2129f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // Don't send me a flush request if you previously wanted me
2130f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // to shutdown.
2131f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();
2132f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2133f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2134f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2135f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2136f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2137f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing that...
2138f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2139f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2140f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2141f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2142f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2143f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2144f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2145f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2146f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2147f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2148f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2149f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2151f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::stateEntered() {
21525bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Executing->Idle", mCodec->mComponentName.c_str());
215331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
215431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    mCodec->mSentFormat = false;
2155f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2156f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2157f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingToIdleState::onOMXEvent(
2158f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2159f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2160f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2161f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2162f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
2163f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateIdle);
2164f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2165f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            changeStateIfWeOwnAllBuffers();
2166f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2167f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2168f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2169f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2170349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventPortSettingsChanged:
2171349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventBufferFlag:
2172349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        {
2173349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            // We're shutting down and don't care about this anymore.
2174349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            return true;
2175349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        }
2176349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2177f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2178f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
2179f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2180f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
21810af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
2182f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::changeStateIfWeOwnAllBuffers() {
2183f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->allYourBuffersAreBelongToUs()) {
2184f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->mOMX->sendCommand(
2185f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mCodec->mNode, OMX_CommandStateSet, OMX_StateLoaded),
2186f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
2187f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2188f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->freeBuffersOnPort(kPortIndexInput), (status_t)OK);
2189f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->freeBuffersOnPort(kPortIndexOutput), (status_t)OK);
2190f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2191f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->changeState(mCodec->mIdleToLoadedState);
2192f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2193f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2194f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2195f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::onInputBufferFilled(
2196f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
2197f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onInputBufferFilled(msg);
2198f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2199f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2200f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2201f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2202f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::onOutputBufferDrained(
2203f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
2204f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onOutputBufferDrained(msg);
2205f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2207f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
2210f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2211f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::IdleToLoadedState::IdleToLoadedState(ACodec *codec)
2212f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
2213f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2214f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2215f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToLoadedState::onMessageReceived(const sp<AMessage> &msg) {
2216f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2217f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2218f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2219f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2220f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2221f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing that...
2222f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2223f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2224f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2225f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2226f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2227f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2228f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2229f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // Don't send me a flush request if you previously wanted me
2230f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // to shutdown.
2231f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();
2232f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2235f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2236f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2237f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2238f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2239f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2240f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2241f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2242f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2243f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::IdleToLoadedState::stateEntered() {
22445bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Idle->Loaded", mCodec->mComponentName.c_str());
2245f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2246f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2247f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToLoadedState::onOMXEvent(
2248f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2249f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2250f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2251f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2252f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
2253f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateLoaded);
2254f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
22555bc087c573c70c84c6a39946457590b42d392a33Andreas Huber            LOGV("[%s] Now Loaded", mCodec->mComponentName.c_str());
2256f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2257f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->freeNode(mCodec->mNode), (status_t)OK);
2258f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2259f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mNativeWindow.clear();
2260f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mNode = NULL;
2261f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mOMX.clear();
2262f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mComponentName.clear();
2263f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2264f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mUninitializedState);
2265f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2266f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
2267f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->setInt32("what", ACodec::kWhatShutdownCompleted);
2268f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
2269f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2270f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2271f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2272f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2273f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2274f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
2275f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2276f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2277f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2278f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
2279f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2280f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::FlushingState::FlushingState(ACodec *codec)
2281f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
2282f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2283f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2284f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::stateEntered() {
22855bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Flushing", mCodec->mComponentName.c_str());
2286f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2287f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mFlushComplete[kPortIndexInput] = mFlushComplete[kPortIndexOutput] = false;
2288f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2289f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2290f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::FlushingState::onMessageReceived(const sp<AMessage> &msg) {
2291f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2292f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2293f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2294f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2295f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2296f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
2297f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2298f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2299f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2300f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2301f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2302f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing this right now.
2303f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2304f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2305f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2306f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2307f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2308f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2309f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2310f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2311f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2312f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2313f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2314f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2315f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::FlushingState::onOMXEvent(
2316f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2317f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2318f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2319f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2320f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandFlush);
2321f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2322f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (data2 == kPortIndexInput || data2 == kPortIndexOutput) {
2323f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(!mFlushComplete[data2]);
2324f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mFlushComplete[data2] = true;
2325e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber
2326e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                if (mFlushComplete[kPortIndexInput]
2327e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                        && mFlushComplete[kPortIndexOutput]) {
2328e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                    changeStateIfWeOwnAllBuffers();
2329e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                }
2330f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else {
2331f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(data2, OMX_ALL);
2332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(mFlushComplete[kPortIndexInput]);
2333f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(mFlushComplete[kPortIndexOutput]);
2334f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                changeStateIfWeOwnAllBuffers();
2336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
2337f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2338f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2341349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventPortSettingsChanged:
2342349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        {
2343349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            sp<AMessage> msg = new AMessage(kWhatOMXMessage, mCodec->id());
2344349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("type", omx_message::EVENT);
2345349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setPointer("node", mCodec->mNode);
2346349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("event", event);
2347349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("data1", data1);
2348349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("data2", data2);
2349349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2350349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            LOGV("[%s] Deferring OMX_EventPortSettingsChanged",
2351349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                 mCodec->mComponentName.c_str());
2352349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2353349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mCodec->deferMessage(msg);
2354349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2355349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            return true;
2356349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        }
2357349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2358f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2359f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
2360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
2363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2365f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::onOutputBufferDrained(const sp<AMessage> &msg) {
2366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onOutputBufferDrained(msg);
2367f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2368f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2369f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2371f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::onInputBufferFilled(const sp<AMessage> &msg) {
2372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onInputBufferFilled(msg);
2373f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2374f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2375f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2376f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2377f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::changeStateIfWeOwnAllBuffers() {
2378f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mFlushComplete[kPortIndexInput]
2379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && mFlushComplete[kPortIndexOutput]
2380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && mCodec->allYourBuffersAreBelongToUs()) {
2381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<AMessage> notify = mCodec->mNotify->dup();
2382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        notify->setInt32("what", ACodec::kWhatFlushCompleted);
2383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        notify->post();
2384f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2385f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->mPortEOS[kPortIndexInput] =
2386f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mPortEOS[kPortIndexOutput] = false;
2387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2388dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber        mCodec->mInputEOSResult = OK;
2389dc9bacd838442a524585887e6ea6696836be2edaAndreas Huber
2390f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->changeState(mCodec->mExecutingState);
2391f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2392f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2393f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2394f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}  // namespace android
2395