1faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana/*
2faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * Copyright (C) 2013 The Android Open Source Project
3faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana *
4faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * Licensed under the Apache License, Version 2.0 (the "License");
5faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * you may not use this file except in compliance with the License.
6faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * You may obtain a copy of the License at
7faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana *
8faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana *      http://www.apache.org/licenses/LICENSE-2.0
9faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana *
10faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * Unless required by applicable law or agreed to in writing, software
11faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * distributed under the License is distributed on an "AS IS" BASIS,
12faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * See the License for the specific language governing permissions and
14faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana * limitations under the License.
15faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana */
16faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
17faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#define LOG_TAG "EffectProxy"
18faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana//#define LOG_NDEBUG 0
19faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
20faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#include <assert.h>
21faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#include <stdlib.h>
22faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#include <string.h>
23faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#include <new>
2460d02077d86d2d1092443519290101f503aa6f7aMark Salyzyn
25faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#include <EffectProxy.h>
2660d02077d86d2d1092443519290101f503aa6f7aMark Salyzyn
2760d02077d86d2d1092443519290101f503aa6f7aMark Salyzyn#include <log/log.h>
28faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#include <utils/threads.h>
2960d02077d86d2d1092443519290101f503aa6f7aMark Salyzyn
30faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#include <media/EffectsFactoryApi.h>
31faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
32faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmananamespace android {
33faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana// This is a dummy proxy descriptor just to return to Factory during the initial
34faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana// GetDescriptor call. Later in the factory, it is replaced with the
35faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana// SW sub effect descriptor
36385e7509eb563c983647e72b1232225c2200435fEric Laurent// proxy UUID af8da7e0-2ca1-11e3-b71d-0002a5d5c51b
37faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaconst effect_descriptor_t gProxyDescriptor = {
38faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        EFFECT_UUID_INITIALIZER, // type
39385e7509eb563c983647e72b1232225c2200435fEric Laurent        {0xaf8da7e0, 0x2ca1, 0x11e3, 0xb71d, { 0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b }}, // uuid
40faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        EFFECT_CONTROL_API_VERSION, //version of effect control API
41faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        (EFFECT_FLAG_TYPE_INSERT | EFFECT_FLAG_INSERT_LAST |
42faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana         EFFECT_FLAG_VOLUME_CTRL), // effect capability flags
43faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        0, // CPU load
44faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        1, // Data memory
45faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        "Proxy", //effect name
46faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        "AOSP", //implementor name
47faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana};
48faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
49faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
50faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaint EffectProxyCreate(const effect_uuid_t *uuid,
51faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                            int32_t             sessionId,
52faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                            int32_t             ioId,
53faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                           effect_handle_t  *pHandle) {
54faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
55faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    effect_descriptor_t* desc;
56f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    audio_effect_library_t** aeli;
57f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    sub_effect_entry_t** sube;
58faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    EffectContext* pContext;
59faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pHandle == NULL || uuid == NULL) {
60faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGE("EffectProxyCreate() called with NULL pointer");
61faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        return -EINVAL;
62faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
63faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    ALOGV("EffectProxyCreate start..");
64faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext = new EffectContext;
65faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext->sessionId = sessionId;
66faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext->ioId = ioId;
67faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext->uuid = *uuid;
68faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext->common_itfe = &gEffectInterface;
695d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent
70faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // The sub effects will be created in effect_command when the first command
71faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // for the effect is received
72faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext->eHandle[SUB_FX_HOST] = pContext->eHandle[SUB_FX_OFFLOAD] = NULL;
73faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
74faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // Get the HW and SW sub effect descriptors from the effects factory
75faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    desc = new effect_descriptor_t[SUB_FX_COUNT];
76f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    aeli = new audio_effect_library_t*[SUB_FX_COUNT];
77f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    sube = new sub_effect_entry_t*[SUB_FX_COUNT];
78f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    pContext->sube = new sub_effect_entry_t*[SUB_FX_COUNT];
79faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext->desc = new effect_descriptor_t[SUB_FX_COUNT];
80f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    pContext->aeli = new audio_effect_library_t*[SUB_FX_COUNT];
81f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    int retValue = EffectGetSubEffects(uuid, sube, SUB_FX_COUNT);
82faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // EffectGetSubEffects returns the number of sub-effects copied.
83faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (retValue != SUB_FX_COUNT) {
84faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana       ALOGE("EffectCreate() could not get the sub effects");
85f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       delete[] sube;
86f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       delete[] desc;
87f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       delete[] aeli;
88f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       delete[] pContext->sube;
89f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       delete[] pContext->desc;
90f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       delete[] pContext->aeli;
91f8dd1bdb12c76822ec1c2ab5d848c31f256cf36aCaroline Tice       delete pContext;
92faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana       return -EINVAL;
93faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
94faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // Check which is the HW descriptor and copy the descriptors
95faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // to the Context desc array
96faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // Also check if there is only one HW and one SW descriptor.
97faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // HW descriptor alone has the HW_TUNNEL flag.
98f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    desc[0] = *(effect_descriptor_t*)(sube[0])->object;
99f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    desc[1] = *(effect_descriptor_t*)(sube[1])->object;
100f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    aeli[0] = sube[0]->lib->desc;
101f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    aeli[1] = sube[1]->lib->desc;
102faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if ((desc[0].flags & EFFECT_FLAG_HW_ACC_TUNNEL) &&
103faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana       !(desc[1].flags & EFFECT_FLAG_HW_ACC_TUNNEL)) {
104f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->sube[SUB_FX_OFFLOAD] = sube[0];
105faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pContext->desc[SUB_FX_OFFLOAD] = desc[0];
106f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->aeli[SUB_FX_OFFLOAD] = aeli[0];
107f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->sube[SUB_FX_HOST] = sube[1];
108faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pContext->desc[SUB_FX_HOST] = desc[1];
109f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->aeli[SUB_FX_HOST] = aeli[1];
110faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
111faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    else if ((desc[1].flags & EFFECT_FLAG_HW_ACC_TUNNEL) &&
112faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana             !(desc[0].flags & EFFECT_FLAG_HW_ACC_TUNNEL)) {
113f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->sube[SUB_FX_HOST] = sube[0];
114faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pContext->desc[SUB_FX_HOST] = desc[0];
115f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->aeli[SUB_FX_HOST] = aeli[0];
116f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->sube[SUB_FX_OFFLOAD] = sube[1];
117faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pContext->desc[SUB_FX_OFFLOAD] = desc[1];
118f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        pContext->aeli[SUB_FX_OFFLOAD] = aeli[1];
119faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
120f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    delete[] desc;
121f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    delete[] aeli;
122f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    delete[] sube;
123faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#if (LOG_NDEBUG == 0)
124faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    effect_uuid_t uuid_print = pContext->desc[SUB_FX_HOST].uuid;
125faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    ALOGV("EffectCreate() UUID of HOST: %08X-%04X-%04X-%04X-%02X%02X%02X%02X"
126faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        "%02X%02X\n",uuid_print.timeLow, uuid_print.timeMid,
127faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        uuid_print.timeHiAndVersion, uuid_print.clockSeq, uuid_print.node[0],
128faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        uuid_print.node[1], uuid_print.node[2], uuid_print.node[3],
129faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        uuid_print.node[4], uuid_print.node[5]);
130faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    ALOGV("EffectCreate() UUID of OFFLOAD: %08X-%04X-%04X-%04X-%02X%02X%02X%02X"
131faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        "%02X%02X\n", uuid_print.timeLow, uuid_print.timeMid,
132faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        uuid_print.timeHiAndVersion, uuid_print.clockSeq, uuid_print.node[0],
133faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        uuid_print.node[1], uuid_print.node[2], uuid_print.node[3],
134faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        uuid_print.node[4], uuid_print.node[5]);
135faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana#endif
1365d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent
1375d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    pContext->replySize = PROXY_REPLY_SIZE_DEFAULT;
1385d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    pContext->replyData = (char *)malloc(PROXY_REPLY_SIZE_DEFAULT);
1395d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent
140faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    *pHandle = (effect_handle_t)pContext;
141faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    ALOGV("EffectCreate end");
142faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    return 0;
143faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana} //end EffectProxyCreate
144faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
145faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaint EffectProxyRelease(effect_handle_t handle) {
146faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    EffectContext * pContext = (EffectContext *)handle;
147faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext == NULL) {
148faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("ERROR : EffectRelease called with NULL pointer");
149faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        return -EINVAL;
150faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
151faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    ALOGV("EffectRelease");
152f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    delete[] pContext->desc;
1535d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    free(pContext->replyData);
1545d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent
155faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext->eHandle[SUB_FX_HOST])
156f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       pContext->aeli[SUB_FX_HOST]->release_effect(pContext->eHandle[SUB_FX_HOST]);
157faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext->eHandle[SUB_FX_OFFLOAD])
158f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana       pContext->aeli[SUB_FX_OFFLOAD]->release_effect(pContext->eHandle[SUB_FX_OFFLOAD]);
159f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    delete[] pContext->aeli;
160f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana    delete[] pContext->sube;
161faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    delete pContext;
162faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pContext = NULL;
163faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    return 0;
164faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana} /*end EffectProxyRelease */
165faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
166faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaint EffectProxyGetDescriptor(const effect_uuid_t *uuid,
167faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                                   effect_descriptor_t *pDescriptor) {
168faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    const effect_descriptor_t *desc = NULL;
169faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
170faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pDescriptor == NULL || uuid == NULL) {
171faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("EffectGetDescriptor() called with NULL pointer");
172faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        return -EINVAL;
173faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
174faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    desc = &gProxyDescriptor;
175faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    *pDescriptor = *desc;
176faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    return 0;
177faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana} /* end EffectProxyGetDescriptor */
178faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
179faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana/* Effect Control Interface Implementation: Process */
180faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaint Effect_process(effect_handle_t     self,
181faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              audio_buffer_t         *inBuffer,
182faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              audio_buffer_t         *outBuffer) {
183faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
184faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    EffectContext *pContext = (EffectContext *) self;
185faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    int ret = 0;
186faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext != NULL) {
187faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        int index = pContext->index;
188faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        // if the index refers to HW , do not do anything. Just return.
189faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        if (index == SUB_FX_HOST) {
190faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            ret = (*pContext->eHandle[index])->process(pContext->eHandle[index],
191faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                                                       inBuffer, outBuffer);
192faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        }
193faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
194faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    return ret;
195faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana}   /* end Effect_process */
196faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
197faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana/* Effect Control Interface Implementation: Command */
198faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaint Effect_command(effect_handle_t  self,
199faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              uint32_t            cmdCode,
200faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              uint32_t            cmdSize,
201faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              void                *pCmdData,
202faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              uint32_t            *replySize,
203faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              void                *pReplyData) {
204faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
205faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    EffectContext *pContext = (EffectContext *) self;
206eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent    int status = 0;
207faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext == NULL) {
208faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_command() Proxy context is NULL");
209faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        return -EINVAL;
210faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
211faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext->eHandle[SUB_FX_HOST] == NULL) {
212faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_command() Calling HOST EffectCreate");
213f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        status = pContext->aeli[SUB_FX_HOST]->create_effect(
214f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana                              &pContext->desc[SUB_FX_HOST].uuid,
215faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              pContext->sessionId, pContext->ioId,
216faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              &(pContext->eHandle[SUB_FX_HOST]));
217faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        if (status != NO_ERROR || (pContext->eHandle[SUB_FX_HOST] == NULL)) {
218faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            ALOGV("Effect_command() Error creating SW sub effect");
219faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            return status;
220faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        }
221faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
222faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext->eHandle[SUB_FX_OFFLOAD] == NULL) {
223faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_command() Calling OFFLOAD EffectCreate");
224f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        status = pContext->aeli[SUB_FX_OFFLOAD]->create_effect(
225f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana                              &pContext->desc[SUB_FX_OFFLOAD].uuid,
226faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              pContext->sessionId, pContext->ioId,
227faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                              &(pContext->eHandle[SUB_FX_OFFLOAD]));
228faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        if (status != NO_ERROR || (pContext->eHandle[SUB_FX_OFFLOAD] == NULL)) {
229faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            ALOGV("Effect_command() Error creating HW effect");
230f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana            pContext->eHandle[SUB_FX_OFFLOAD] = NULL;
231faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            // Do not return error here as SW effect is created
232faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            // Return error if the CMD_OFFLOAD sends the index as OFFLOAD
233faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        }
234faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pContext->index = SUB_FX_HOST;
235faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
236faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // EFFECT_CMD_OFFLOAD used to (1) send whether the thread is offload or not
237faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // (2) Send the ioHandle of the effectThread when the effect
238faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // is moved from one type of thread to another.
239faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // pCmdData points to a memory holding effect_offload_param_t structure
240faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (cmdCode == EFFECT_CMD_OFFLOAD) {
241faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_command() cmdCode = EFFECT_CMD_OFFLOAD");
24248946cd0cdaddfd8abcf6827ac4836cc111cd89cAndy Hung        if (replySize == NULL || *replySize < sizeof(int)) {
24348946cd0cdaddfd8abcf6827ac4836cc111cd89cAndy Hung            ALOGV("effectsOffload: Effect_command: CMD_OFFLOAD has no reply");
24448946cd0cdaddfd8abcf6827ac4836cc111cd89cAndy Hung            android_errorWriteLog(0x534e4554, "32448121");
24548946cd0cdaddfd8abcf6827ac4836cc111cd89cAndy Hung            return FAILED_TRANSACTION;
24648946cd0cdaddfd8abcf6827ac4836cc111cd89cAndy Hung        }
247faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        if (cmdSize == 0 || pCmdData == NULL) {
248faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            ALOGV("effectsOffload: Effect_command: CMD_OFFLOAD has no data");
249faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana             *(int*)pReplyData = FAILED_TRANSACTION;
250faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            return FAILED_TRANSACTION;
251faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        }
252faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        effect_offload_param_t* offloadParam = (effect_offload_param_t*)pCmdData;
253faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        // Assign the effect context index based on isOffload field of the structure
254faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pContext->index = offloadParam->isOffload ? SUB_FX_OFFLOAD : SUB_FX_HOST;
255faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        // if the index is HW and the HW effect is unavailable, return error
256faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        // and reset the index to SW
257faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        if (pContext->eHandle[pContext->index] == NULL) {
258faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            ALOGV("Effect_command()CMD_OFFLOAD sub effect unavailable");
259faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            *(int*)pReplyData = FAILED_TRANSACTION;
260faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana            return FAILED_TRANSACTION;
261faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        }
262faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pContext->ioId = offloadParam->ioHandle;
263faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_command()CMD_OFFLOAD index:%d io %d", pContext->index, pContext->ioId);
264faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        // Update the DSP wrapper with the new ioHandle.
265faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        // Pass the OFFLOAD command to the wrapper.
266faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        // The DSP wrapper needs to handle this CMD
267f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        if (pContext->eHandle[SUB_FX_OFFLOAD]) {
268f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana            ALOGV("Effect_command: Calling OFFLOAD command");
269f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana            return (*pContext->eHandle[SUB_FX_OFFLOAD])->command(
270f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana                           pContext->eHandle[SUB_FX_OFFLOAD], cmdCode, cmdSize,
271f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana                           pCmdData, replySize, pReplyData);
272f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        }
273f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        *(int*)pReplyData = NO_ERROR;
274f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        ALOGV("Effect_command OFFLOAD return 0, replyData %d",
275f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana                                                *(int*)pReplyData);
276f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana
277f90c7e0bb8d83d8b7f733bdf430d331ea3f221e8jpadmana        return NO_ERROR;
278faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
279faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
280faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    int index = pContext->index;
281faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (index != SUB_FX_HOST && index != SUB_FX_OFFLOAD) {
282faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_command: effect index is neither offload nor host");
283faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        return -EINVAL;
284faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
285eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent
286eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent    // Getter commands are only sent to the active sub effect.
2875d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    int *subStatus[SUB_FX_COUNT];
2885d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    uint32_t *subReplySize[SUB_FX_COUNT];
2895d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    void *subReplyData[SUB_FX_COUNT];
2905d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    uint32_t tmpSize;
2915d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    int tmpStatus;
292eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent
2935d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    // grow temp reply buffer if needed
2945d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    if (replySize != NULL) {
2955d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        tmpSize = pContext->replySize;
2965d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        while (tmpSize < *replySize && tmpSize < PROXY_REPLY_SIZE_MAX) {
2975d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            tmpSize *= 2;
2985d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        }
2995d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        if (tmpSize > pContext->replySize) {
3005d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            ALOGV("Effect_command grow reply buf to %d", tmpSize);
3015d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            pContext->replyData = (char *)realloc(pContext->replyData, tmpSize);
3025d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            pContext->replySize = tmpSize;
3035d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        }
3045d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        if (tmpSize > *replySize) {
3055d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            tmpSize = *replySize;
3065d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        }
3075d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    } else {
3085d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        tmpSize = 0;
309faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
3105d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    // tmpSize is now the actual reply size for the non active sub effect
3115d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent
3125d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    // Send command to sub effects. The command is sent to all sub effects so that their internal
3135d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    // state is kept in sync.
3145d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    // Only the reply from the active sub effect is returned to the caller. The reply from the
3155d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    // other sub effect is lost in pContext->replyData
3165d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent    for (int i = 0; i < SUB_FX_COUNT; i++) {
3175d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        if (pContext->eHandle[i] == NULL) {
3185d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            continue;
319eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent        }
3205d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        if (i == index) {
3215d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            subStatus[i] = &status;
3225d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            subReplySize[i] = replySize;
3235d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            subReplyData[i] = pReplyData;
3245d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        } else {
3255d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            subStatus[i] = &tmpStatus;
3265d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            subReplySize[i] = replySize == NULL ? NULL : &tmpSize;
3275d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent            subReplyData[i] = pReplyData == NULL ? NULL : pContext->replyData;
328eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent        }
3295d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent        *subStatus[i] = (*pContext->eHandle[i])->command(
3305d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent                             pContext->eHandle[i], cmdCode, cmdSize,
3315d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent                             pCmdData, subReplySize[i], subReplyData[i]);
332eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent    }
3335d6d86a4d102704f49b9235eaf282c428d7100b6Eric Laurent
334eba9bf72fb5e036bb15ca4a1dc126883a2cb938dEric Laurent    return status;
335faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana}    /* end Effect_command */
336faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
337faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
338faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana/* Effect Control Interface Implementation: get_descriptor */
339faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaint Effect_getDescriptor(effect_handle_t   self,
340faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana                         effect_descriptor_t *pDescriptor) {
341faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
342faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    EffectContext * pContext = (EffectContext *) self;
343faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    const effect_descriptor_t *desc;
344faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
345faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    ALOGV("Effect_getDescriptor");
346faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext == NULL || pDescriptor == NULL) {
347faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_getDescriptor() invalid param");
348faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        return -EINVAL;
349faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
350faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext->desc == NULL) {
351faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        ALOGV("Effect_getDescriptor() could not get descriptor");
352faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        return -EINVAL;
353faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    }
354faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    desc = &pContext->desc[SUB_FX_HOST];
355faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    *pDescriptor = *desc;
356faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    pDescriptor->uuid = pContext->uuid; // Replace the uuid with the Proxy UUID
357faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    // Also set/clear the EFFECT_FLAG_OFFLOAD_SUPPORTED flag based on the sub effects availability
358faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    if (pContext->eHandle[SUB_FX_OFFLOAD] != NULL)
359faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pDescriptor->flags |= EFFECT_FLAG_OFFLOAD_SUPPORTED;
360faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    else
361faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana        pDescriptor->flags &= ~EFFECT_FLAG_OFFLOAD_SUPPORTED;
362faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana    return 0;
363faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana} /* end Effect_getDescriptor */
364faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
365faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana} // namespace android
366faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana
367faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana__attribute__ ((visibility ("default")))
368faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmanaaudio_effect_library_t AUDIO_EFFECT_LIBRARY_INFO_SYM = {
3699803acb6b2c1b9c01444e0c8c0124adbe9a7157asynergy dev    .tag = AUDIO_EFFECT_LIBRARY_TAG,
3709803acb6b2c1b9c01444e0c8c0124adbe9a7157asynergy dev    .version = EFFECT_LIBRARY_API_VERSION,
3719803acb6b2c1b9c01444e0c8c0124adbe9a7157asynergy dev    .name = "Effect Proxy",
3729803acb6b2c1b9c01444e0c8c0124adbe9a7157asynergy dev    .implementor = "AOSP",
3739803acb6b2c1b9c01444e0c8c0124adbe9a7157asynergy dev    .create_effect = android::EffectProxyCreate,
3749803acb6b2c1b9c01444e0c8c0124adbe9a7157asynergy dev    .release_effect = android::EffectProxyRelease,
3759803acb6b2c1b9c01444e0c8c0124adbe9a7157asynergy dev    .get_descriptor = android::EffectProxyGetDescriptor,
376faca05e96744dfaa2f352e3dbb29eead4e55cfa0jpadmana};
377