OMXNodeInstance.cpp revision a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58
1318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber/*
2318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * Copyright (C) 2009 The Android Open Source Project
3318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber *
4318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * Licensed under the Apache License, Version 2.0 (the "License");
5318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * you may not use this file except in compliance with the License.
6318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * You may obtain a copy of the License at
7318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber *
8318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber *      http://www.apache.org/licenses/LICENSE-2.0
9318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber *
10318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * Unless required by applicable law or agreed to in writing, software
11318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * distributed under the License is distributed on an "AS IS" BASIS,
12318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * See the License for the specific language governing permissions and
14318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber * limitations under the License.
15318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber */
16318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
170c03d5c7c2fa4d17f7f5159e3fddd2adf6bfc923Andreas Huber//#define LOG_NDEBUG 0
18318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber#define LOG_TAG "OMXNodeInstance"
19318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber#include <utils/Log.h>
20318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
21318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber#include "../include/OMXNodeInstance.h"
22f0fb96c352f30b812a4903a1d783a715e1e817bdAndreas Huber#include "OMXMaster.h"
23f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden#include "GraphicBufferSource.h"
24318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
25e3ec3cec3a2e27033249ff82964d2cbd441d9873Andreas Huber#include <OMX_Component.h>
26318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
27318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber#include <binder/IMemory.h>
28f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden#include <gui/BufferQueue.h>
296c6b4d0d2b98a7ceee8b697daaf611f8df3254fbJames Dong#include <HardwareAPI.h>
30f1d5aa162c02a16b7195a43a9bcea4d592600ac4James Dong#include <media/stagefright/foundation/ADebug.h>
312a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber#include <media/stagefright/MediaErrors.h>
32318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
33f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddenstatic const OMX_U32 kPortIndexInput = 0;
34f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
35318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Hubernamespace android {
36318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
37318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstruct BufferMeta {
38318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta(const sp<IMemory> &mem, bool is_backup = false)
39318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        : mMem(mem),
40318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber          mIsBackup(is_backup) {
41318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
42318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
43318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta(size_t size)
44318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        : mSize(size),
45318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber          mIsBackup(false) {
46318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
47318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
4883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    BufferMeta(const sp<GraphicBuffer> &graphicBuffer)
4983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        : mGraphicBuffer(graphicBuffer),
5083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis          mIsBackup(false) {
5183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    }
5283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
53318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    void CopyFromOMX(const OMX_BUFFERHEADERTYPE *header) {
54318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        if (!mIsBackup) {
55318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            return;
56318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        }
57318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
58318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        memcpy((OMX_U8 *)mMem->pointer() + header->nOffset,
59318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber               header->pBuffer + header->nOffset,
60318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber               header->nFilledLen);
61318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
62318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
63318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    void CopyToOMX(const OMX_BUFFERHEADERTYPE *header) {
64318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        if (!mIsBackup) {
65318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            return;
66318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        }
67318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
68318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        memcpy(header->pBuffer + header->nOffset,
69318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber               (const OMX_U8 *)mMem->pointer() + header->nOffset,
70318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber               header->nFilledLen);
71318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
72318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
73d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    void setGraphicBuffer(const sp<GraphicBuffer> &graphicBuffer) {
74d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar        mGraphicBuffer = graphicBuffer;
75d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    }
76d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar
77318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberprivate:
7883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    sp<GraphicBuffer> mGraphicBuffer;
79318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    sp<IMemory> mMem;
80318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    size_t mSize;
81318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    bool mIsBackup;
82318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
83318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta(const BufferMeta &);
84318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta &operator=(const BufferMeta &);
85318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber};
86318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
87318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber// static
88318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMX_CALLBACKTYPE OMXNodeInstance::kCallbacks = {
89318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    &OnEvent, &OnEmptyBufferDone, &OnFillBufferDone
90318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber};
91318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
92318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMXNodeInstance::OMXNodeInstance(
93318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX *owner, const sp<IOMXObserver> &observer)
94318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    : mOwner(owner),
95318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber      mNodeID(NULL),
96318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber      mHandle(NULL),
97134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber      mObserver(observer),
98134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber      mDying(false) {
99318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
100318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
101318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMXNodeInstance::~OMXNodeInstance() {
102f1d5aa162c02a16b7195a43a9bcea4d592600ac4James Dong    CHECK(mHandle == NULL);
103318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
104318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
105318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Hubervoid OMXNodeInstance::setHandle(OMX::node_id node_id, OMX_HANDLETYPE handle) {
106f1d5aa162c02a16b7195a43a9bcea4d592600ac4James Dong    CHECK(mHandle == NULL);
107318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    mNodeID = node_id;
108318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    mHandle = handle;
109318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
110318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
111f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddensp<GraphicBufferSource> OMXNodeInstance::getGraphicBufferSource() {
112f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    Mutex::Autolock autoLock(mGraphicBufferSourceLock);
113f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    return mGraphicBufferSource;
114f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden}
115f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
116f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddenvoid OMXNodeInstance::setGraphicBufferSource(
117f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        const sp<GraphicBufferSource>& bufferSource) {
118f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    Mutex::Autolock autoLock(mGraphicBufferSourceLock);
119f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    mGraphicBufferSource = bufferSource;
120f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden}
121f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
122318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMX *OMXNodeInstance::owner() {
123318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return mOwner;
124318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
125318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
126318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Hubersp<IOMXObserver> OMXNodeInstance::observer() {
127318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return mObserver;
128318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
129318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
130318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMX::node_id OMXNodeInstance::nodeID() {
131318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return mNodeID;
132318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
133318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
134318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatic status_t StatusFromOMXError(OMX_ERRORTYPE err) {
1352a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    switch (err) {
1362a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        case OMX_ErrorNone:
1372a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            return OK;
1382a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        case OMX_ErrorUnsupportedSetting:
1392a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            return ERROR_UNSUPPORTED;
1402a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber        default:
1412a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber            return UNKNOWN_ERROR;
1422a3847ee1cbdaa8a65eee397a0173bb02211c459Andreas Huber    }
143318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
144318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
145f0fb96c352f30b812a4903a1d783a715e1e817bdAndreas Huberstatus_t OMXNodeInstance::freeNode(OMXMaster *master) {
14643e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber    static int32_t kMaxNumIterations = 10;
14743e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber
148d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    // Transition the node from its current state all the way down
149d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    // to "Loaded".
150d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    // This ensures that all active buffers are properly freed even
151d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    // for components that don't do this themselves on a call to
152d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    // "FreeHandle".
153d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
154134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    // The code below may trigger some more events to be dispatched
155134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    // by the OMX component - we want to ignore them as our client
156134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    // does not expect them.
157134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    mDying = true;
158134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber
159d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    OMX_STATETYPE state;
160d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    CHECK_EQ(OMX_GetState(mHandle, &state), OMX_ErrorNone);
161d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    switch (state) {
162d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        case OMX_StateExecuting:
163d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        {
1643856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV("forcing Executing->Idle");
165d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            sendCommand(OMX_CommandStateSet, OMX_StateIdle);
166d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            OMX_ERRORTYPE err;
16743e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber            int32_t iteration = 0;
168d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            while ((err = OMX_GetState(mHandle, &state)) == OMX_ErrorNone
1690d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber                   && state != OMX_StateIdle
1700d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber                   && state != OMX_StateInvalid) {
17143e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                if (++iteration > kMaxNumIterations) {
17229357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block                    ALOGE("component failed to enter Idle state, aborting.");
17343e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                    state = OMX_StateInvalid;
17443e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                    break;
17543e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                }
17643e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber
177d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber                usleep(100000);
178d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            }
179d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            CHECK_EQ(err, OMX_ErrorNone);
180d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
1810d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber            if (state == OMX_StateInvalid) {
1820d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber                break;
1830d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber            }
1840d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber
185d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            // fall through
186d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        }
187d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
188d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        case OMX_StateIdle:
189d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        {
1903856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV("forcing Idle->Loaded");
191d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            sendCommand(OMX_CommandStateSet, OMX_StateLoaded);
192d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
193d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            freeActiveBuffers();
194d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
195d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            OMX_ERRORTYPE err;
19643e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber            int32_t iteration = 0;
197d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            while ((err = OMX_GetState(mHandle, &state)) == OMX_ErrorNone
1980d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber                   && state != OMX_StateLoaded
1990d681df3b0ded2c1e335b6b5785439da4ce2c238Andreas Huber                   && state != OMX_StateInvalid) {
20043e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                if (++iteration > kMaxNumIterations) {
20129357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block                    ALOGE("component failed to enter Loaded state, aborting.");
20243e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                    state = OMX_StateInvalid;
20343e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                    break;
20443e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber                }
20543e5eca7048a3b7b3ee0223b7f3cbd837ed10ae5Andreas Huber
2063856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("waiting for Loaded state...");
207d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber                usleep(100000);
208d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            }
209d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            CHECK_EQ(err, OMX_ErrorNone);
210d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
211d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            // fall through
212d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        }
213d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
214d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        case OMX_StateLoaded:
215fa70cad40b01627ac1c22e04cdd548ece9c2654fAndreas Huber        case OMX_StateInvalid:
216d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            break;
217d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
218d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        default:
219d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            CHECK(!"should not be here, unknown state.");
220d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            break;
221d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    }
222d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
2233856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("calling destroyComponentInstance");
224f0fb96c352f30b812a4903a1d783a715e1e817bdAndreas Huber    OMX_ERRORTYPE err = master->destroyComponentInstance(
225f0fb96c352f30b812a4903a1d783a715e1e817bdAndreas Huber            static_cast<OMX_COMPONENTTYPE *>(mHandle));
2263856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("destroyComponentInstance returned err %d", err);
227f0fb96c352f30b812a4903a1d783a715e1e817bdAndreas Huber
228318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    mHandle = NULL;
229318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
230318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    if (err != OMX_ErrorNone) {
23129357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("FreeHandle FAILED with error 0x%08x.", err);
232318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
233318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
234318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    mOwner->invalidateNodeID(mNodeID);
235318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    mNodeID = NULL;
236318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
2373856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("OMXNodeInstance going away.");
23847bed1a7755ed58fa5d4c0d35b20468deb83bd60Andreas Huber    delete this;
239318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
240318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
241318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
242318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
243318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::sendCommand(
244318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_COMMANDTYPE cmd, OMX_S32 param) {
245e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    const sp<GraphicBufferSource>& bufferSource(getGraphicBufferSource());
246e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    if (bufferSource != NULL
247e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            && cmd == OMX_CommandStateSet
248e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            && param == OMX_StateLoaded) {
249e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        // Initiating transition from Executing -> Loaded
250e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        // Buffers are about to be freed.
251e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        bufferSource->omxLoaded();
252e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        setGraphicBufferSource(NULL);
253e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
254e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        // fall through
255e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    }
256e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
257318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
258318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
259318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_SendCommand(mHandle, cmd, param, NULL);
260318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
261318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
262318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
263318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::getParameter(
264318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_INDEXTYPE index, void *params, size_t size) {
265318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
266318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
267318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_GetParameter(mHandle, index, params);
268bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
269318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
270318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
271318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
272318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::setParameter(
273318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_INDEXTYPE index, const void *params, size_t size) {
274318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
275318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
276318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_SetParameter(
277318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            mHandle, index, const_cast<void *>(params));
278318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
279318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
280318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
281318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
282318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::getConfig(
283318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_INDEXTYPE index, void *params, size_t size) {
284318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
285318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
286318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_GetConfig(mHandle, index, params);
287318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
288318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
289318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
290318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::setConfig(
291318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_INDEXTYPE index, const void *params, size_t size) {
292318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
293318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
294318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_SetConfig(
295318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            mHandle, index, const_cast<void *>(params));
296318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
297318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
298b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennis}
299b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennis
300b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennisstatus_t OMXNodeInstance::getState(OMX_STATETYPE* state) {
301b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennis    Mutex::Autolock autoLock(mLock);
302b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennis
303b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennis    OMX_ERRORTYPE err = OMX_GetState(mHandle, state);
304b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennis
305b1d666f5cb555d135eb69e005e88a03330bbb54cJamie Gennis    return StatusFromOMXError(err);
306318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
307318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
30883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennisstatus_t OMXNodeInstance::enableGraphicBuffers(
30983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        OMX_U32 portIndex, OMX_BOOL enable) {
31083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    Mutex::Autolock autoLock(mLock);
311ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis    OMX_STRING name = const_cast<OMX_STRING>(
312ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis            "OMX.google.android.index.enableAndroidNativeBuffers");
31383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
31483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    OMX_INDEXTYPE index;
315ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis    OMX_ERRORTYPE err = OMX_GetExtensionIndex(mHandle, name, &index);
31683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
31783750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    if (err != OMX_ErrorNone) {
318d8754298b576aca4e85a059ea7b7e8f7a9502226Jamie Gennis        if (enable) {
319d8754298b576aca4e85a059ea7b7e8f7a9502226Jamie Gennis            ALOGE("OMX_GetExtensionIndex %s failed", name);
320d8754298b576aca4e85a059ea7b7e8f7a9502226Jamie Gennis        }
32183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
32283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        return StatusFromOMXError(err);
32383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    }
32483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
32583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    OMX_VERSIONTYPE ver;
32683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nVersionMajor = 1;
32783750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nVersionMinor = 0;
32883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nRevision = 0;
32983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nStep = 0;
33083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    EnableAndroidNativeBuffersParams params = {
33183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        sizeof(EnableAndroidNativeBuffersParams), ver, portIndex, enable,
33283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    };
33383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
33483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    err = OMX_SetParameter(mHandle, index, &params);
33583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
33683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    if (err != OMX_ErrorNone) {
33729357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_EnableAndroidNativeBuffers failed with error %d (0x%08x)",
33883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis                err, err);
33983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
34083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        return UNKNOWN_ERROR;
34183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    }
34283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
34383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    return OK;
34483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis}
34583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
346e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennisstatus_t OMXNodeInstance::getGraphicBufferUsage(
347e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis        OMX_U32 portIndex, OMX_U32* usage) {
348e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    Mutex::Autolock autoLock(mLock);
349e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
350e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    OMX_INDEXTYPE index;
351ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis    OMX_STRING name = const_cast<OMX_STRING>(
352ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis            "OMX.google.android.index.getAndroidNativeBufferUsage");
353ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis    OMX_ERRORTYPE err = OMX_GetExtensionIndex(mHandle, name, &index);
354e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
355e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    if (err != OMX_ErrorNone) {
356ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis        ALOGE("OMX_GetExtensionIndex %s failed", name);
357e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
358e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis        return StatusFromOMXError(err);
359e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    }
360e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
361e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    OMX_VERSIONTYPE ver;
362e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    ver.s.nVersionMajor = 1;
363e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    ver.s.nVersionMinor = 0;
364e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    ver.s.nRevision = 0;
365e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    ver.s.nStep = 0;
366e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    GetAndroidNativeBufferUsageParams params = {
367e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis        sizeof(GetAndroidNativeBufferUsageParams), ver, portIndex, 0,
368e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    };
369e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
370e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    err = OMX_GetParameter(mHandle, index, &params);
371e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
372e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    if (err != OMX_ErrorNone) {
37329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_GetAndroidNativeBufferUsage failed with error %d (0x%08x)",
374e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis                err, err);
375e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis        return UNKNOWN_ERROR;
376e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    }
377e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
378e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    *usage = params.nUsage;
379e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
380e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis    return OK;
381e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis}
382e2ce6458659c6e1bad420357b61dc10cd8bbe2abJamie Gennis
383e870772a78ffe08b1c14a791e368f1499f1be0f3James Dongstatus_t OMXNodeInstance::storeMetaDataInBuffers(
384e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong        OMX_U32 portIndex,
385e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong        OMX_BOOL enable) {
386e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    Mutex::Autolock autolock(mLock);
387f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    return storeMetaDataInBuffers_l(portIndex, enable);
388f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden}
389e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong
390f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddenstatus_t OMXNodeInstance::storeMetaDataInBuffers_l(
391f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        OMX_U32 portIndex,
392f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        OMX_BOOL enable) {
393e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    OMX_INDEXTYPE index;
394e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    OMX_STRING name = const_cast<OMX_STRING>(
395e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong            "OMX.google.android.index.storeMetaDataInBuffers");
396e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong
397e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    OMX_ERRORTYPE err = OMX_GetExtensionIndex(mHandle, name, &index);
398e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    if (err != OMX_ErrorNone) {
399d8754298b576aca4e85a059ea7b7e8f7a9502226Jamie Gennis        ALOGE("OMX_GetExtensionIndex %s failed", name);
400d8754298b576aca4e85a059ea7b7e8f7a9502226Jamie Gennis
401e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong        return StatusFromOMXError(err);
402e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    }
403e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong
404e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    StoreMetaDataInBuffersParams params;
405e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    memset(&params, 0, sizeof(params));
406e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    params.nSize = sizeof(params);
407e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong
408e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    // Version: 1.0.0.0
409e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    params.nVersion.s.nVersionMajor = 1;
410e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong
411e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    params.nPortIndex = portIndex;
412e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    params.bStoreMetaData = enable;
413e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    if ((err = OMX_SetParameter(mHandle, index, &params)) != OMX_ErrorNone) {
41429357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_SetParameter() failed for StoreMetaDataInBuffers: 0x%08x", err);
415e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong        return UNKNOWN_ERROR;
416e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    }
417e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong    return err;
418e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong}
419e870772a78ffe08b1c14a791e368f1499f1be0f3James Dong
420318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::useBuffer(
421318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_U32 portIndex, const sp<IMemory> &params,
422318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX::buffer_id *buffer) {
423318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
424318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
425318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta *buffer_meta = new BufferMeta(params);
426318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
427318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_BUFFERHEADERTYPE *header;
428318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
429318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_UseBuffer(
430318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            mHandle, &header, portIndex, buffer_meta,
431318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            params->size(), static_cast<OMX_U8 *>(params->pointer()));
432318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
433318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    if (err != OMX_ErrorNone) {
43429357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_UseBuffer failed with error %d (0x%08x)", err, err);
435318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
436318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        delete buffer_meta;
437318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        buffer_meta = NULL;
438318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
439318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        *buffer = 0;
440318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
441318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        return UNKNOWN_ERROR;
442318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
443318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
44403b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber    CHECK_EQ(header->pAppPrivate, buffer_meta);
44503b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber
446318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    *buffer = header;
447318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
448d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    addActiveBuffer(portIndex, *buffer);
449d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
450f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    sp<GraphicBufferSource> bufferSource(getGraphicBufferSource());
451f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if (bufferSource != NULL && portIndex == kPortIndexInput) {
452f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        bufferSource->addCodecBuffer(header);
453f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
454f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
455318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return OK;
456318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
457318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
458c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajanstatus_t OMXNodeInstance::useGraphicBuffer2_l(
459c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        OMX_U32 portIndex, const sp<GraphicBuffer>& graphicBuffer,
460c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        OMX::buffer_id *buffer) {
461c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
462c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    // port definition
463c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    OMX_PARAM_PORTDEFINITIONTYPE def;
464c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    def.nSize = sizeof(OMX_PARAM_PORTDEFINITIONTYPE);
465c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    def.nVersion.s.nVersionMajor = 1;
466c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    def.nVersion.s.nVersionMinor = 0;
467c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    def.nVersion.s.nRevision = 0;
468c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    def.nVersion.s.nStep = 0;
469c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    def.nPortIndex = portIndex;
470c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    OMX_ERRORTYPE err = OMX_GetParameter(mHandle, OMX_IndexParamPortDefinition, &def);
471a0dac9e24ae7520cb7d7f0505bf0936bffbcd047Jamie Gennis    if (err != OMX_ErrorNone)
472c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    {
47329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("%s::%d:Error getting OMX_IndexParamPortDefinition", __FUNCTION__, __LINE__);
474c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        return err;
475c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    }
476c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
477c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    BufferMeta *bufferMeta = new BufferMeta(graphicBuffer);
478c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
479c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    OMX_BUFFERHEADERTYPE *header = NULL;
480c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    OMX_U8* bufferHandle = const_cast<OMX_U8*>(
481c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            reinterpret_cast<const OMX_U8*>(graphicBuffer->handle));
482c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
483c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    err = OMX_UseBuffer(
484c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            mHandle,
485c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            &header,
486c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            portIndex,
487c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            bufferMeta,
488c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            def.nBufferSize,
489c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            bufferHandle);
490c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
491c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    if (err != OMX_ErrorNone) {
49229357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_UseBuffer failed with error %d (0x%08x)", err, err);
493c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        delete bufferMeta;
494c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        bufferMeta = NULL;
495c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        *buffer = 0;
496c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        return UNKNOWN_ERROR;
497c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    }
498c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
499c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    CHECK_EQ(header->pBuffer, bufferHandle);
500c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    CHECK_EQ(header->pAppPrivate, bufferMeta);
501c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
502c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    *buffer = header;
503c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
504c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    addActiveBuffer(portIndex, *buffer);
505c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
506c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    return OK;
507c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan}
508c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
509c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan// XXX: This function is here for backwards compatibility.  Once the OMX
510c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan// implementations have been updated this can be removed and useGraphicBuffer2
511c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan// can be renamed to useGraphicBuffer.
51283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennisstatus_t OMXNodeInstance::useGraphicBuffer(
51383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        OMX_U32 portIndex, const sp<GraphicBuffer>& graphicBuffer,
51483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        OMX::buffer_id *buffer) {
51583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    Mutex::Autolock autoLock(mLock);
51683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
517c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    // See if the newer version of the extension is present.
51883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    OMX_INDEXTYPE index;
519c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    if (OMX_GetExtensionIndex(
520c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            mHandle,
521c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            const_cast<OMX_STRING>("OMX.google.android.index.useAndroidNativeBuffer2"),
522c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan            &index) == OMX_ErrorNone) {
523c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan        return useGraphicBuffer2_l(portIndex, graphicBuffer, buffer);
524c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan    }
525c5a57efb706ec79032fc09c43b16d11ed0876604Anu Sundararajan
526ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis    OMX_STRING name = const_cast<OMX_STRING>(
527ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis        "OMX.google.android.index.useAndroidNativeBuffer");
528ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis    OMX_ERRORTYPE err = OMX_GetExtensionIndex(mHandle, name, &index);
52983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
53083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    if (err != OMX_ErrorNone) {
531ce18d7d85a78ac6642624fef1b5831eff4c72d56Jamie Gennis        ALOGE("OMX_GetExtensionIndex %s failed", name);
53283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
53383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        return StatusFromOMXError(err);
53483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    }
53583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
53683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    BufferMeta *bufferMeta = new BufferMeta(graphicBuffer);
53783750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
53883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    OMX_BUFFERHEADERTYPE *header;
53983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
54083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    OMX_VERSIONTYPE ver;
54183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nVersionMajor = 1;
54283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nVersionMinor = 0;
54383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nRevision = 0;
54483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    ver.s.nStep = 0;
54583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    UseAndroidNativeBufferParams params = {
54683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        sizeof(UseAndroidNativeBufferParams), ver, portIndex, bufferMeta,
54783750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        &header, graphicBuffer,
54883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    };
54983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
55083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    err = OMX_SetParameter(mHandle, index, &params);
55183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
55283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    if (err != OMX_ErrorNone) {
55329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_UseAndroidNativeBuffer failed with error %d (0x%08x)", err,
55483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis                err);
55583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
55683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        delete bufferMeta;
55783750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        bufferMeta = NULL;
55883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
55983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        *buffer = 0;
56083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
56183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis        return UNKNOWN_ERROR;
56283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    }
56383750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
56483750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    CHECK_EQ(header->pAppPrivate, bufferMeta);
56583750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
56683750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    *buffer = header;
56783750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
56883750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    addActiveBuffer(portIndex, *buffer);
56983750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
57083750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis    return OK;
57183750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis}
57283750eaf5a3f38c243a9e7eb81d4b2421e3a0d88Jamie Gennis
573d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnarstatus_t OMXNodeInstance::updateGraphicBufferInMeta(
574d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar        OMX_U32 portIndex, const sp<GraphicBuffer>& graphicBuffer,
575d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar        OMX::buffer_id buffer) {
576d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    Mutex::Autolock autoLock(mLock);
577d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar
578d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    OMX_BUFFERHEADERTYPE *header = (OMX_BUFFERHEADERTYPE *)(buffer);
579d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    VideoDecoderOutputMetaData *metadata =
580d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar        (VideoDecoderOutputMetaData *)(header->pBuffer);
581d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    BufferMeta *bufferMeta = (BufferMeta *)(header->pAppPrivate);
582d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    bufferMeta->setGraphicBuffer(graphicBuffer);
583d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    metadata->eType = kMetadataBufferTypeGrallocSource;
584d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    metadata->pHandle = graphicBuffer->handle;
585d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar
586d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar    return OK;
587d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar}
588d0715867861c216e88a4a7523b6da8a3cb128724Lajos Molnar
589f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddenstatus_t OMXNodeInstance::createInputSurface(
590f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        OMX_U32 portIndex, sp<IGraphicBufferProducer> *bufferProducer) {
591f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    Mutex::Autolock autolock(mLock);
592f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    status_t err;
593f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
594f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    const sp<GraphicBufferSource>& surfaceCheck = getGraphicBufferSource();
595f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if (surfaceCheck != NULL) {
596f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        return ALREADY_EXISTS;
597f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
598f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
599f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    // Input buffers will hold meta-data (gralloc references).
600f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    err = storeMetaDataInBuffers_l(portIndex, OMX_TRUE);
601f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if (err != OK) {
602f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        return err;
603f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
604f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
605f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    // Retrieve the width and height of the graphic buffer, set when the
606f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    // codec was configured.
607f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    OMX_PARAM_PORTDEFINITIONTYPE def;
608f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    def.nSize = sizeof(def);
609f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    def.nVersion.s.nVersionMajor = 1;
610f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    def.nVersion.s.nVersionMinor = 0;
611f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    def.nVersion.s.nRevision = 0;
612f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    def.nVersion.s.nStep = 0;
613f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    def.nPortIndex = portIndex;
614f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    OMX_ERRORTYPE oerr = OMX_GetParameter(
615f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden            mHandle, OMX_IndexParamPortDefinition, &def);
616f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    CHECK(oerr == OMX_ErrorNone);
617f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
618ba6218eae3dbcf3f962b3561b26374a214dbf5e2Andy McFadden    if (def.format.video.eColorFormat != OMX_COLOR_FormatAndroidOpaque) {
61992cb8f928dc9e237c356c942d10b5c0c1e04b2aeAndy McFadden        ALOGE("createInputSurface requires COLOR_FormatSurface "
62092cb8f928dc9e237c356c942d10b5c0c1e04b2aeAndy McFadden              "(AndroidOpaque) color format");
621ba6218eae3dbcf3f962b3561b26374a214dbf5e2Andy McFadden        return INVALID_OPERATION;
622ba6218eae3dbcf3f962b3561b26374a214dbf5e2Andy McFadden    }
623ba6218eae3dbcf3f962b3561b26374a214dbf5e2Andy McFadden
624f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    GraphicBufferSource* bufferSource = new GraphicBufferSource(
6250c37f9d1320bb87fd242f9425c67dacd6ce20112Andy McFadden            this, def.format.video.nFrameWidth, def.format.video.nFrameHeight,
6260c37f9d1320bb87fd242f9425c67dacd6ce20112Andy McFadden            def.nBufferCountActual);
627f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if ((err = bufferSource->initCheck()) != OK) {
628f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        delete bufferSource;
629f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        return err;
630f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
631f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    setGraphicBufferSource(bufferSource);
632f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
633f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    *bufferProducer = bufferSource->getIGraphicBufferProducer();
634f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    return OK;
635f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden}
636f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
637f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddenstatus_t OMXNodeInstance::signalEndOfInputStream() {
638f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    // For non-Surface input, the MediaCodec should convert the call to a
639f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    // pair of requests (dequeue input buffer, queue input buffer with EOS
640f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    // flag set).  Seems easier than doing the equivalent from here.
641f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    sp<GraphicBufferSource> bufferSource(getGraphicBufferSource());
642f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if (bufferSource == NULL) {
643ba6218eae3dbcf3f962b3561b26374a214dbf5e2Andy McFadden        ALOGW("signalEndOfInputStream can only be used with Surface input");
644f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        return INVALID_OPERATION;
645f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    };
646ba6218eae3dbcf3f962b3561b26374a214dbf5e2Andy McFadden    return bufferSource->signalEndOfInputStream();
647f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden}
648f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
649318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::allocateBuffer(
650570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        OMX_U32 portIndex, size_t size, OMX::buffer_id *buffer,
651570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber        void **buffer_data) {
652318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
653318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
654318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta *buffer_meta = new BufferMeta(size);
655318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
656318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_BUFFERHEADERTYPE *header;
657318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
658318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_AllocateBuffer(
659318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            mHandle, &header, portIndex, buffer_meta, size);
660318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
661318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    if (err != OMX_ErrorNone) {
66229357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_AllocateBuffer failed with error %d (0x%08x)", err, err);
663318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
664318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        delete buffer_meta;
665318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        buffer_meta = NULL;
666318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
667318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        *buffer = 0;
668318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
669318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        return UNKNOWN_ERROR;
670318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
671318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
67203b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber    CHECK_EQ(header->pAppPrivate, buffer_meta);
67303b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber
674318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    *buffer = header;
675570a3cb7582daa030cb38eedc5eb6a06f86ecc7fAndreas Huber    *buffer_data = header->pBuffer;
676318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
677d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    addActiveBuffer(portIndex, *buffer);
678d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
679f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    sp<GraphicBufferSource> bufferSource(getGraphicBufferSource());
680f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if (bufferSource != NULL && portIndex == kPortIndexInput) {
681f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        bufferSource->addCodecBuffer(header);
682f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
683f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
684318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return OK;
685318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
686318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
687318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::allocateBufferWithBackup(
688318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_U32 portIndex, const sp<IMemory> &params,
689318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX::buffer_id *buffer) {
690318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
691318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
692318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta *buffer_meta = new BufferMeta(params, true);
693318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
694318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_BUFFERHEADERTYPE *header;
695318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
696318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_AllocateBuffer(
697318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            mHandle, &header, portIndex, buffer_meta, params->size());
698318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
699318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    if (err != OMX_ErrorNone) {
70029357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("OMX_AllocateBuffer failed with error %d (0x%08x)", err, err);
701318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
702318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        delete buffer_meta;
703318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        buffer_meta = NULL;
704318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
705318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        *buffer = 0;
706318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
707318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        return UNKNOWN_ERROR;
708318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
709318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
71003b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber    CHECK_EQ(header->pAppPrivate, buffer_meta);
71103b268eac37ca2589bfff0bf58daf79d29cc14f4Andreas Huber
712318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    *buffer = header;
713318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
714d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    addActiveBuffer(portIndex, *buffer);
715d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
716f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    sp<GraphicBufferSource> bufferSource(getGraphicBufferSource());
717f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if (bufferSource != NULL && portIndex == kPortIndexInput) {
718f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        bufferSource->addCodecBuffer(header);
719f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
720f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
721318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return OK;
722318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
723318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
724318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::freeBuffer(
725318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_U32 portIndex, OMX::buffer_id buffer) {
726318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
727318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
728d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    removeActiveBuffer(portIndex, buffer);
729d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
730318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_BUFFERHEADERTYPE *header = (OMX_BUFFERHEADERTYPE *)buffer;
731318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta *buffer_meta = static_cast<BufferMeta *>(header->pAppPrivate);
732318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
733318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_FreeBuffer(mHandle, portIndex, header);
734318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
735318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    delete buffer_meta;
736318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    buffer_meta = NULL;
737318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
738318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
739318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
740318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
741318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::fillBuffer(OMX::buffer_id buffer) {
742318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
743318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
744318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_BUFFERHEADERTYPE *header = (OMX_BUFFERHEADERTYPE *)buffer;
745318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    header->nFilledLen = 0;
746318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    header->nOffset = 0;
747318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    header->nFlags = 0;
748318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
749318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_FillThisBuffer(mHandle, header);
750318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
751318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
752318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
753318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
754318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::emptyBuffer(
755318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX::buffer_id buffer,
756318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_U32 rangeOffset, OMX_U32 rangeLength,
757318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_U32 flags, OMX_TICKS timestamp) {
758318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
759318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
760318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_BUFFERHEADERTYPE *header = (OMX_BUFFERHEADERTYPE *)buffer;
761318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    header->nFilledLen = rangeLength;
762318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    header->nOffset = rangeOffset;
763318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    header->nFlags = flags;
764318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    header->nTimeStamp = timestamp;
765318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
766318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    BufferMeta *buffer_meta =
767318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        static_cast<BufferMeta *>(header->pAppPrivate);
768318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    buffer_meta->CopyToOMX(header);
769318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
770318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_EmptyThisBuffer(mHandle, header);
771318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
772318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
773318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
774318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
775f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden// like emptyBuffer, but the data is already in header->pBuffer
776f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddenstatus_t OMXNodeInstance::emptyDirectBuffer(
777f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        OMX_BUFFERHEADERTYPE *header,
778f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        OMX_U32 rangeOffset, OMX_U32 rangeLength,
779f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        OMX_U32 flags, OMX_TICKS timestamp) {
780f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    Mutex::Autolock autoLock(mLock);
781f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
782f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    header->nFilledLen = rangeLength;
783f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    header->nOffset = rangeOffset;
784f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    header->nFlags = flags;
785f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    header->nTimeStamp = timestamp;
786f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
787f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    OMX_ERRORTYPE err = OMX_EmptyThisBuffer(mHandle, header);
788f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    if (err != OMX_ErrorNone) {
789f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        ALOGW("emptyDirectBuffer failed, OMX err=0x%x", err);
790f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
791f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
792f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    return StatusFromOMXError(err);
793f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden}
794f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
795318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huberstatus_t OMXNodeInstance::getExtensionIndex(
796318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        const char *parameterName, OMX_INDEXTYPE *index) {
797318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
798318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
799318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMX_ERRORTYPE err = OMX_GetExtensionIndex(
800318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            mHandle, const_cast<char *>(parameterName), index);
801318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
802318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return StatusFromOMXError(err);
803318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
804318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
805e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huberstatus_t OMXNodeInstance::setInternalOption(
806e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        OMX_U32 portIndex,
807e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        IOMX::InternalOptionType type,
808e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        const void *data,
809e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        size_t size) {
810e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    switch (type) {
811e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        case IOMX::INTERNAL_OPTION_SUSPEND:
812a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber        case IOMX::INTERNAL_OPTION_REPEAT_PREVIOUS_FRAME_DELAY:
813e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        {
814e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            const sp<GraphicBufferSource> &bufferSource =
815e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber                getGraphicBufferSource();
816e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
817e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            if (bufferSource == NULL || portIndex != kPortIndexInput) {
818e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber                return ERROR_UNSUPPORTED;
819e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            }
820e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
821a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber            if (type == IOMX::INTERNAL_OPTION_SUSPEND) {
822a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                if (size != sizeof(bool)) {
823a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                    return INVALID_OPERATION;
824a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                }
825a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber
826a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                bool suspend = *(bool *)data;
827a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                bufferSource->suspend(suspend);
828a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber            } else {
829a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                if (size != sizeof(int64_t)) {
830a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                    return INVALID_OPERATION;
831a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                }
832a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber
833a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                int64_t delayUs = *(int64_t *)data;
834e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
835a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber                return bufferSource->setRepeatPreviousFrameDelayUs(delayUs);
836a61285dcf1da8a2cf40c499ee3a7b9fc4d74ac58Andreas Huber            }
837e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
838e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            return OK;
839e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        }
840e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
841e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        default:
842e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            return ERROR_UNSUPPORTED;
843e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    }
844e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber}
845e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber
846318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Hubervoid OMXNodeInstance::onMessage(const omx_message &msg) {
847318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    if (msg.type == omx_message::FILL_BUFFER_DONE) {
848318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_BUFFERHEADERTYPE *buffer =
849318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            static_cast<OMX_BUFFERHEADERTYPE *>(
850318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber                    msg.u.extended_buffer_data.buffer);
851318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
852318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        BufferMeta *buffer_meta =
853318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            static_cast<BufferMeta *>(buffer->pAppPrivate);
854318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
855318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        buffer_meta->CopyFromOMX(buffer);
8561b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber    } else if (msg.type == omx_message::EMPTY_BUFFER_DONE) {
8571b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber        const sp<GraphicBufferSource>& bufferSource(getGraphicBufferSource());
8581b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber
8591b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber        if (bufferSource != NULL) {
8601b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            // This is one of the buffers used exclusively by
8611b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            // GraphicBufferSource.
8621b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            // Don't dispatch a message back to ACodec, since it doesn't
8631b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            // know that anyone asked to have the buffer emptied and will
8641b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            // be very confused.
8651b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber
8661b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            OMX_BUFFERHEADERTYPE *buffer =
8671b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber                static_cast<OMX_BUFFERHEADERTYPE *>(
8681b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber                        msg.u.buffer_data.buffer);
8691b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber
8701b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            bufferSource->codecBufferEmptied(buffer);
8711b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber            return;
8721b4ca5cebd7f42a8f8842e45bfabe19001e9a435Andreas Huber        }
873318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
874318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
875318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    mObserver->onMessage(msg);
876318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
877318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
878f0fb96c352f30b812a4903a1d783a715e1e817bdAndreas Hubervoid OMXNodeInstance::onObserverDied(OMXMaster *master) {
87929357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block    ALOGE("!!! Observer died. Quickly, do something, ... anything...");
880318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
881318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    // Try to force shutdown of the node and hope for the best.
882f0fb96c352f30b812a4903a1d783a715e1e817bdAndreas Huber    freeNode(master);
883318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
884318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
885318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Hubervoid OMXNodeInstance::onGetHandleFailed() {
886318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    delete this;
887318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
888318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
889f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden// OMXNodeInstance::OnEvent calls OMX::OnEvent, which then calls here.
890f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden// Don't try to acquire mLock here -- in rare circumstances this will hang.
891f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFaddenvoid OMXNodeInstance::onEvent(
892f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden        OMX_EVENTTYPE event, OMX_U32 arg1, OMX_U32 arg2) {
893f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    const sp<GraphicBufferSource>& bufferSource(getGraphicBufferSource());
894f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
895e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber    if (bufferSource != NULL
896e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            && event == OMX_EventCmdComplete
897e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            && arg1 == OMX_CommandStateSet
898e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber            && arg2 == OMX_StateExecuting) {
899e40cda70eec141fa05cbcca1de420fdb22b98be6Andreas Huber        bufferSource->omxExecuting();
900f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden    }
901f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden}
902f779bb50d9746d9526541c3e6dcdf619cac941b7Andy McFadden
903318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber// static
904318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMX_ERRORTYPE OMXNodeInstance::OnEvent(
905318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_HANDLETYPE hComponent,
906318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_PTR pAppData,
907318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_EVENTTYPE eEvent,
908318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_U32 nData1,
909318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_U32 nData2,
910318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_PTR pEventData) {
911318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMXNodeInstance *instance = static_cast<OMXNodeInstance *>(pAppData);
912134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    if (instance->mDying) {
913134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber        return OMX_ErrorNone;
914134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    }
915318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return instance->owner()->OnEvent(
916318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber            instance->nodeID(), eEvent, nData1, nData2, pEventData);
917318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
918318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
919318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber// static
920318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMX_ERRORTYPE OMXNodeInstance::OnEmptyBufferDone(
921318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_HANDLETYPE hComponent,
922318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_PTR pAppData,
923318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_BUFFERHEADERTYPE* pBuffer) {
924318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMXNodeInstance *instance = static_cast<OMXNodeInstance *>(pAppData);
925134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    if (instance->mDying) {
926134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber        return OMX_ErrorNone;
927134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    }
928318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return instance->owner()->OnEmptyBufferDone(instance->nodeID(), pBuffer);
929318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
930318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
931318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber// static
932318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas HuberOMX_ERRORTYPE OMXNodeInstance::OnFillBufferDone(
933318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_HANDLETYPE hComponent,
934318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_PTR pAppData,
935318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        OMX_IN OMX_BUFFERHEADERTYPE* pBuffer) {
936318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    OMXNodeInstance *instance = static_cast<OMXNodeInstance *>(pAppData);
937134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    if (instance->mDying) {
938134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber        return OMX_ErrorNone;
939134ee6a324c35f39e3576172e4eae4c6de6eb9dcAndreas Huber    }
940318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return instance->owner()->OnFillBufferDone(instance->nodeID(), pBuffer);
941318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}
942318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
943d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Hubervoid OMXNodeInstance::addActiveBuffer(OMX_U32 portIndex, OMX::buffer_id id) {
944d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    ActiveBuffer active;
945d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    active.mPortIndex = portIndex;
946d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    active.mID = id;
947d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    mActiveBuffers.push(active);
948d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber}
949d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
950d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Hubervoid OMXNodeInstance::removeActiveBuffer(
951d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        OMX_U32 portIndex, OMX::buffer_id id) {
952d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    bool found = false;
953d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    for (size_t i = 0; i < mActiveBuffers.size(); ++i) {
954d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        if (mActiveBuffers[i].mPortIndex == portIndex
955d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            && mActiveBuffers[i].mID == id) {
956d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            found = true;
957d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            mActiveBuffers.removeItemsAt(i);
958d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber            break;
959d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        }
960d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    }
961d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
962d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    if (!found) {
9635ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("Attempt to remove an active buffer we know nothing about...");
964d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    }
965d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber}
966d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
967d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Hubervoid OMXNodeInstance::freeActiveBuffers() {
968d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    // Make sure to count down here, as freeBuffer will in turn remove
969d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    // the active buffer from the vector...
970d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    for (size_t i = mActiveBuffers.size(); i--;) {
971d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber        freeBuffer(mActiveBuffers[i].mPortIndex, mActiveBuffers[i].mID);
972d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber    }
973d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber}
974d6703ab22c85b43fdb2deb78a37e51465c902a5fAndreas Huber
975318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber}  // namespace android
976