ACodec.cpp revision cb67cd1b51ff9ab221e3124cf7e546515fef3c87
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;
326f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
327f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeState(mUninitializedState);
328f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
329f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
330f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::~ACodec() {
331f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
333f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::setNotificationMessage(const sp<AMessage> &msg) {
334f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mNotify = msg;
335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
337f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::initiateSetup(const sp<AMessage> &msg) {
338f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->setWhat(kWhatSetup);
339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->setTarget(id());
340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    msg->post();
341f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
342f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
343f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::signalFlush() {
344f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    (new AMessage(kWhatFlush, id()))->post();
345f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
346f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
347f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::signalResume() {
348f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    (new AMessage(kWhatResume, id()))->post();
349f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
350f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
351f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::initiateShutdown() {
352f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    (new AMessage(kWhatShutdown, id()))->post();
353f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
354f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
355f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::allocateBuffersOnPort(OMX_U32 portIndex) {
356f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(portIndex == kPortIndexInput || portIndex == kPortIndexOutput);
357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
358f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(mDealer[portIndex] == NULL);
359f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(mBuffers[portIndex].isEmpty());
360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mNativeWindow != NULL && portIndex == kPortIndexOutput) {
362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return allocateOutputBuffersFromNativeWindow();
363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
365f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
367f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
368f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
369f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
373f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
374f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
375f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
376f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LOGV("[%s] Allocating %lu buffers of size %lu on %s port",
377f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mComponentName.c_str(),
378f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.nBufferCountActual, def.nBufferSize,
379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            portIndex == kPortIndexInput ? "input" : "output");
380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    size_t totalSize = def.nBufferCountActual * def.nBufferSize;
382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDealer[portIndex] = new MemoryDealer(totalSize, "OMXCodec");
383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
384f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (OMX_U32 i = 0; i < def.nBufferCountActual; ++i) {
385f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<IMemory> mem = mDealer[portIndex]->allocate(def.nBufferSize);
386f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(mem.get() != NULL);
387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
388f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id buffer;
3891065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
3901065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        if (!strcasecmp(
3911065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                    mComponentName.c_str(), "OMX.TI.DUCATI1.VIDEO.DECODER")) {
3921065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            if (portIndex == kPortIndexInput && i == 0) {
3931065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                // Only log this warning once per allocation round.
3941065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
3951065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                LOGW("OMX.TI.DUCATI1.VIDEO.DECODER requires the use of "
3961065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                     "OMX_AllocateBuffer instead of the preferred "
3971065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                     "OMX_UseBuffer. Vendor must fix this.");
3981065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            }
3991065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
4001065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            err = mOMX->allocateBufferWithBackup(
4011065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber                    mNode, portIndex, mem, &buffer);
4021065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        } else {
4031065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            err = mOMX->useBuffer(mNode, portIndex, mem, &buffer);
4041065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        }
405f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
406f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
407f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return err;
408f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
409f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
410f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo info;
411f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info.mBufferID = buffer;
412f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info.mStatus = BufferInfo::OWNED_BY_US;
413f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info.mData = new ABuffer(mem->pointer(), def.nBufferSize);
414f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mBuffers[portIndex].push(info);
415f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
416f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
417f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
418f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
419f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
420f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::allocateOutputBuffersFromNativeWindow() {
421f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
422f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
423f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = kPortIndexOutput;
424f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
425f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
426f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
427f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
428f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
429f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
430f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
431f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
4329bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    err = native_window_set_scaling_mode(mNativeWindow.get(),
4339bc7af17974f448291a44912566ec7472a0d798bMathias Agopian            NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW);
4349bc7af17974f448291a44912566ec7472a0d798bMathias Agopian
4359bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    if (err != OK) {
4369bc7af17974f448291a44912566ec7472a0d798bMathias Agopian        return err;
4379bc7af17974f448291a44912566ec7472a0d798bMathias Agopian    }
4389bc7af17974f448291a44912566ec7472a0d798bMathias Agopian
439f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = native_window_set_buffers_geometry(
440f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNativeWindow.get(),
441f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.format.video.nFrameWidth,
442f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.format.video.nFrameHeight,
443f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.format.video.eColorFormat);
444f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
445f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
446f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGE("native_window_set_buffers_geometry failed: %s (%d)",
447f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                strerror(-err), -err);
448f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
449f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
450f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
451f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Set up the native window.
4523c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    OMX_U32 usage = 0;
4533c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    err = mOMX->getGraphicBufferUsage(mNode, kPortIndexOutput, &usage);
4543c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    if (err != 0) {
4553c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        LOGW("querying usage flags from OMX IL component failed: %d", err);
4563c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        // XXX: Currently this error is logged, but not fatal.
4573c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis        usage = 0;
4583c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis    }
4593c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis
460f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = native_window_set_usage(
461f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNativeWindow.get(),
4623c14b9745c4afc88cec247d9dd0b003e087cbb52Jamie Gennis            usage | GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_EXTERNAL_DISP);
463f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
464f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
465f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGE("native_window_set_usage failed: %s (%d)", strerror(-err), -err);
466f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
467f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
468f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
469258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    int minUndequeuedBufs = 0;
470258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    err = mNativeWindow->query(
471258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            mNativeWindow.get(), NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS,
472258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            &minUndequeuedBufs);
473258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
474258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    if (err != 0) {
475258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        LOGE("NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS query failed: %s (%d)",
476258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                strerror(-err), -err);
477258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        return err;
478258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
479258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
480258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // XXX: Is this the right logic to use?  It's not clear to me what the OMX
481258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // buffer counts refer to - how do they account for the renderer holding on
482258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    // to buffers?
483258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    if (def.nBufferCountActual < def.nBufferCountMin + minUndequeuedBufs) {
484258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        OMX_U32 newBufferCount = def.nBufferCountMin + minUndequeuedBufs;
485258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        def.nBufferCountActual = newBufferCount;
486258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        err = mOMX->setParameter(
487258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
488258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
489258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        if (err != OK) {
490258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            LOGE("[%s] setting nBufferCountActual to %lu failed: %d",
491258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis                    mComponentName.c_str(), newBufferCount, err);
492258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis            return err;
493258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        }
494258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis    }
495258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis
496f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = native_window_set_buffer_count(
497f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNativeWindow.get(), def.nBufferCountActual);
498f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
499f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
500f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGE("native_window_set_buffer_count failed: %s (%d)", strerror(-err),
501f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                -err);
502f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
503f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
504f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
505f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LOGV("[%s] Allocating %lu buffers from a native window of size %lu on "
506f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         "output port",
507f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mComponentName.c_str(), def.nBufferCountActual, def.nBufferSize);
508f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
509f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Dequeue buffers and send them to OMX
51074006804065941841883c4b46ee785070164023fJamie Gennis    for (OMX_U32 i = 0; i < def.nBufferCountActual; i++) {
5118ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev        ANativeWindowBuffer *buf;
512f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf);
513f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != 0) {
514f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            LOGE("dequeueBuffer failed: %s (%d)", strerror(-err), -err);
515f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
516f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
517f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(buf, false));
51974006804065941841883c4b46ee785070164023fJamie Gennis        BufferInfo info;
52074006804065941841883c4b46ee785070164023fJamie Gennis        info.mStatus = BufferInfo::OWNED_BY_US;
52174006804065941841883c4b46ee785070164023fJamie Gennis        info.mData = new ABuffer(0);
52274006804065941841883c4b46ee785070164023fJamie Gennis        info.mGraphicBuffer = graphicBuffer;
52374006804065941841883c4b46ee785070164023fJamie Gennis        mBuffers[kPortIndexOutput].push(info);
52474006804065941841883c4b46ee785070164023fJamie Gennis
525f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id bufferId;
526f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        err = mOMX->useGraphicBuffer(mNode, kPortIndexOutput, graphicBuffer,
527f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &bufferId);
528f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != 0) {
52974006804065941841883c4b46ee785070164023fJamie Gennis            LOGE("registering GraphicBuffer %lu with OMX IL component failed: "
53074006804065941841883c4b46ee785070164023fJamie Gennis                 "%d", i, err);
531f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
532f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
533f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
53474006804065941841883c4b46ee785070164023fJamie Gennis        mBuffers[kPortIndexOutput].editItemAt(i).mBufferID = bufferId;
53574006804065941841883c4b46ee785070164023fJamie Gennis
536f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        LOGV("[%s] Registered graphic buffer with ID %p (pointer = %p)",
537f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             mComponentName.c_str(),
538f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             bufferId, graphicBuffer.get());
539f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
540f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
541f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 cancelStart;
542f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 cancelEnd;
543f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
544f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != 0) {
545f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // If an error occurred while dequeuing we need to cancel any buffers
546f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // that were dequeued.
547f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        cancelStart = 0;
54874006804065941841883c4b46ee785070164023fJamie Gennis        cancelEnd = mBuffers[kPortIndexOutput].size();
549f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
550f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // Return the last two buffers to the native window.
551258d4e3aef7984574b0972a66871afc8a13d8e4eJamie Gennis        cancelStart = def.nBufferCountActual - minUndequeuedBufs;
552f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        cancelEnd = def.nBufferCountActual;
553f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
554f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
555f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (OMX_U32 i = cancelStart; i < cancelEnd; i++) {
556f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[kPortIndexOutput].editItemAt(i);
557f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        cancelBufferToNativeWindow(info);
558f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
559f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
560f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
561f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
562f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
563f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::cancelBufferToNativeWindow(BufferInfo *info) {
564f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
565f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
566f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    LOGV("[%s] Calling cancelBuffer on buffer %p",
567f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mComponentName.c_str(), info->mBufferID);
568f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
569f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int err = mNativeWindow->cancelBuffer(
570f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mNativeWindow.get(), info->mGraphicBuffer.get());
571f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
572f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(err, 0);
573f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
574f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_NATIVE_WINDOW;
575f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
576f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
577f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
578f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
579f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BufferInfo *ACodec::dequeueBufferFromNativeWindow() {
5808ce2364512f7c32c824f5ec5719688830ba72427Iliyan Malchev    ANativeWindowBuffer *buf;
581c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    if (mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf) != 0) {
582c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        LOGE("dequeueBuffer failed.");
583c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        return NULL;
584c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    }
585f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
586f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = mBuffers[kPortIndexOutput].size(); i-- > 0;) {
587f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info =
588f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &mBuffers[kPortIndexOutput].editItemAt(i);
589f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
590f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mGraphicBuffer->handle == buf->handle) {
591f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)info->mStatus,
592f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (int)BufferInfo::OWNED_BY_NATIVE_WINDOW);
593f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
594f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            info->mStatus = BufferInfo::OWNED_BY_US;
595f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
596f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return info;
597f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
598f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
599f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
600f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    TRESPASS();
601f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
602f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return NULL;
603f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
604f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
605f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::freeBuffersOnPort(OMX_U32 portIndex) {
606f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = mBuffers[portIndex].size(); i-- > 0;) {
607f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ((status_t)OK, freeBuffer(portIndex, i));
608f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
609f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
610f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDealer[portIndex].clear();
611f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
612f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
613f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
614f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
615349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huberstatus_t ACodec::freeOutputBuffersNotOwnedByComponent() {
616f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = mBuffers[kPortIndexOutput].size(); i-- > 0;) {
617f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info =
618f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &mBuffers[kPortIndexOutput].editItemAt(i);
619f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
620349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        if (info->mStatus !=
621349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                BufferInfo::OWNED_BY_COMPONENT) {
622349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            // We shouldn't have sent out any buffers to the client at this
623349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            // point.
624349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            CHECK_NE((int)info->mStatus, (int)BufferInfo::OWNED_BY_DOWNSTREAM);
625349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
626f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((status_t)OK, freeBuffer(kPortIndexOutput, i));
627f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
628f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
629f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
630f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
631f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
632f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
633f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::freeBuffer(OMX_U32 portIndex, size_t i) {
634f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
635f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
636f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(info->mStatus == BufferInfo::OWNED_BY_US
637f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            || info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW);
638f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
639f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexOutput && mNativeWindow != NULL
640f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && info->mStatus == BufferInfo::OWNED_BY_US) {
641f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ((status_t)OK, cancelBufferToNativeWindow(info));
642f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
643f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
644f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(mOMX->freeBuffer(
645f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, portIndex, info->mBufferID),
646f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             (status_t)OK);
647f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
648f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mBuffers[portIndex].removeAt(i);
649f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
650f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
651f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
652f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
653f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BufferInfo *ACodec::findBufferByID(
654f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        uint32_t portIndex, IOMX::buffer_id bufferID,
655f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ssize_t *index) {
656f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
657f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
658f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
659f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mBufferID == bufferID) {
660f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (index != NULL) {
661f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                *index = i;
662f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
663f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return info;
664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
666f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
667f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    TRESPASS();
668f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
669f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return NULL;
670f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
671f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
672f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::setComponentRole(
673f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        bool isEncoder, const char *mime) {
674f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    struct MimeToRole {
675f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime;
676f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *decoderRole;
677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *encoderRole;
678f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
679f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
680f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    static const MimeToRole kMimeToRole[] = {
681f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_MPEG,
682f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.mp3", "audio_encoder.mp3" },
683f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_NB,
684f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.amrnb", "audio_encoder.amrnb" },
685f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AMR_WB,
686f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.amrwb", "audio_encoder.amrwb" },
687f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_AUDIO_AAC,
688f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "audio_decoder.aac", "audio_encoder.aac" },
689f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_AVC,
690f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.avc", "video_encoder.avc" },
691f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_MPEG4,
692f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.mpeg4", "video_encoder.mpeg4" },
693f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        { MEDIA_MIMETYPE_VIDEO_H263,
694f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            "video_decoder.h263", "video_encoder.h263" },
695f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    };
696f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
697f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    static const size_t kNumMimeToRole =
698f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sizeof(kMimeToRole) / sizeof(kMimeToRole[0]);
699f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
700f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    size_t i;
701f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (i = 0; i < kNumMimeToRole; ++i) {
702f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (!strcasecmp(mime, kMimeToRole[i].mime)) {
703f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
704f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
705f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
706f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
707f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (i == kNumMimeToRole) {
708f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
709f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
710f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
711f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    const char *role =
712f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        isEncoder ? kMimeToRole[i].encoderRole
713f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                  : kMimeToRole[i].decoderRole;
714f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
715f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (role != NULL) {
716f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_PARAM_COMPONENTROLETYPE roleParams;
717f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        InitOMXParams(&roleParams);
718f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
719f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        strncpy((char *)roleParams.cRole,
720f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                role, OMX_MAX_STRINGNAME_SIZE - 1);
721f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
722f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        roleParams.cRole[OMX_MAX_STRINGNAME_SIZE - 1] = '\0';
723f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
724f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        status_t err = mOMX->setParameter(
725f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, OMX_IndexParamStandardComponentRole,
726f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &roleParams, sizeof(roleParams));
727f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
728f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
729f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            LOGW("[%s] Failed to set standard component role '%s'.",
730f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 mComponentName.c_str(), role);
731f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
732f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
733f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
734f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
735f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::configureCodec(
736f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime, const sp<AMessage> &msg) {
737f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    setComponentRole(false /* isEncoder */, mime);
738f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
739f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!strncasecmp(mime, "video/", 6)) {
740f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int32_t width, height;
741f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("width", &width));
742f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("height", &height));
743f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
744f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setupVideoDecoder(mime, width, height),
745f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
746f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_AAC)) {
747f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int32_t numChannels, sampleRate;
748f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("channel-count", &numChannels));
749f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("sample-rate", &sampleRate));
750f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
751f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setupAACDecoder(numChannels, sampleRate), (status_t)OK);
752f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_MPEG)) {
753f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
754f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        TRESPASS();
755f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
757f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t maxInputSize;
758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (msg->findInt32("max-input-size", &maxInputSize)) {
759f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setMinBufferSize(kPortIndexInput, (size_t)maxInputSize),
760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
761f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcmp("OMX.Nvidia.aac.decoder", mComponentName.c_str())) {
762f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(setMinBufferSize(kPortIndexInput, 8192),  // XXX
763f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
764f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
765f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
766f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
767f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setMinBufferSize(OMX_U32 portIndex, size_t size) {
768f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
769f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
770f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
771f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
772f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
773f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
774f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
775f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
776f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
777f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
778f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
779f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (def.nBufferSize >= size) {
780f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return OK;
781f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
782f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
783f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nBufferSize = size;
784f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
785f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
786f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
787f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
788f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
789f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
790f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
791f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
792f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->getParameter(
793f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
794f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
795f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
796f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
797f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
798f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
799f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(def.nBufferSize >= size);
800f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
801f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
802f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
803f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
804f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setupAACDecoder(int32_t numChannels, int32_t sampleRate) {
805f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_AUDIO_PARAM_AACPROFILETYPE profile;
806f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&profile);
807f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nPortIndex = kPortIndexInput;
808f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
809f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
810f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
811f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
812f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
813f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
814f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
815f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
816f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nChannels = numChannels;
817f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.nSampleRate = sampleRate;
818f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    profile.eAACStreamFormat = OMX_AUDIO_AACStreamFormatMP4ADTS;
819f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
820f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
821f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamAudioAac, &profile, sizeof(profile));
822f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
823f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
824f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
825f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
826f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setVideoPortFormatType(
827f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex,
828f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_VIDEO_CODINGTYPE compressionFormat,
829f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_COLOR_FORMATTYPE colorFormat) {
830f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PARAM_PORTFORMATTYPE format;
831f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&format);
832f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nPortIndex = portIndex;
833f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nIndex = 0;
834f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool found = false;
835f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
836f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_U32 index = 0;
837f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (;;) {
838f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        format.nIndex = index;
839f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        status_t err = mOMX->getParameter(
840f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mNode, OMX_IndexParamVideoPortFormat,
841f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                &format, sizeof(format));
842f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
843f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (err != OK) {
844f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return err;
845f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
846f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
847f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // The following assertion is violated by TI's video decoder.
848f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // CHECK_EQ(format.nIndex, index);
849f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
850f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (!strcmp("OMX.TI.Video.encoder", mComponentName.c_str())) {
851f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (portIndex == kPortIndexInput
852f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    && colorFormat == format.eColorFormat) {
853f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // eCompressionFormat does not seem right.
854f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                found = true;
855f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
856f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
857f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (portIndex == kPortIndexOutput
858f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    && compressionFormat == format.eCompressionFormat) {
859f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // eColorFormat does not seem right.
860f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                found = true;
861f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
862f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
863f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
864f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
865f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (format.eCompressionFormat == compressionFormat
866f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && format.eColorFormat == colorFormat) {
867f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            found = true;
868f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
869f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
870f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
871f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ++index;
872f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
873f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
874f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!found) {
875f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return UNKNOWN_ERROR;
876f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
877f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
878f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->setParameter(
879f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
880f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
881f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
882f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
883f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
884f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
885f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setSupportedOutputFormat() {
886f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PARAM_PORTFORMATTYPE format;
887f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&format);
888f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nPortIndex = kPortIndexOutput;
889f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    format.nIndex = 0;
890f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
891f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
892f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
893f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
894f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(err, (status_t)OK);
895f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)format.eCompressionFormat, (int)OMX_VIDEO_CodingUnused);
896f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
897f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(format.eColorFormat == OMX_COLOR_FormatYUV420Planar
898f933441648ef6a71dee783d733aac17b9508b452Andreas Huber           || format.eColorFormat == OMX_COLOR_FormatYUV420SemiPlanar
899f933441648ef6a71dee783d733aac17b9508b452Andreas Huber           || format.eColorFormat == OMX_COLOR_FormatCbYCrY
9001065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber           || format.eColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar
901f933441648ef6a71dee783d733aac17b9508b452Andreas Huber           || format.eColorFormat == OMX_QCOM_COLOR_FormatYVU420SemiPlanar);
902f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
903f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return mOMX->setParameter(
904f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamVideoPortFormat,
905f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            &format, sizeof(format));
906f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
907f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
908f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setupVideoDecoder(
909f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const char *mime, int32_t width, int32_t height) {
910f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_CODINGTYPE compressionFormat = OMX_VIDEO_CodingUnused;
911f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mime)) {
912f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        compressionFormat = OMX_VIDEO_CodingAVC;
913f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_MPEG4, mime)) {
914f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        compressionFormat = OMX_VIDEO_CodingMPEG4;
915f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_H263, mime)) {
916f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        compressionFormat = OMX_VIDEO_CodingH263;
917386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber    } else if (!strcasecmp(MEDIA_MIMETYPE_VIDEO_MPEG2, mime)) {
918386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber        compressionFormat = OMX_VIDEO_CodingMPEG2;
919f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
920f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        TRESPASS();
921f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
922f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
923f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = setVideoPortFormatType(
924f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexInput, compressionFormat, OMX_COLOR_FormatUnused);
925f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
926f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
927f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
928f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
929f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
930f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setSupportedOutputFormat();
931f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
932f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
933f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
934f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
935f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
936f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setVideoFormatOnPort(
937f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexInput, width, height, compressionFormat);
938f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
939f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
940f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
941f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
942f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
943f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = setVideoFormatOnPort(
944f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            kPortIndexOutput, width, height, OMX_VIDEO_CodingUnused);
945f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
946f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
947f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
948f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
949f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
950f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
951f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
952f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
953f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::setVideoFormatOnPort(
954f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex,
955f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int32_t width, int32_t height, OMX_VIDEO_CODINGTYPE compressionFormat) {
956f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
957f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    InitOMXParams(&def);
958f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    def.nPortIndex = portIndex;
959f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
960f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMX_VIDEO_PORTDEFINITIONTYPE *video_def = &def.format.video;
961f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
962f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mOMX->getParameter(
963f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
964f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
965f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(err, (status_t)OK);
966f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
967f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexInput) {
968f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // XXX Need a (much) better heuristic to compute input buffer sizes.
969f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const size_t X = 64 * 1024;
970f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (def.nBufferSize < X) {
971f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            def.nBufferSize = X;
972f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
973f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
974f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
975f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)def.eDomain, (int)OMX_PortDomainVideo);
976f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
977f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    video_def->nFrameWidth = width;
978f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    video_def->nFrameHeight = height;
979f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
980f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexInput) {
981f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        video_def->eCompressionFormat = compressionFormat;
982f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        video_def->eColorFormat = OMX_COLOR_FormatUnused;
983f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
984f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
985f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    err = mOMX->setParameter(
986f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mNode, OMX_IndexParamPortDefinition, &def, sizeof(def));
987f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
988f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return err;
989f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
990f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
991f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::initNativeWindow() {
992f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mNativeWindow != NULL) {
993f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_TRUE);
994f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
995f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
996f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mOMX->enableGraphicBuffers(mNode, kPortIndexOutput, OMX_FALSE);
997f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return OK;
998f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
999f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1000f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::allYourBuffersAreBelongToUs(
1001f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
1002f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mBuffers[portIndex].size(); ++i) {
1003f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mBuffers[portIndex].editItemAt(i);
1004f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1005f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus != BufferInfo::OWNED_BY_US
1006f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                && info->mStatus != BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1007f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            LOGV("[%s] Buffer %p on port %ld still has status %d",
1008f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mComponentName.c_str(),
1009f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mBufferID, portIndex, info->mStatus);
1010f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
1011f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1012f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1013f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1014f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1015f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1016f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1017f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::allYourBuffersAreBelongToUs() {
1018f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return allYourBuffersAreBelongToUs(kPortIndexInput)
1019f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        && allYourBuffersAreBelongToUs(kPortIndexOutput);
1020f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1021f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1022f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::deferMessage(const sp<AMessage> &msg) {
1023f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool wasEmptyBefore = mDeferredQueue.empty();
1024f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDeferredQueue.push_back(msg);
1025f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1026f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1027f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::processDeferredMessages() {
1028f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    List<sp<AMessage> > queue = mDeferredQueue;
1029f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mDeferredQueue.clear();
1030f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1031f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    List<sp<AMessage> >::iterator it = queue.begin();
1032f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    while (it != queue.end()) {
1033f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        onMessageReceived(*it++);
1034f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1035f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1036f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
103731e2508c75018145a8238925ff1a08cbde4e799aAndreas Hubervoid ACodec::sendFormatChange() {
103831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    sp<AMessage> notify = mNotify->dup();
103931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    notify->setInt32("what", kWhatOutputFormatChanged);
104031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
104131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    OMX_PARAM_PORTDEFINITIONTYPE def;
104231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    InitOMXParams(&def);
104331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    def.nPortIndex = kPortIndexOutput;
104431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
104531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    CHECK_EQ(mOMX->getParameter(
104631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mNode, OMX_IndexParamPortDefinition, &def, sizeof(def)),
104731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber             (status_t)OK);
104831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
104931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    CHECK_EQ((int)def.eDir, (int)OMX_DirOutput);
105031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
105131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    switch (def.eDomain) {
105231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        case OMX_PortDomainVideo:
105331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        {
105431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_VIDEO_PORTDEFINITIONTYPE *videoDef = &def.format.video;
105531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
105631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setString("mime", MEDIA_MIMETYPE_VIDEO_RAW);
105731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("width", videoDef->nFrameWidth);
105831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("height", videoDef->nFrameHeight);
105931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
106031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_CONFIG_RECTTYPE rect;
106131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            InitOMXParams(&rect);
106231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            rect.nPortIndex = kPortIndexOutput;
106331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
106431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            if (mOMX->getConfig(
106531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        mNode, OMX_IndexConfigCommonOutputCrop,
106631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        &rect, sizeof(rect)) != OK) {
106731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nLeft = 0;
106831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nTop = 0;
106931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nWidth = videoDef->nFrameWidth;
107031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                rect.nHeight = videoDef->nFrameHeight;
107131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            }
107231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
107331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nLeft, 0);
107431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nTop, 0);
107531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nWidth, 0u);
107631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_GE(rect.nHeight, 0u);
107731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_LE(rect.nLeft + rect.nWidth - 1, videoDef->nFrameWidth);
107831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_LE(rect.nTop + rect.nHeight - 1, videoDef->nFrameHeight);
107931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
108031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setRect(
108131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    "crop",
108231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nLeft,
108331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nTop,
108431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nLeft + rect.nWidth - 1,
108531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    rect.nTop + rect.nHeight - 1);
108631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
108731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            if (mNativeWindow != NULL) {
108831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                android_native_rect_t crop;
108931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                crop.left = rect.nLeft;
109031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                crop.top = rect.nTop;
109189c120e7adbe09c6283591789594c5e591aa5032Andreas Huber                crop.right = rect.nLeft + rect.nWidth;
109289c120e7adbe09c6283591789594c5e591aa5032Andreas Huber                crop.bottom = rect.nTop + rect.nHeight;
109331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
109431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                CHECK_EQ(0, native_window_set_crop(
109531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                            mNativeWindow.get(), &crop));
109631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            }
109731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            break;
109831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        }
109931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
110031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        case OMX_PortDomainAudio:
110131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        {
110231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_AUDIO_PORTDEFINITIONTYPE *audioDef = &def.format.audio;
110331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ((int)audioDef->eEncoding, (int)OMX_AUDIO_CodingPCM);
110431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
110531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            OMX_AUDIO_PARAM_PCMMODETYPE params;
110631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            InitOMXParams(&params);
110731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            params.nPortIndex = kPortIndexOutput;
110831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
110931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ(mOMX->getParameter(
111031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        mNode, OMX_IndexParamAudioPcm,
111131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                        &params, sizeof(params)),
111231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                     (status_t)OK);
111331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
111431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK(params.nChannels == 1 || params.bInterleaved);
111531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ(params.nBitPerSample, 16u);
111631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ((int)params.eNumData, (int)OMX_NumericalDataSigned);
111731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            CHECK_EQ((int)params.ePCMMode, (int)OMX_AUDIO_PCMModeLinear);
111831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
111931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setString("mime", MEDIA_MIMETYPE_AUDIO_RAW);
112031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("channel-count", params.nChannels);
112131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            notify->setInt32("sample-rate", params.nSamplingRate);
112231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            break;
112331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        }
112431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
112531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber        default:
112631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            TRESPASS();
112731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    }
112831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
112931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    notify->post();
113031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
113131e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    mSentFormat = true;
113231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber}
113331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
1134f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1135f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1136f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::BaseState(ACodec *codec, const sp<AState> &parentState)
1137f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : AState(parentState),
1138f933441648ef6a71dee783d733aac17b9508b452Andreas Huber      mCodec(codec) {
1139f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1140f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1141f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::BaseState::getPortMode(OMX_U32 portIndex) {
1142f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return KEEP_BUFFERS;
1143f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1144f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1145f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onMessageReceived(const sp<AMessage> &msg) {
1146f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1147f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatInputBufferFilled:
1148f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1149f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onInputBufferFilled(msg);
1150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1151f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1152f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1153f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatOutputBufferDrained:
1154f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1155f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onOutputBufferDrained(msg);
1156f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1157f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1158f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1159f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatOMXMessage:
1160f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1161f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXMessage(msg);
1162f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1163f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1164f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1165f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
1166f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1167f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1168f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1169f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1170f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1171f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXMessage(const sp<AMessage> &msg) {
1172f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t type;
1173f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findInt32("type", &type));
1174f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1175f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::node_id nodeID;
1176f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findPointer("node", &nodeID));
1177f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(nodeID, mCodec->mNode);
1178f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1179f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (type) {
1180f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::EVENT:
1181f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1182f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int32_t event, data1, data2;
1183f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("event", &event));
1184f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("data1", &data1));
1185f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("data2", &data2));
1186f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
11870af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber            if (event == OMX_EventCmdComplete
11880af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                    && data1 == OMX_CommandFlush
11890af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                    && data2 == (int32_t)OMX_ALL) {
11900af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // Use of this notification is not consistent across
11910af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // implementations. We'll drop this notification and rely
11920af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // on flush-complete notifications on the individual port
11930af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                // indices instead.
11940af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
11950af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber                return true;
11960af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber            }
11970af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
1198f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXEvent(
1199f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_EVENTTYPE>(event),
1200f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_U32>(data1),
1201f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    static_cast<OMX_U32>(data2));
1202f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1203f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1204f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::EMPTY_BUFFER_DONE:
1205f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID;
1207f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("buffer", &bufferID));
1208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXEmptyBufferDone(bufferID);
1210f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1211f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1212f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case omx_message::FILL_BUFFER_DONE:
1213f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1214f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            IOMX::buffer_id bufferID;
1215f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("buffer", &bufferID));
1216f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1217f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int32_t rangeOffset, rangeLength, flags;
1218f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            int64_t timeUs;
1219f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            void *platformPrivate;
1220f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            void *dataPtr;
1221f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1222f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("range_offset", &rangeOffset));
1223f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("range_length", &rangeLength));
1224f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt32("flags", &flags));
1225f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findInt64("timestamp", &timeUs));
1226f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("platform_private", &platformPrivate));
1227f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(msg->findPointer("data_ptr", &dataPtr));
1228f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1229f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return onOMXFillBufferDone(
1230f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    bufferID,
1231f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    (size_t)rangeOffset, (size_t)rangeLength,
1232f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    (OMX_U32)flags,
1233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    timeUs,
1234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    platformPrivate,
1235f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    dataPtr);
1236f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1237f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1238f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1239f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();
1240f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1241f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1242f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1243f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1244f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXEvent(
1245f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1246f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (event != OMX_EventError) {
12475bc087c573c70c84c6a39946457590b42d392a33Andreas Huber        LOGV("[%s] EVENT(%d, 0x%08lx, 0x%08lx)",
1248f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             mCodec->mComponentName.c_str(), event, data1, data2);
1249f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1250f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return false;
1251f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1252f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1253c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    LOGE("[%s] ERROR(0x%08lx)", mCodec->mComponentName.c_str(), data1);
1254f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1255c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    sp<AMessage> notify = mCodec->mNotify->dup();
1256c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    notify->setInt32("what", ACodec::kWhatError);
1257c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    notify->setInt32("omx-error", data1);
1258c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    notify->post();
1259f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1260f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1261f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1262f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1263f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXEmptyBufferDone(IOMX::buffer_id bufferID) {
1264349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    LOGV("[%s] onOMXEmptyBufferDone %p",
1265349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber         mCodec->mComponentName.c_str(), bufferID);
1266349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1267f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info =
1268f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->findBufferByID(kPortIndexInput, bufferID);
1269f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1270f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_COMPONENT);
1271f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
1272f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1273f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexInput);
1274f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1275f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1276f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1277f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1278f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1279f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1280f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            postFillThisBuffer(info);
1281f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1282f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1283f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1284f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1285f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1286f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();  // Not currently used
1287f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1288f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1289f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1290f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1291f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1292f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1293f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1294f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::postFillThisBuffer(BufferInfo *info) {
1295f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mPortEOS[kPortIndexInput]) {
1296f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
1297f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1298f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1299f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
1300f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1301f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> notify = mCodec->mNotify->dup();
1302f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setInt32("what", ACodec::kWhatFillThisBuffer);
1303f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setPointer("buffer-id", info->mBufferID);
1304f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1305f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mData->meta()->clear();
1306f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setObject("buffer", info->mData);
1307f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1308f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> reply = new AMessage(kWhatInputBufferFilled, mCodec->id());
1309f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    reply->setPointer("buffer-id", info->mBufferID);
1310f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1311f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->setMessage("reply", reply);
1312f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1313f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    notify->post();
1314f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1315f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_UPSTREAM;
1316f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1317f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1318f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::onInputBufferFilled(const sp<AMessage> &msg) {
1319f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::buffer_id bufferID;
1320f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findPointer("buffer-id", &bufferID));
1321f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1322f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<RefBase> obj;
1323f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t err = OK;
1324f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (!msg->findObject("buffer", &obj)) {
1325f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK(msg->findInt32("err", &err));
1326f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
13273831a066bcf1019864a94d2bc7b4c9241efc5c22Andreas Huber        LOGV("[%s] saw error %d instead of an input buffer",
13283831a066bcf1019864a94d2bc7b4c9241efc5c22Andreas Huber             mCodec->mComponentName.c_str(), err);
13293831a066bcf1019864a94d2bc7b4c9241efc5c22Andreas Huber
1330f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        obj.clear();
1331f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1332f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1333f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<ABuffer> buffer = static_cast<ABuffer *>(obj.get());
1334f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1335f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = mCodec->findBufferByID(kPortIndexInput, bufferID);
1336f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_UPSTREAM);
1337f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1338f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
1339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexInput);
1341f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1342f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1343f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1344f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1345f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (buffer == NULL) {
1346f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mPortEOS[kPortIndexInput] = true;
1347f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1348f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1349f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1350f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1351f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1352f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1353f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (buffer != NULL) {
1354f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(!mCodec->mPortEOS[kPortIndexInput]);
1355f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1356f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                int64_t timeUs;
1357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(buffer->meta()->findInt64("timeUs", &timeUs));
1358f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1359f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                OMX_U32 flags = OMX_BUFFERFLAG_ENDOFFRAME;
1360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                int32_t isCSD;
1362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (buffer->meta()->findInt32("csd", &isCSD) && isCSD != 0) {
1363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    flags |= OMX_BUFFERFLAG_CODECCONFIG;
1364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1365f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (buffer != info->mData) {
1367349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                    if (0 && !(flags & OMX_BUFFERFLAG_CODECCONFIG)) {
1368f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        LOGV("[%s] Needs to copy input data.",
1369f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                             mCodec->mComponentName.c_str());
1370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    }
1371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    CHECK_LE(buffer->size(), info->mData->capacity());
1373f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    memcpy(info->mData->data(), buffer->data(), buffer->size());
1374f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1375f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1376349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                LOGV("[%s] calling emptyBuffer %p",
1377349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                     mCodec->mComponentName.c_str(), bufferID);
1378349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1379f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->emptyBuffer(
1380f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode,
1381f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            bufferID,
1382f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0,
1383f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            buffer->size(),
1384f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            flags,
1385f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            timeUs),
1386f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
1387f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1388f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1389f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1390f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                getMoreInputDataIfPossible();
1391f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else if (!mCodec->mPortEOS[kPortIndexInput]) {
1392f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                LOGV("[%s] Signalling EOS on the input port",
1393f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->mComponentName.c_str());
1394f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1395349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                LOGV("[%s] calling emptyBuffer %p",
1396349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                     mCodec->mComponentName.c_str(), bufferID);
1397349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1398f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->emptyBuffer(
1399f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode,
1400f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            bufferID,
1401f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0,
1402f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0,
1403f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            OMX_BUFFERFLAG_EOS,
1404f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            0),
1405f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
1406f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1407f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1408f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1409f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mPortEOS[kPortIndexInput] = true;
1410f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1411f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1412f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1413f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            default:
1414f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1415f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                break;
1416f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1417f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1418f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1419f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1420f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::getMoreInputDataIfPossible() {
1421f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mPortEOS[kPortIndexInput]) {
1422f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
1423f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1424f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1425f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *eligible = NULL;
1426f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1427f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexInput].size(); ++i) {
1428f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mCodec->mBuffers[kPortIndexInput].editItemAt(i);
1429f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1430f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#if 0
1431f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus == BufferInfo::OWNED_BY_UPSTREAM) {
1432f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // There's already a "read" pending.
1433f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return;
1434f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1435f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#endif
1436f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1437f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (info->mStatus == BufferInfo::OWNED_BY_US) {
1438f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            eligible = info;
1439f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1440f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1441f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1442f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (eligible == NULL) {
1443f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return;
1444f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1445f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1446f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    postFillThisBuffer(eligible);
1447f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1448f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1449f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::BaseState::onOMXFillBufferDone(
1450f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        IOMX::buffer_id bufferID,
1451f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        size_t rangeOffset, size_t rangeLength,
1452f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 flags,
1453f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        int64_t timeUs,
1454f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        void *platformPrivate,
1455f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        void *dataPtr) {
1456349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    LOGV("[%s] onOMXFillBufferDone %p",
1457349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber         mCodec->mComponentName.c_str(), bufferID);
1458349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1459f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ssize_t index;
1460f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info =
1461f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->findBufferByID(kPortIndexOutput, bufferID, &index);
1462f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1463f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_COMPONENT);
1464f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1465f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    info->mStatus = BufferInfo::OWNED_BY_US;
1466f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1467f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexOutput);
1468f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1469f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1470f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1471f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1472f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1473f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1474f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1475f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (rangeLength == 0) {
1476f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (!(flags & OMX_BUFFERFLAG_EOS)) {
1477349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                    LOGV("[%s] calling fillBuffer %p",
1478349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                         mCodec->mComponentName.c_str(), info->mBufferID);
1479349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1480f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    CHECK_EQ(mCodec->mOMX->fillBuffer(
1481f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                                mCodec->mNode, info->mBufferID),
1482f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                             (status_t)OK);
1483f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1484f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1485f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1486f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else {
148731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                if (!mCodec->mSentFormat) {
148831e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                    mCodec->sendFormatChange();
148931e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                }
149031e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
1491f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (mCodec->mNativeWindow == NULL) {
1492f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mData->setRange(rangeOffset, rangeLength);
1493f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1494f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1495f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mData->meta()->setInt64("timeUs", timeUs);
1496f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1497f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                sp<AMessage> notify = mCodec->mNotify->dup();
1498f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setInt32("what", ACodec::kWhatDrainThisBuffer);
1499f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setPointer("buffer-id", info->mBufferID);
1500f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setObject("buffer", info->mData);
1501f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1502f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                sp<AMessage> reply =
1503f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    new AMessage(kWhatOutputBufferDrained, mCodec->id());
1504f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1505f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                reply->setPointer("buffer-id", info->mBufferID);
1506f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1507f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setMessage("reply", reply);
1508f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1509f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->post();
1510f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1511f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info->mStatus = BufferInfo::OWNED_BY_DOWNSTREAM;
1512f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1513f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1514f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (flags & OMX_BUFFERFLAG_EOS) {
1515f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                sp<AMessage> notify = mCodec->mNotify->dup();
1516f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->setInt32("what", ACodec::kWhatEOS);
1517f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                notify->post();
1518f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1519f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mPortEOS[kPortIndexOutput] = true;
1520f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1521f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1522f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1523f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1524f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1525f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1526f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1527f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1528f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((status_t)OK,
1529f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->freeBuffer(kPortIndexOutput, index));
1530f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1531f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1532f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1533f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1534f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
1535f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1536f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1537f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::BaseState::onOutputBufferDrained(const sp<AMessage> &msg) {
1538f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    IOMX::buffer_id bufferID;
1539f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findPointer("buffer-id", &bufferID));
1540f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1541f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    ssize_t index;
1542f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info =
1543f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->findBufferByID(kPortIndexOutput, bufferID, &index);
1544f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_DOWNSTREAM);
1545f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1546f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    int32_t render;
1547f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->mNativeWindow != NULL
1548f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && msg->findInt32("render", &render) && render != 0) {
1549f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        // The client wants this buffer to be rendered.
1550f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1551f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->mNativeWindow->queueBuffer(
1552f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mCodec->mNativeWindow.get(),
1553f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mGraphicBuffer.get()),
1554f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 0);
1555f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1556f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info->mStatus = BufferInfo::OWNED_BY_NATIVE_WINDOW;
1557f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    } else {
1558f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info->mStatus = BufferInfo::OWNED_BY_US;
1559f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1560f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1561f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    PortMode mode = getPortMode(kPortIndexOutput);
1562f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1563f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (mode) {
1564f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case KEEP_BUFFERS:
1565f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1566f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // XXX fishy, revisit!!! What about the FREE_BUFFERS case below?
1567f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1568f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1569f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // We cannot resubmit the buffer we just rendered, dequeue
1570f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                // the spare instead.
1571f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1572f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                info = mCodec->dequeueBufferFromNativeWindow();
1573f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1574f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1575f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1576f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1577f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case RESUBMIT_BUFFERS:
1578f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1579f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (!mCodec->mPortEOS[kPortIndexOutput]) {
1580f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1581f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    // We cannot resubmit the buffer we just rendered, dequeue
1582f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    // the spare instead.
1583f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1584f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info = mCodec->dequeueBufferFromNativeWindow();
1585f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                }
1586f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1587c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                if (info != NULL) {
1588c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                    LOGV("[%s] calling fillBuffer %p",
1589c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                         mCodec->mComponentName.c_str(), info->mBufferID);
1590349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1591c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                    CHECK_EQ(mCodec->mOMX->fillBuffer(mCodec->mNode, info->mBufferID),
1592c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                             (status_t)OK);
1593f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1594c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                    info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1595c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber                }
1596f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1597f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1598f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1599f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1600f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1601f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1602f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)mode, (int)FREE_BUFFERS);
1603f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1604f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((status_t)OK,
1605f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->freeBuffer(kPortIndexOutput, index));
1606f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1607f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1608f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1609f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1610f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1611f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1612f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1613f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::UninitializedState::UninitializedState(ACodec *codec)
1614f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
1615f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1616f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1617f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::UninitializedState::onMessageReceived(const sp<AMessage> &msg) {
1618f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
1619f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1620f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1621f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatSetup:
1622f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1623f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            onSetup(msg);
1624f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1625f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1626f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1627f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1628f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1629f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatShutdown:
1630f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1631f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
1632f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->setInt32("what", ACodec::kWhatShutdownCompleted);
1633f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
1634f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1635f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1636c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber            break;
1637f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1638f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1639f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case ACodec::kWhatFlush:
1640f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1641f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
1642f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->setInt32("what", ACodec::kWhatFlushCompleted);
1643f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
1644f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1645f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1646c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber            break;
1647f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1648f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1649f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1650f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
1651f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1652f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1653f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
1654f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1655f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1656f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::UninitializedState::onSetup(
1657f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
1658f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    OMXClient client;
1659f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(client.connect(), (status_t)OK);
1660f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1661f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<IOMX> omx = client.interface();
1662f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1663f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    AString mime;
1664f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK(msg->findString("mime", &mime));
1665f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
16661065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    Vector<String8> matchingCodecs;
16671065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    OMXCodec::findMatchingCodecs(
16681065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            mime.c_str(),
16691065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            false, // createEncoder
16701065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            NULL,  // matchComponentName
16711065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            0,     // flags
16721065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            &matchingCodecs);
16731065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
16741065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    sp<CodecObserver> observer = new CodecObserver;
16751065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    IOMX::node_id node = NULL;
16761065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
1677f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    AString componentName;
1678f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
16791065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    for (size_t matchIndex = 0; matchIndex < matchingCodecs.size();
16801065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            ++matchIndex) {
16811065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        componentName = matchingCodecs.itemAt(matchIndex).string();
1682f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
16831065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        status_t err = omx->allocateNode(componentName.c_str(), observer, &node);
1684f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
16851065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        if (err == OK) {
16861065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber            break;
16871065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        }
16881065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
16891065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber        node = NULL;
16901065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber    }
16911065b3f17d3048948e7d522049d1980b90df3dc1Andreas Huber
1692c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    if (node == NULL) {
1693c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        LOGE("Unable to instantiate a decoder for type '%s'.", mime.c_str());
1694c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber
1695c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        sp<AMessage> notify = mCodec->mNotify->dup();
1696c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        notify->setInt32("what", ACodec::kWhatError);
1697c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        notify->setInt32("omx-error", OMX_ErrorComponentNotFound);
1698c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        notify->post();
1699c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber
1700c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber        return;
1701c92fd24c10a6bf80b346d7e261325434d9c6964bAndreas Huber    }
1702f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1703f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<AMessage> notify = new AMessage(kWhatOMXMessage, mCodec->id());
1704f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    observer->setNotificationMessage(notify);
1705f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1706f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mComponentName = componentName;
1707f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mOMX = omx;
1708f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->mNode = node;
1709f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1710349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    mCodec->mPortEOS[kPortIndexInput] =
1711349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        mCodec->mPortEOS[kPortIndexOutput] = false;
1712349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1713f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->configureCodec(mime.c_str(), msg);
1714f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1715f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    sp<RefBase> obj;
1716386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber    if (msg->findObject("native-window", &obj)
1717386d609dc513e838c7e7c4c46c604493ccd560beAndreas Huber            && strncmp("OMX.google.", componentName.c_str(), 11)) {
17181173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        sp<NativeWindowWrapper> nativeWindow(
17191173118eace0e9e347cb007f0da817cee87579edGlenn Kasten                static_cast<NativeWindowWrapper *>(obj.get()));
17201173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        CHECK(nativeWindow != NULL);
17211173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        mCodec->mNativeWindow = nativeWindow->getNativeWindow();
1722f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1723f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1724f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ((status_t)OK, mCodec->initNativeWindow());
1725f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1726f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(omx->sendCommand(node, OMX_CommandStateSet, OMX_StateIdle),
1727f933441648ef6a71dee783d733aac17b9508b452Andreas Huber             (status_t)OK);
1728f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1729f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->changeState(mCodec->mLoadedToIdleState);
1730f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1731f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1732f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1733f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1734f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::LoadedToIdleState::LoadedToIdleState(ACodec *codec)
1735f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
1736f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1737f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1738f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::LoadedToIdleState::stateEntered() {
17395bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Loaded->Idle", mCodec->mComponentName.c_str());
1740f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1741cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    status_t err;
1742cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    if ((err = allocateBuffers()) != OK) {
1743cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber        LOGE("Failed to allocate buffers after transitioning to IDLE state "
1744cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber             "(error 0x%08x)",
1745cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber             err);
1746cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber
1747cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber        sp<AMessage> notify = mCodec->mNotify->dup();
1748cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber        notify->setInt32("what", ACodec::kWhatError);
1749cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber        notify->setInt32("omx-error", OMX_ErrorUndefined);
1750cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber        notify->post();
1751cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber    }
1752f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1753f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1754f933441648ef6a71dee783d733aac17b9508b452Andreas Huberstatus_t ACodec::LoadedToIdleState::allocateBuffers() {
1755f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    status_t err = mCodec->allocateBuffersOnPort(kPortIndexInput);
1756f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1757f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (err != OK) {
1758f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return err;
1759f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1760f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1761f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return mCodec->allocateBuffersOnPort(kPortIndexOutput);
1762f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1763f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1764f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::LoadedToIdleState::onMessageReceived(const sp<AMessage> &msg) {
1765f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1766f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
1767f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1768f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
1769f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1770f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1771f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1772f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1773f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
1774f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1775f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1776f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1777f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::LoadedToIdleState::onOMXEvent(
1778f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1779f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
1780f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
1781f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1782f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
1783f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateIdle);
1784f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1785f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->sendCommand(
1786f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mNode, OMX_CommandStateSet, OMX_StateExecuting),
1787f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (status_t)OK);
1788f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1789f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mIdleToExecutingState);
1790f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1791f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1792f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1793f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1794f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1795f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
1796f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1797f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1798f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1799f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1800f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1801f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::IdleToExecutingState::IdleToExecutingState(ACodec *codec)
1802f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
1803f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1804f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1805f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::IdleToExecutingState::stateEntered() {
18065bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Idle->Executing", mCodec->mComponentName.c_str());
1807f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1808f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1809f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToExecutingState::onMessageReceived(const sp<AMessage> &msg) {
1810f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1811f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
1812f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1813f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
1814f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1815f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1816f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1817f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1818f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onMessageReceived(msg);
1819f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1820f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1821f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1822f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToExecutingState::onOMXEvent(
1823f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1824f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
1825f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
1826f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1827f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
1828f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateExecuting);
1829f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1830f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mExecutingState->resume();
1831f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mExecutingState);
1832f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1833f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1834f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1835f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1836f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1837f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
1838f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1839f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1840f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1841f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1842f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1843f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ExecutingState::ExecutingState(ACodec *codec)
1844349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    : BaseState(codec),
1845349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber      mActive(false) {
1846f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1847f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1848f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::ExecutingState::getPortMode(
1849f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
1850f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return RESUBMIT_BUFFERS;
1851f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1852f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1853f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::submitOutputBuffers() {
1854f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    for (size_t i = 0; i < mCodec->mBuffers[kPortIndexOutput].size(); ++i) {
1855f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        BufferInfo *info = &mCodec->mBuffers[kPortIndexOutput].editItemAt(i);
1856f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1857f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        if (mCodec->mNativeWindow != NULL) {
1858f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK(info->mStatus == BufferInfo::OWNED_BY_US
1859f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    || info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW);
1860f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1861f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (info->mStatus == BufferInfo::OWNED_BY_NATIVE_WINDOW) {
1862f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                continue;
1863f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1864f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1865f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            status_t err = mCodec->mNativeWindow->lockBuffer(
1866f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mCodec->mNativeWindow.get(),
1867f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    info->mGraphicBuffer.get());
1868f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(err, (status_t)OK);
1869f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        } else {
1870f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ((int)info->mStatus, (int)BufferInfo::OWNED_BY_US);
1871f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1872f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1873349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        LOGV("[%s] calling fillBuffer %p",
1874349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber             mCodec->mComponentName.c_str(), info->mBufferID);
1875349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1876f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->mOMX->fillBuffer(mCodec->mNode, info->mBufferID),
1877f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
1878f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1879f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        info->mStatus = BufferInfo::OWNED_BY_COMPONENT;
1880f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1881f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1882f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1883f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::resume() {
1884349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    if (mActive) {
1885349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        LOGV("[%s] We're already active, no need to resume.",
1886349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber             mCodec->mComponentName.c_str());
1887349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1888349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        return;
1889349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    }
1890349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1891f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    submitOutputBuffers();
1892f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1893f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    // Post the first input buffer.
1894f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_GT(mCodec->mBuffers[kPortIndexInput].size(), 0u);
1895f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BufferInfo *info = &mCodec->mBuffers[kPortIndexInput].editItemAt(0);
1896f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1897f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    postFillThisBuffer(info);
1898349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1899349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber    mActive = true;
1900f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1901f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1902f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingState::stateEntered() {
19035bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Executing", mCodec->mComponentName.c_str());
1904f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1905f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mCodec->processDeferredMessages();
1906f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1907f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1908f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingState::onMessageReceived(const sp<AMessage> &msg) {
1909f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
1910f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1911f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
1912f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
1913f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1914349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mActive = false;
1915349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1916f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->sendCommand(
1917f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mNode, OMX_CommandStateSet, OMX_StateIdle),
1918f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (status_t)OK);
1919f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1920f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mExecutingToIdleState);
1921f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1922f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1923f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1924f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1925f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1926f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
1927f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1928349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mActive = false;
1929349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
1930f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->sendCommand(
1931f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mNode, OMX_CommandFlush, OMX_ALL),
1932f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     (status_t)OK);
1933f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1934f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mFlushingState);
1935f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1936f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1937f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1938f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1939f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1940f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatResume:
1941f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1942f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            resume();
1943f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1944f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
1945f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1946f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1947f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1948f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1949f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
1950f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
1951f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1952f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1953f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
1954f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1955f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1956f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingState::onOMXEvent(
1957f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
1958f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
1959f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventPortSettingsChanged:
1960f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1961f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)kPortIndexOutput);
1962f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
196331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            if (data2 == 0 || data2 == OMX_IndexParamPortDefinition) {
1964f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->sendCommand(
1965f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode,
1966f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            OMX_CommandPortDisable, kPortIndexOutput),
1967f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
1968f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1969349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                mCodec->freeOutputBuffersNotOwnedByComponent();
1970f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1971f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->changeState(mCodec->mOutputPortSettingsChangedState);
197231e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber            } else if (data2 == OMX_IndexConfigCommonOutputCrop) {
197331e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mCodec->mSentFormat = false;
1974f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else {
1975f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                LOGV("[%s] OMX_EventPortSettingsChanged 0x%08lx",
1976f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                     mCodec->mComponentName.c_str(), data2);
1977f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
1978f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1979f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1980f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1981f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1982f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventBufferFlag:
1983f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
1984f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
1985f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
1986f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1987f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
1988f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
1989f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
1990f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1991f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1992f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
1993f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1994f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::OutputPortSettingsChangedState::OutputPortSettingsChangedState(
1995f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        ACodec *codec)
1996f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
1997f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
1998f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
1999f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::BaseState::PortMode ACodec::OutputPortSettingsChangedState::getPortMode(
2000f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_U32 portIndex) {
2001f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (portIndex == kPortIndexOutput) {
2002f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        return FREE_BUFFERS;
2003f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2004f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2005f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    CHECK_EQ(portIndex, (OMX_U32)kPortIndexInput);
2006f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2007f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return RESUBMIT_BUFFERS;
2008f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2009f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2010f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::OutputPortSettingsChangedState::onMessageReceived(
2011f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
2012f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2013f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2014f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2015f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2016f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2017349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case kWhatResume:
2018f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2019349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            if (msg->what() == kWhatResume) {
2020349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                LOGV("[%s] Deferring resume", mCodec->mComponentName.c_str());
2021349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            }
2022349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2023f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
2024f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2025f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2026f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2027f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2028f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2029f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2030f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2031f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2032f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2033f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2034f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2035f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2036f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::OutputPortSettingsChangedState::stateEntered() {
20375bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now handling output port settings change",
2038f933441648ef6a71dee783d733aac17b9508b452Andreas Huber         mCodec->mComponentName.c_str());
2039f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2040f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2041f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::OutputPortSettingsChangedState::onOMXEvent(
2042f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2043f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2044f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2045f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2046f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (data1 == (OMX_U32)OMX_CommandPortDisable) {
2047f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(data2, (OMX_U32)kPortIndexOutput);
2048f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2049f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                LOGV("[%s] Output port now disabled.",
2050f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mComponentName.c_str());
2051f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2052f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(mCodec->mBuffers[kPortIndexOutput].isEmpty());
2053f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->mDealer[kPortIndexOutput].clear();
2054f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2055f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(mCodec->mOMX->sendCommand(
2056f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                            mCodec->mNode, OMX_CommandPortEnable, kPortIndexOutput),
2057f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                         (status_t)OK);
2058f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2059cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                status_t err;
2060cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                if ((err = mCodec->allocateBuffersOnPort(
2061cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                                kPortIndexOutput)) != OK) {
2062cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                    LOGE("Failed to allocate output port buffers after "
2063cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                         "port reconfiguration (error 0x%08x)",
2064cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                         err);
2065cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber
2066cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                    sp<AMessage> notify = mCodec->mNotify->dup();
2067cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                    notify->setInt32("what", ACodec::kWhatError);
2068cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                    notify->setInt32("omx-error", OMX_ErrorUndefined);
2069cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                    notify->post();
2070cb67cd1b51ff9ab221e3124cf7e546515fef3c87Andreas Huber                }
2071f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2072f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                return true;
2073f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else if (data1 == (OMX_U32)OMX_CommandPortEnable) {
2074f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(data2, (OMX_U32)kPortIndexOutput);
2075f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
207631e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber                mCodec->mSentFormat = false;
207731e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
2078f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                LOGV("[%s] Output port now reenabled.",
2079f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                        mCodec->mComponentName.c_str());
2080f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2081349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                if (mCodec->mExecutingState->active()) {
2082349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                    mCodec->mExecutingState->submitOutputBuffers();
2083349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                }
2084349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2085f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mCodec->changeState(mCodec->mExecutingState);
2086f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2087f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                return true;
2088f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
2089f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2090f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
2091f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2092f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2093f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2094f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return false;
2095f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2096f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2097f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2098f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
2099f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2100f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::ExecutingToIdleState::ExecutingToIdleState(ACodec *codec)
2101f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
2102f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2103f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2104f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingToIdleState::onMessageReceived(const sp<AMessage> &msg) {
2105f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2106f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2107f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2108f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2109f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2110f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // Don't send me a flush request if you previously wanted me
2111f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // to shutdown.
2112f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();
2113f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2114f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2115f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2116f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2117f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2118f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing that...
2119f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2120f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2121f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2122f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2123f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2124f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2125f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2126f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2127f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2128f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2129f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2130f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2131f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2132f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::stateEntered() {
21335bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Executing->Idle", mCodec->mComponentName.c_str());
213431e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber
213531e2508c75018145a8238925ff1a08cbde4e799aAndreas Huber    mCodec->mSentFormat = false;
2136f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2137f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2138f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::ExecutingToIdleState::onOMXEvent(
2139f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2140f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2141f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2142f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2143f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
2144f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateIdle);
2145f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2146f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            changeStateIfWeOwnAllBuffers();
2147f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2148f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2149f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2150f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2151349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventPortSettingsChanged:
2152349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventBufferFlag:
2153349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        {
2154349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            // We're shutting down and don't care about this anymore.
2155349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            return true;
2156349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        }
2157349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2158f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2159f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
2160f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2161f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
21620af941927ef8f35f2da5380dfd8d04b7f4ed3532Andreas Huber
2163f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::changeStateIfWeOwnAllBuffers() {
2164f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mCodec->allYourBuffersAreBelongToUs()) {
2165f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->mOMX->sendCommand(
2166f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                    mCodec->mNode, OMX_CommandStateSet, OMX_StateLoaded),
2167f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                 (status_t)OK);
2168f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2169f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->freeBuffersOnPort(kPortIndexInput), (status_t)OK);
2170f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        CHECK_EQ(mCodec->freeBuffersOnPort(kPortIndexOutput), (status_t)OK);
2171f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2172f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->changeState(mCodec->mIdleToLoadedState);
2173f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2174f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2175f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2176f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::onInputBufferFilled(
2177f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
2178f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onInputBufferFilled(msg);
2179f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2180f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2181f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2182f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2183f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::ExecutingToIdleState::onOutputBufferDrained(
2184f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        const sp<AMessage> &msg) {
2185f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onOutputBufferDrained(msg);
2186f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2187f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2188f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2189f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2190f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
2191f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2192f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::IdleToLoadedState::IdleToLoadedState(ACodec *codec)
2193f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
2194f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2195f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2196f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToLoadedState::onMessageReceived(const sp<AMessage> &msg) {
2197f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2198f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2199f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2200f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2201f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2202f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing that...
2203f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2204f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2205f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2206f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2207f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2208f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2209f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2210f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // Don't send me a flush request if you previously wanted me
2211f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // to shutdown.
2212f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            TRESPASS();
2213f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2214f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2215f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2216f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2217f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2218f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2219f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2220f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2221f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2222f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2223f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2224f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::IdleToLoadedState::stateEntered() {
22255bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Idle->Loaded", mCodec->mComponentName.c_str());
2226f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2227f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2228f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::IdleToLoadedState::onOMXEvent(
2229f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2230f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2231f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2232f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2233f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandStateSet);
2234f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data2, (OMX_U32)OMX_StateLoaded);
2235f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
22365bc087c573c70c84c6a39946457590b42d392a33Andreas Huber            LOGV("[%s] Now Loaded", mCodec->mComponentName.c_str());
2237f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2238f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(mCodec->mOMX->freeNode(mCodec->mNode), (status_t)OK);
2239f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2240f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mNativeWindow.clear();
2241f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mNode = NULL;
2242f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mOMX.clear();
2243f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mComponentName.clear();
2244f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2245f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->changeState(mCodec->mUninitializedState);
2246f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2247f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            sp<AMessage> notify = mCodec->mNotify->dup();
2248f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->setInt32("what", ACodec::kWhatShutdownCompleted);
2249f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            notify->post();
2250f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2251f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2252f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2253f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2254f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2255f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
2256f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2257f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2258f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2259f933441648ef6a71dee783d733aac17b9508b452Andreas Huber////////////////////////////////////////////////////////////////////////////////
2260f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2261f933441648ef6a71dee783d733aac17b9508b452Andreas HuberACodec::FlushingState::FlushingState(ACodec *codec)
2262f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    : BaseState(codec) {
2263f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2264f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2265f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::stateEntered() {
22665bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    LOGV("[%s] Now Flushing", mCodec->mComponentName.c_str());
2267f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2268f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    mFlushComplete[kPortIndexInput] = mFlushComplete[kPortIndexOutput] = false;
2269f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2270f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2271f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::FlushingState::onMessageReceived(const sp<AMessage> &msg) {
2272f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    bool handled = false;
2273f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2274f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (msg->what()) {
2275f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatShutdown:
2276f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2277f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->deferMessage(msg);
2278f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2279f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2280f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2281f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case kWhatFlush:
2282f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2283f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            // We're already doing this right now.
2284f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = true;
2285f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2286f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2287f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2288f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2289f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            handled = BaseState::onMessageReceived(msg);
2290f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
2291f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2292f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2293f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return handled;
2294f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2295f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2296f933441648ef6a71dee783d733aac17b9508b452Andreas Huberbool ACodec::FlushingState::onOMXEvent(
2297f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        OMX_EVENTTYPE event, OMX_U32 data1, OMX_U32 data2) {
2298f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    switch (event) {
2299f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case OMX_EventCmdComplete:
2300f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        {
2301f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            CHECK_EQ(data1, (OMX_U32)OMX_CommandFlush);
2302f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2303f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            if (data2 == kPortIndexInput || data2 == kPortIndexOutput) {
2304f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(!mFlushComplete[data2]);
2305f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                mFlushComplete[data2] = true;
2306e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber
2307e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                if (mFlushComplete[kPortIndexInput]
2308e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                        && mFlushComplete[kPortIndexOutput]) {
2309e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                    changeStateIfWeOwnAllBuffers();
2310e5bf6409774fe73f968b7db215f535b64de66139Andreas Huber                }
2311f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            } else {
2312f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK_EQ(data2, OMX_ALL);
2313f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(mFlushComplete[kPortIndexInput]);
2314f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                CHECK(mFlushComplete[kPortIndexOutput]);
2315f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2316f933441648ef6a71dee783d733aac17b9508b452Andreas Huber                changeStateIfWeOwnAllBuffers();
2317f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            }
2318f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2319f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return true;
2320f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        }
2321f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2322349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        case OMX_EventPortSettingsChanged:
2323349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        {
2324349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            sp<AMessage> msg = new AMessage(kWhatOMXMessage, mCodec->id());
2325349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("type", omx_message::EVENT);
2326349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setPointer("node", mCodec->mNode);
2327349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("event", event);
2328349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("data1", data1);
2329349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            msg->setInt32("data2", data2);
2330349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2331349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            LOGV("[%s] Deferring OMX_EventPortSettingsChanged",
2332349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber                 mCodec->mComponentName.c_str());
2333349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2334349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            mCodec->deferMessage(msg);
2335349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2336349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber            return true;
2337349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber        }
2338349d3fcb4afacf754f7b5b5186d2f258f5bf35e7Andreas Huber
2339f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        default:
2340f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            return BaseState::onOMXEvent(event, data1, data2);
2341f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2342f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2343f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    return true;
2344f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2345f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2346f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::onOutputBufferDrained(const sp<AMessage> &msg) {
2347f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onOutputBufferDrained(msg);
2348f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2349f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2350f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2351f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2352f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::onInputBufferFilled(const sp<AMessage> &msg) {
2353f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    BaseState::onInputBufferFilled(msg);
2354f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2355f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    changeStateIfWeOwnAllBuffers();
2356f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2357f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2358f933441648ef6a71dee783d733aac17b9508b452Andreas Hubervoid ACodec::FlushingState::changeStateIfWeOwnAllBuffers() {
2359f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    if (mFlushComplete[kPortIndexInput]
2360f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && mFlushComplete[kPortIndexOutput]
2361f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            && mCodec->allYourBuffersAreBelongToUs()) {
2362f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        sp<AMessage> notify = mCodec->mNotify->dup();
2363f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        notify->setInt32("what", ACodec::kWhatFlushCompleted);
2364f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        notify->post();
2365f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2366f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->mPortEOS[kPortIndexInput] =
2367f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            mCodec->mPortEOS[kPortIndexOutput] = false;
2368f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2369f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        mCodec->changeState(mCodec->mExecutingState);
2370f933441648ef6a71dee783d733aac17b9508b452Andreas Huber    }
2371f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}
2372f933441648ef6a71dee783d733aac17b9508b452Andreas Huber
2373f933441648ef6a71dee783d733aac17b9508b452Andreas Huber}  // namespace android
2374