PreProcessing.cpp revision 3f11ef01574219f323a7f9ee4216d610733df50f
1a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent/*
2a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * Copyright (C) 2011 The Android Open Source Project
3a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent *
4a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * Licensed under the Apache License, Version 2.0 (the "License");
5a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * you may not use this file except in compliance with the License.
6a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * You may obtain a copy of the License at
7a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent *
8a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent *      http://www.apache.org/licenses/LICENSE-2.0
9a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent *
10a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * Unless required by applicable law or agreed to in writing, software
11a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * distributed under the License is distributed on an "AS IS" BASIS,
12a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * See the License for the specific language governing permissions and
14a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent * limitations under the License.
15a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent */
16a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
17a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <stdlib.h>
18a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <string.h>
19a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#define LOG_TAG "PreProcessing"
20a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//#define LOG_NDEBUG 0
21a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <utils/Log.h>
22a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <utils/Timers.h>
23a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <hardware/audio_effect.h>
24a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <audio_effects/effect_aec.h>
25a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <audio_effects/effect_agc.h>
26a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include <audio_effects/effect_ns.h>
275387696d25bc710f8cd0e6d08079e2aa8d6c1417Eric Laurent#include <module_common_types.h>
285387696d25bc710f8cd0e6d08079e2aa8d6c1417Eric Laurent#include <audio_processing.h>
29a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#include "speex/speex_resampler.h"
30a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
313f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent// undefine to perform multi channels API functional tests
323f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent//#define DUAL_MIC_TEST
33a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
34a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
35a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// local definitions
36a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
37a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
38a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// maximum number of sessions
39a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#define PREPROC_NUM_SESSIONS 8
40a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
41a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// types of pre processing modules
42a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentenum preproc_id
43a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
44a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_AGC,        // Automatic Gain Control
45a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_AEC,        // Acoustic Echo Canceler
46a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_NS,         // Noise Suppressor
47a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_NUM_EFFECTS
48a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
49a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
50a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Session state
51a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentenum preproc_session_state {
52a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_SESSION_STATE_INIT,        // initialized
53a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_SESSION_STATE_CONFIG       // configuration received
54a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
55a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
56a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Effect/Preprocessor state
57a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentenum preproc_effect_state {
58a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_EFFECT_STATE_INIT,         // initialized
59a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_EFFECT_STATE_CREATED,      // webRTC engine created
60a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_EFFECT_STATE_CONFIG,       // configuration received/disabled
61a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PREPROC_EFFECT_STATE_ACTIVE        // active/enabled
62a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
63a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
64a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// handle on webRTC engine
65a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurenttypedef void* preproc_fx_handle_t;
66a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
67a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurenttypedef struct preproc_session_s preproc_session_t;
68a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurenttypedef struct preproc_effect_s preproc_effect_t;
69a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurenttypedef struct preproc_ops_s preproc_ops_t;
70a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
71a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Effect operation table. Functions for all pre processors are declared in sPreProcOps[] table.
72a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Function pointer can be null if no action required.
73a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstruct preproc_ops_s {
74a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int (* create)(preproc_effect_t *fx);
75a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int (* init)(preproc_effect_t *fx);
76a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int (* reset)(preproc_effect_t *fx);
77a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    void (* enable)(preproc_effect_t *fx);
78a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    void (* disable)(preproc_effect_t *fx);
79a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int (* set_parameter)(preproc_effect_t *fx, void *param, void *value);
80b302bd5d288be2d3363b80053ca2392560b00b25Ashok Bhat    int (* get_parameter)(preproc_effect_t *fx, void *param, uint32_t *size, void *value);
81a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int (* set_device)(preproc_effect_t *fx, uint32_t device);
82a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
83a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
84a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Effect context
85a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstruct preproc_effect_s {
86a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    const struct effect_interface_s *itfe;
87a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t procId;                // type of pre processor (enum preproc_id)
88a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t state;                 // current state (enum preproc_effect_state)
89a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_session_t *session;     // session the effect is on
90a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    const preproc_ops_t *ops;       // effect ops table
91a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_fx_handle_t engine;     // handle on webRTC engine
923f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#ifdef DUAL_MIC_TEST
933f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    bool aux_channels_on;           // support auxiliary channels
943f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    size_t cur_channel_config;      // current auciliary channel configuration
953f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#endif
96a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
97a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
98a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Session context
99a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstruct preproc_session_s {
100a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    struct preproc_effect_s effects[PREPROC_NUM_EFFECTS]; // effects in this session
101a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t state;                     // current state (enum preproc_session_state)
102a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int id;                             // audio session ID
103a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int io;                             // handle of input stream this session is on
104a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::AudioProcessing* apm;       // handle on webRTC audio processing module (APM)
105a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t apmFrameCount;               // buffer size for webRTC process (10 ms)
106a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t apmSamplingRate;           // webRTC APM sampling rate (8/16 or 32 kHz)
107a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t frameCount;                  // buffer size before input resampler ( <=> apmFrameCount)
108a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t samplingRate;              // sampling rate at effect process interface
109a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t inChannelCount;            // input channel count
110a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t outChannelCount;           // output channel count
111a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t createdMsk;                // bit field containing IDs of crested pre processors
112a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t enabledMsk;                // bit field containing IDs of enabled pre processors
113a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t processedMsk;              // bit field containing IDs of pre processors already
114a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                        // processed in current round
115a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::AudioFrame *procFrame;      // audio frame passed to webRTC AMP ProcessStream()
116a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int16_t *inBuf;                     // input buffer used when resampling
117a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t inBufSize;                   // input buffer size in frames
118a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t framesIn;                    // number of frames in input buffer
119a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    SpeexResamplerState *inResampler;   // handle on input speex resampler
120a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int16_t *outBuf;                    // output buffer used when resampling
121a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t outBufSize;                  // output buffer size in frames
122a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t framesOut;                   // number of frames in output buffer
123a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    SpeexResamplerState *outResampler;  // handle on output speex resampler
124a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t revChannelCount;           // number of channels on reverse stream
125a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t revEnabledMsk;             // bit field containing IDs of enabled pre processors
126a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                        // with reverse channel
127a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t revProcessedMsk;           // bit field containing IDs of pre processors with reverse
128a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                        // channel already processed in current round
129a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::AudioFrame *revFrame;       // audio frame passed to webRTC AMP AnalyzeReverseStream()
130a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int16_t *revBuf;                    // reverse channel input buffer
131a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t revBufSize;                  // reverse channel input buffer size
132a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t framesRev;                   // number of frames in reverse channel input buffer
133a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    SpeexResamplerState *revResampler;  // handle on reverse channel input speex resampler
134a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
135a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1363f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#ifdef DUAL_MIC_TEST
1373f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurentenum {
1383f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    PREPROC_CMD_DUAL_MIC_ENABLE = EFFECT_CMD_FIRST_PROPRIETARY, // enable dual mic mode
1393f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    PREPROC_CMD_DUAL_MIC_PCM_DUMP_START,                        // start pcm capture
1403f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    PREPROC_CMD_DUAL_MIC_PCM_DUMP_STOP                          // stop pcm capture
1413f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent};
1423f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
1433f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurentenum {
1443f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    CHANNEL_CFG_MONO,
1453f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    CHANNEL_CFG_STEREO,
1463f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    CHANNEL_CFG_MONO_AUX,
1473f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    CHANNEL_CFG_STEREO_AUX,
1483f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    CHANNEL_CFG_CNT,
1493f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    CHANNEL_CFG_FIRST_AUX = CHANNEL_CFG_MONO_AUX,
1503f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent};
1513f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
1523f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurentconst channel_config_t sDualMicConfigs[CHANNEL_CFG_CNT] = {
1533f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        {AUDIO_CHANNEL_IN_MONO , 0},
1543f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        {AUDIO_CHANNEL_IN_STEREO , 0},
1553f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        {AUDIO_CHANNEL_IN_FRONT , AUDIO_CHANNEL_IN_BACK},
1563f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        {AUDIO_CHANNEL_IN_STEREO , AUDIO_CHANNEL_IN_RIGHT}
1573f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent};
1583f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
1593f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurentbool sHasAuxChannels[PREPROC_NUM_EFFECTS] = {
1603f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        false,   // PREPROC_AGC
1613f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        true,   // PREPROC_AEC
1623f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        true,   // PREPROC_NS
1633f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent};
1643f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
1653f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurentbool gDualMicEnabled;
1663f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric LaurentFILE *gPcmDumpFh;
1673f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurentstatic pthread_mutex_t gPcmDumpLock = PTHREAD_MUTEX_INITIALIZER;
1683f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#endif
1693f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
1703f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
171a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
172a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Effect descriptors
173a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
174a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
175a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// UUIDs for effect types have been generated from http://www.itu.int/ITU-T/asn1/uuid.html
176a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// as the pre processing effects are not defined by OpenSL ES
177a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
178a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Automatic Gain Control
179a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const effect_descriptor_t sAgcDescriptor = {
180a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        { 0x0a8abfe0, 0x654c, 0x11e0, 0xba26, { 0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b } }, // type
181a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        { 0xaa8130e0, 0x66fc, 0x11e0, 0xbad0, { 0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b } }, // uuid
182a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        EFFECT_CONTROL_API_VERSION,
183a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        (EFFECT_FLAG_TYPE_PRE_PROC|EFFECT_FLAG_DEVICE_IND),
184a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        0, //FIXME indicate CPU load
185a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        0, //FIXME indicate memory usage
186a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        "Automatic Gain Control",
187a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        "The Android Open Source Project"
188a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
189a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
190a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Acoustic Echo Cancellation
191a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const effect_descriptor_t sAecDescriptor = {
192a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        { 0x7b491460, 0x8d4d, 0x11e0, 0xbd61, { 0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b } }, // type
193a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        { 0xbb392ec0, 0x8d4d, 0x11e0, 0xa896, { 0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b } }, // uuid
194a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        EFFECT_CONTROL_API_VERSION,
195a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        (EFFECT_FLAG_TYPE_PRE_PROC|EFFECT_FLAG_DEVICE_IND),
196a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        0, //FIXME indicate CPU load
197a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        0, //FIXME indicate memory usage
198a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        "Acoustic Echo Canceler",
199a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        "The Android Open Source Project"
200a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
201a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
202a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Noise suppression
203a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const effect_descriptor_t sNsDescriptor = {
204a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        { 0x58b4b260, 0x8e06, 0x11e0, 0xaa8e, { 0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b } }, // type
205a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        { 0xc06c8400, 0x8e06, 0x11e0, 0x9cb6, { 0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b } }, // uuid
206a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        EFFECT_CONTROL_API_VERSION,
207a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        (EFFECT_FLAG_TYPE_PRE_PROC|EFFECT_FLAG_DEVICE_IND),
208a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        0, //FIXME indicate CPU load
209a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        0, //FIXME indicate memory usage
210a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        "Noise Suppression",
211a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        "The Android Open Source Project"
212a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
213a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
214a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
215a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const effect_descriptor_t *sDescriptors[PREPROC_NUM_EFFECTS] = {
216a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        &sAgcDescriptor,
217a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        &sAecDescriptor,
218a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        &sNsDescriptor
219a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
220a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
221a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
222a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Helper functions
223a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
224a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
225a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentconst effect_uuid_t * const sUuidToPreProcTable[PREPROC_NUM_EFFECTS] = {
226a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        FX_IID_AGC,
227a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        FX_IID_AEC,
228a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        FX_IID_NS
229a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
230a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
231a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
232a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentconst effect_uuid_t * ProcIdToUuid(int procId)
233a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
234a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (procId >= PREPROC_NUM_EFFECTS) {
235a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return EFFECT_UUID_NULL;
236a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
237a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return sUuidToPreProcTable[procId];
238a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
239a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
240a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentuint32_t UuidToProcId(const effect_uuid_t * uuid)
241a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
242a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t i;
243a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    for (i = 0; i < PREPROC_NUM_EFFECTS; i++) {
244a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (memcmp(uuid, sUuidToPreProcTable[i], sizeof(*uuid)) == 0) {
245a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
246a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
247a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
248a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return i;
249a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
250a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
251a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentbool HasReverseStream(uint32_t procId)
252a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
253a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (procId == PREPROC_AEC) {
254a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return true;
255a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
256a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return false;
257a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
258a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
259a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
260a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
261a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Automatic Gain Control (AGC)
262a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
263a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
2645387696d25bc710f8cd0e6d08079e2aa8d6c1417Eric Laurentstatic const int kAgcDefaultTargetLevel = 3;
2655387696d25bc710f8cd0e6d08079e2aa8d6c1417Eric Laurentstatic const int kAgcDefaultCompGain = 9;
266a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const bool kAgcDefaultLimiter = true;
267a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
268a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint  AgcInit (preproc_effect_t *effect)
269a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
2703856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AgcInit");
271a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::GainControl *agc = static_cast<webrtc::GainControl *>(effect->engine);
272a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    agc->set_mode(webrtc::GainControl::kFixedDigital);
273a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    agc->set_target_level_dbfs(kAgcDefaultTargetLevel);
274a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    agc->set_compression_gain_db(kAgcDefaultCompGain);
275a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    agc->enable_limiter(kAgcDefaultLimiter);
276a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
277a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
278a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
279a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint  AgcCreate(preproc_effect_t *effect)
280a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
281a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::GainControl *agc = effect->session->apm->gain_control();
2823856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AgcCreate got agc %p", agc);
283a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (agc == NULL) {
2845ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AgcCreate Error");
285a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -ENOMEM;
286a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
287a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    effect->engine = static_cast<preproc_fx_handle_t>(agc);
288a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    AgcInit(effect);
289a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
290a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
291a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
292a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint AgcGetParameter(preproc_effect_t *effect,
293a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                    void *pParam,
294b302bd5d288be2d3363b80053ca2392560b00b25Ashok Bhat                    uint32_t *pValueSize,
295a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                    void *pValue)
296a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
297a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
298a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t param = *(uint32_t *)pParam;
299a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    t_agc_settings *pProperties = (t_agc_settings *)pValue;
300a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::GainControl *agc = static_cast<webrtc::GainControl *>(effect->engine);
301a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
302a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch (param) {
303a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_TARGET_LEVEL:
304a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_COMP_GAIN:
305a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (*pValueSize < sizeof(int16_t)) {
306a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *pValueSize = 0;
307a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return -EINVAL;
308a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
309a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
310a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_LIMITER_ENA:
311a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (*pValueSize < sizeof(bool)) {
312a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *pValueSize = 0;
313a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return -EINVAL;
314a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
315a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
316a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_PROPERTIES:
317a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (*pValueSize < sizeof(t_agc_settings)) {
318a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *pValueSize = 0;
319a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return -EINVAL;
320a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
321a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
322a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
323a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    default:
3245ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AgcGetParameter() unknown param %08x", param);
325a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = -EINVAL;
326a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
327a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
328a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
329a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch (param) {
330a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_TARGET_LEVEL:
331a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        *(int16_t *) pValue = (int16_t)(agc->target_level_dbfs() * -100);
3323856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AgcGetParameter() target level %d milliBels", *(int16_t *) pValue);
333a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
334a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_COMP_GAIN:
335a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        *(int16_t *) pValue = (int16_t)(agc->compression_gain_db() * 100);
3363856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AgcGetParameter() comp gain %d milliBels", *(int16_t *) pValue);
337a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
338a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_LIMITER_ENA:
339a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        *(bool *) pValue = (bool)agc->is_limiter_enabled();
3403856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AgcGetParameter() limiter enabled %s",
341a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent             (*(int16_t *) pValue != 0) ? "true" : "false");
342a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
343a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_PROPERTIES:
344a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        pProperties->targetLevel = (int16_t)(agc->target_level_dbfs() * -100);
345a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        pProperties->compGain = (int16_t)(agc->compression_gain_db() * 100);
346a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        pProperties->limiterEnabled = (bool)agc->is_limiter_enabled();
347a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
348a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    default:
3495ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AgcGetParameter() unknown param %d", param);
350a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = -EINVAL;
351a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
352a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
353a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
354a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
355a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
356a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint AgcSetParameter (preproc_effect_t *effect, void *pParam, void *pValue)
357a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
358a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
359a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t param = *(uint32_t *)pParam;
360a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    t_agc_settings *pProperties = (t_agc_settings *)pValue;
361a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::GainControl *agc = static_cast<webrtc::GainControl *>(effect->engine);
362a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
363a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch (param) {
364a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_TARGET_LEVEL:
3653856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AgcSetParameter() target level %d milliBels", *(int16_t *)pValue);
366a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = agc->set_target_level_dbfs(-(*(int16_t *)pValue / 100));
367a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
368a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_COMP_GAIN:
3693856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AgcSetParameter() comp gain %d milliBels", *(int16_t *)pValue);
370a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = agc->set_compression_gain_db(*(int16_t *)pValue / 100);
371a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
372a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_LIMITER_ENA:
3733856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AgcSetParameter() limiter enabled %s", *(bool *)pValue ? "true" : "false");
374a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = agc->enable_limiter(*(bool *)pValue);
375a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
376a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AGC_PARAM_PROPERTIES:
3773856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AgcSetParameter() properties level %d, gain %d limiter %d",
378a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent             pProperties->targetLevel,
379a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent             pProperties->compGain,
380a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent             pProperties->limiterEnabled);
381a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = agc->set_target_level_dbfs(-(pProperties->targetLevel / 100));
382a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (status != 0) break;
383a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = agc->set_compression_gain_db(pProperties->compGain / 100);
384a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (status != 0) break;
385a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = agc->enable_limiter(pProperties->limiterEnabled);
386a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
387a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    default:
3885ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AgcSetParameter() unknown param %08x value %08x", param, *(uint32_t *)pValue);
389a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = -EINVAL;
390a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
391a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
392a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
3933856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AgcSetParameter() done status %d", status);
394a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
395a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
396a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
397a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
398a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid AgcEnable(preproc_effect_t *effect)
399a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
400a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::GainControl *agc = static_cast<webrtc::GainControl *>(effect->engine);
4013856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AgcEnable agc %p", agc);
402a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    agc->Enable(true);
403a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
404a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
405a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid AgcDisable(preproc_effect_t *effect)
406a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
4073856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AgcDisable");
408a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::GainControl *agc = static_cast<webrtc::GainControl *>(effect->engine);
409a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    agc->Enable(false);
410a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
411a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
412a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
413a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const preproc_ops_t sAgcOps = {
414a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AgcCreate,
415a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AgcInit,
416a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NULL,
417a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AgcEnable,
418a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AgcDisable,
419a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AgcSetParameter,
420a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AgcGetParameter,
421a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NULL
422a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
423a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
424a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
425a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
426a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Acoustic Echo Canceler (AEC)
427a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
428a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
429a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const webrtc::EchoControlMobile::RoutingMode kAecDefaultMode =
430a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        webrtc::EchoControlMobile::kEarpiece;
431a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const bool kAecDefaultComfortNoise = true;
432a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
433a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint  AecInit (preproc_effect_t *effect)
434a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
4353856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AecInit");
436a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::EchoControlMobile *aec = static_cast<webrtc::EchoControlMobile *>(effect->engine);
437a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    aec->set_routing_mode(kAecDefaultMode);
438a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    aec->enable_comfort_noise(kAecDefaultComfortNoise);
439a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
440a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
441a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
442a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint  AecCreate(preproc_effect_t *effect)
443a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
444a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::EchoControlMobile *aec = effect->session->apm->echo_control_mobile();
4453856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AecCreate got aec %p", aec);
446a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (aec == NULL) {
4475ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AgcCreate Error");
448a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -ENOMEM;
449a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
450a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    effect->engine = static_cast<preproc_fx_handle_t>(aec);
451a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    AecInit (effect);
452a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
453a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
454a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
455b302bd5d288be2d3363b80053ca2392560b00b25Ashok Bhatint AecGetParameter(preproc_effect_t  *effect,
456a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                    void              *pParam,
457b302bd5d288be2d3363b80053ca2392560b00b25Ashok Bhat                    uint32_t          *pValueSize,
458a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                    void              *pValue)
459a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
460a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
461a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t param = *(uint32_t *)pParam;
462a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
463a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (*pValueSize < sizeof(uint32_t)) {
464a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
465a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
466a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch (param) {
467a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AEC_PARAM_ECHO_DELAY:
468a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AEC_PARAM_PROPERTIES:
469a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        *(uint32_t *)pValue = 1000 * effect->session->apm->stream_delay_ms();
4703856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AecGetParameter() echo delay %d us", *(uint32_t *)pValue);
471a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
472a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    default:
4735ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AecGetParameter() unknown param %08x value %08x", param, *(uint32_t *)pValue);
474a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = -EINVAL;
475a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
476a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
477a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
478a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
479a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
480a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint AecSetParameter (preproc_effect_t *effect, void *pParam, void *pValue)
481a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
482a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
483a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t param = *(uint32_t *)pParam;
484a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t value = *(uint32_t *)pValue;
485a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
486a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch (param) {
487a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AEC_PARAM_ECHO_DELAY:
488a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AEC_PARAM_PROPERTIES:
489a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = effect->session->apm->set_stream_delay_ms(value/1000);
4903856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("AecSetParameter() echo delay %d us, status %d", value, status);
491a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
492a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    default:
4935ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AecSetParameter() unknown param %08x value %08x", param, *(uint32_t *)pValue);
494a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = -EINVAL;
495a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
496a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
497a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
498a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
499a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
500a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid AecEnable(preproc_effect_t *effect)
501a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
502a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::EchoControlMobile *aec = static_cast<webrtc::EchoControlMobile *>(effect->engine);
5033856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AecEnable aec %p", aec);
504a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    aec->Enable(true);
505a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
506a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
507a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid AecDisable(preproc_effect_t *effect)
508a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
5093856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AecDisable");
510a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::EchoControlMobile *aec = static_cast<webrtc::EchoControlMobile *>(effect->engine);
511a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    aec->Enable(false);
512a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
513a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
514a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint AecSetDevice(preproc_effect_t *effect, uint32_t device)
515a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
5163856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AecSetDevice %08x", device);
517a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::EchoControlMobile *aec = static_cast<webrtc::EchoControlMobile *>(effect->engine);
518a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::EchoControlMobile::RoutingMode mode = webrtc::EchoControlMobile::kQuietEarpieceOrHeadset;
519a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
5208895925e38d2521607fd2010f3f3b14ecd15962cEric Laurent    if (audio_is_input_device(device)) {
5218895925e38d2521607fd2010f3f3b14ecd15962cEric Laurent        return 0;
5228895925e38d2521607fd2010f3f3b14ecd15962cEric Laurent    }
5238895925e38d2521607fd2010f3f3b14ecd15962cEric Laurent
524a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch(device) {
525a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AUDIO_DEVICE_OUT_EARPIECE:
526a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        mode = webrtc::EchoControlMobile::kEarpiece;
527a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
528a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AUDIO_DEVICE_OUT_SPEAKER:
529a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        mode = webrtc::EchoControlMobile::kSpeakerphone;
530a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
531a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AUDIO_DEVICE_OUT_WIRED_HEADSET:
532a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case AUDIO_DEVICE_OUT_WIRED_HEADPHONE:
533a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    default:
534a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
535a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
536a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    aec->set_routing_mode(mode);
537a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
538a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
539a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
540a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const preproc_ops_t sAecOps = {
541a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AecCreate,
542a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AecInit,
543a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NULL,
544a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AecEnable,
545a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AecDisable,
546a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AecSetParameter,
547a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AecGetParameter,
548a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        AecSetDevice
549a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
550a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
551a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
552a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Noise Suppression (NS)
553a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
554a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
555a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const webrtc::NoiseSuppression::Level kNsDefaultLevel = webrtc::NoiseSuppression::kModerate;
556a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
557a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint  NsInit (preproc_effect_t *effect)
558a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
5593856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("NsInit");
560a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::NoiseSuppression *ns = static_cast<webrtc::NoiseSuppression *>(effect->engine);
561a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    ns->set_level(kNsDefaultLevel);
562a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
563a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
564a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
565a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint  NsCreate(preproc_effect_t *effect)
566a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
567a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::NoiseSuppression *ns = effect->session->apm->noise_suppression();
5683856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("NsCreate got ns %p", ns);
569a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (ns == NULL) {
5705ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("AgcCreate Error");
571a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -ENOMEM;
572a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
573a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    effect->engine = static_cast<preproc_fx_handle_t>(ns);
574a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    NsInit (effect);
575a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
576a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
577a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
5780f714a464d2425afe00d6450535e763131b40844Eric Laurentint NsGetParameter(preproc_effect_t  *effect __unused,
5790f714a464d2425afe00d6450535e763131b40844Eric Laurent                   void              *pParam __unused,
5800f714a464d2425afe00d6450535e763131b40844Eric Laurent                   uint32_t          *pValueSize __unused,
5810f714a464d2425afe00d6450535e763131b40844Eric Laurent                   void              *pValue __unused)
582a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
583a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
584a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
585a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
586a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
5870f714a464d2425afe00d6450535e763131b40844Eric Laurentint NsSetParameter (preproc_effect_t *effect __unused,
5880f714a464d2425afe00d6450535e763131b40844Eric Laurent                    void *pParam __unused,
5890f714a464d2425afe00d6450535e763131b40844Eric Laurent                    void *pValue __unused)
590a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
591a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
592a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
593a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
594a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
595a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid NsEnable(preproc_effect_t *effect)
596a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
597a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::NoiseSuppression *ns = static_cast<webrtc::NoiseSuppression *>(effect->engine);
5983856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("NsEnable ns %p", ns);
599a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    ns->Enable(true);
600a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
601a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
602a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid NsDisable(preproc_effect_t *effect)
603a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
6043856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("NsDisable");
605a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    webrtc::NoiseSuppression *ns = static_cast<webrtc::NoiseSuppression *>(effect->engine);
606a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    ns->Enable(false);
607a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
608a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
609a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const preproc_ops_t sNsOps = {
610a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NsCreate,
611a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NsInit,
612a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NULL,
613a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NsEnable,
614a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NsDisable,
615a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NsSetParameter,
616a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NsGetParameter,
617a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        NULL
618a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
619a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
620a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
621a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const preproc_ops_t *sPreProcOps[PREPROC_NUM_EFFECTS] = {
622a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        &sAgcOps,
623a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        &sAecOps,
624a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        &sNsOps
625a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
626a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
627a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
628a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
629a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Effect functions
630a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
631a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
632a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid Session_SetProcEnabled(preproc_session_t *session, uint32_t procId, bool enabled);
633a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
634a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentextern "C" const struct effect_interface_s sEffectInterface;
635a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentextern "C" const struct effect_interface_s sEffectInterfaceReverse;
636a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
637a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#define BAD_STATE_ABORT(from, to) \
638a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        LOG_ALWAYS_FATAL("Bad state transition from %d to %d", from, to);
639a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
640a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Effect_SetState(preproc_effect_t *effect, uint32_t state)
641a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
642a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
6433856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Effect_SetState proc %d, new %d old %d", effect->procId, state, effect->state);
644a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch(state) {
645a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case PREPROC_EFFECT_STATE_INIT:
646a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        switch(effect->state) {
647a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_ACTIVE:
648a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            effect->ops->disable(effect);
649a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            Session_SetProcEnabled(effect->session, effect->procId, false);
650a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CONFIG:
651a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CREATED:
652a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_INIT:
653a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
654a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        default:
655a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            BAD_STATE_ABORT(effect->state, state);
656a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
657a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
658a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case PREPROC_EFFECT_STATE_CREATED:
659a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        switch(effect->state) {
660a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_INIT:
661a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            status = effect->ops->create(effect);
662a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
663a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CREATED:
664a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_ACTIVE:
665a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CONFIG:
66629357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("Effect_SetState invalid transition");
667a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            status = -ENOSYS;
668a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
669a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        default:
670a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            BAD_STATE_ABORT(effect->state, state);
671a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
672a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
673a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case PREPROC_EFFECT_STATE_CONFIG:
674a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        switch(effect->state) {
675a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_INIT:
67629357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("Effect_SetState invalid transition");
677a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            status = -ENOSYS;
678a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
679a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_ACTIVE:
680a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            effect->ops->disable(effect);
681a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            Session_SetProcEnabled(effect->session, effect->procId, false);
682a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
683a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CREATED:
684a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CONFIG:
685a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
686a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        default:
687a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            BAD_STATE_ABORT(effect->state, state);
688a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
689a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
690a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    case PREPROC_EFFECT_STATE_ACTIVE:
691a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        switch(effect->state) {
692a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_INIT:
693a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CREATED:
69429357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("Effect_SetState invalid transition");
695a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            status = -ENOSYS;
696a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
6973f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case PREPROC_EFFECT_STATE_ACTIVE:
6983f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            // enabling an already enabled effect is just ignored
6993f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            break;
700a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case PREPROC_EFFECT_STATE_CONFIG:
701a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            effect->ops->enable(effect);
702a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            Session_SetProcEnabled(effect->session, effect->procId, true);
703a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
704a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        default:
705a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            BAD_STATE_ABORT(effect->state, state);
706a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
707a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        break;
708a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    default:
709a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        BAD_STATE_ABORT(effect->state, state);
710a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
711a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (status == 0) {
712a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        effect->state = state;
713a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
714a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
715a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
716a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
717a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Effect_Init(preproc_effect_t *effect, uint32_t procId)
718a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
719a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (HasReverseStream(procId)) {
720a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        effect->itfe = &sEffectInterfaceReverse;
721a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    } else {
722a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        effect->itfe = &sEffectInterface;
723a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
724a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    effect->ops = sPreProcOps[procId];
725a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    effect->procId = procId;
726a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    effect->state = PREPROC_EFFECT_STATE_INIT;
727a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
728a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
729a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
730a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Effect_Create(preproc_effect_t *effect,
731a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent               preproc_session_t *session,
732a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent               effect_handle_t  *interface)
733a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
734a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    effect->session = session;
735a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    *interface = (effect_handle_t)&effect->itfe;
736a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return Effect_SetState(effect, PREPROC_EFFECT_STATE_CREATED);
737a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
738a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
739a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Effect_Release(preproc_effect_t *effect)
740a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
741a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return Effect_SetState(effect, PREPROC_EFFECT_STATE_INIT);
742a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
743a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
744a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
745a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
746a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Session functions
747a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
748a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
749a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent#define RESAMPLER_QUALITY SPEEX_RESAMPLER_QUALITY_VOIP
750a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
751a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const int kPreprocDefaultSr = 16000;
752a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic const int kPreProcDefaultCnl = 1;
753a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
754a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Session_Init(preproc_session_t *session)
755a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
756a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t i;
757a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
758a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
759a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->state = PREPROC_SESSION_STATE_INIT;
760a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->id = 0;
761a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->io = 0;
762a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->createdMsk = 0;
763a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->apm = NULL;
764a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    for (i = 0; i < PREPROC_NUM_EFFECTS && status == 0; i++) {
765a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = Effect_Init(&session->effects[i], i);
766a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
767a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
768a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
769a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
770a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
771a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentextern "C" int Session_CreateEffect(preproc_session_t *session,
772a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    int32_t procId,
773a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    effect_handle_t  *interface)
774a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
775a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = -ENOMEM;
776a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
7773856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Session_CreateEffect procId %d, createdMsk %08x", procId, session->createdMsk);
778a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
779a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->createdMsk == 0) {
7809718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs        session->apm = webrtc::AudioProcessing::Create();
781a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->apm == NULL) {
7825ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Session_CreateEffect could not get apm engine");
783a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            goto error;
784a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
7859718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs        const webrtc::ProcessingConfig processing_config = {
7869718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs            {{kPreprocDefaultSr, kPreProcDefaultCnl},
7879718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs             {kPreprocDefaultSr, kPreProcDefaultCnl},
7889718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs             {kPreprocDefaultSr, kPreProcDefaultCnl},
7899718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs             {kPreprocDefaultSr, kPreProcDefaultCnl}}};
7909718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs        session->apm->Initialize(processing_config);
791a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->procFrame = new webrtc::AudioFrame();
792a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->procFrame == NULL) {
7935ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Session_CreateEffect could not allocate audio frame");
794a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            goto error;
795a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
796a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revFrame = new webrtc::AudioFrame();
797a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->revFrame == NULL) {
7985ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Session_CreateEffect could not allocate reverse audio frame");
799a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            goto error;
800a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
801a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->apmSamplingRate = kPreprocDefaultSr;
802a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->apmFrameCount = (kPreprocDefaultSr) / 100;
803a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->frameCount = session->apmFrameCount;
804a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->samplingRate = kPreprocDefaultSr;
805a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->inChannelCount = kPreProcDefaultCnl;
806a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->outChannelCount = kPreProcDefaultCnl;
807de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh        session->procFrame->sample_rate_hz_ = kPreprocDefaultSr;
808de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh        session->procFrame->num_channels_ = kPreProcDefaultCnl;
809a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revChannelCount = kPreProcDefaultCnl;
810de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh        session->revFrame->sample_rate_hz_ = kPreprocDefaultSr;
811de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh        session->revFrame->num_channels_ = kPreProcDefaultCnl;
812a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->enabledMsk = 0;
813a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->processedMsk = 0;
814a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revEnabledMsk = 0;
815a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revProcessedMsk = 0;
816a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->inResampler = NULL;
817a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->inBuf = NULL;
818a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->inBufSize = 0;
819a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->outResampler = NULL;
820a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->outBuf = NULL;
821a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->outBufSize = 0;
822a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revResampler = NULL;
823a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revBuf = NULL;
824a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revBufSize = 0;
825a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
826a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    status = Effect_Create(&session->effects[procId], session, interface);
827a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (status < 0) {
828a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        goto error;
829a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
8303856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Session_CreateEffect OK");
831a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->createdMsk |= (1<<procId);
832a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
833a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
834a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurenterror:
835a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->createdMsk == 0) {
836a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        delete session->revFrame;
837a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revFrame = NULL;
838a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        delete session->procFrame;
839a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->procFrame = NULL;
8409718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs        delete session->apm;
841a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->apm = NULL;
842a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
843a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
844a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
845a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
846a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Session_ReleaseEffect(preproc_session_t *session,
847a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                          preproc_effect_t *fx)
848a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
8495ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block    ALOGW_IF(Effect_Release(fx) != 0, " Effect_Release() failed for proc ID %d", fx->procId);
850a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->createdMsk &= ~(1<<fx->procId);
851a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->createdMsk == 0) {
8529718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs        delete session->apm;
853a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->apm = NULL;
854a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        delete session->procFrame;
855a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->procFrame = NULL;
856a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        delete session->revFrame;
857a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revFrame = NULL;
858a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->inResampler != NULL) {
859a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            speex_resampler_destroy(session->inResampler);
860a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->inResampler = NULL;
861a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
862a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->outResampler != NULL) {
863a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            speex_resampler_destroy(session->outResampler);
864a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->outResampler = NULL;
865a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
866a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->revResampler != NULL) {
867a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            speex_resampler_destroy(session->revResampler);
868a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->revResampler = NULL;
869a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
870a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        delete session->inBuf;
871a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->inBuf = NULL;
872a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        delete session->outBuf;
873a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->outBuf = NULL;
874a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        delete session->revBuf;
875a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revBuf = NULL;
876a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
877a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->io = 0;
878a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
879a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
880a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
881a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
882a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
883a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
884a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Session_SetConfig(preproc_session_t *session, effect_config_t *config)
885a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
886a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t sr;
8873f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs    uint32_t inCnl = audio_channel_count_from_in_mask(config->inputCfg.channels);
8883f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs    uint32_t outCnl = audio_channel_count_from_in_mask(config->outputCfg.channels);
889a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
890a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (config->inputCfg.samplingRate != config->outputCfg.samplingRate ||
891a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        config->inputCfg.format != config->outputCfg.format ||
892a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        config->inputCfg.format != AUDIO_FORMAT_PCM_16_BIT) {
893a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
894a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
895a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
8963856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Session_SetConfig sr %d cnl %08x",
897a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent         config->inputCfg.samplingRate, config->inputCfg.channels);
898a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status;
899a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
90076533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent    // if at least one process is enabled, do not accept configuration changes
90176533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent    if (session->enabledMsk) {
90276533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent        if (session->samplingRate != config->inputCfg.samplingRate ||
90376533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent                session->inChannelCount != inCnl ||
90476533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent                session->outChannelCount != outCnl) {
90576533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent            return -ENOSYS;
90676533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent        } else {
90776533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent            return 0;
90876533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent        }
90976533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent    }
91076533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent
911a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    // AEC implementation is limited to 16kHz
912a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (config->inputCfg.samplingRate >= 32000 && !(session->createdMsk & (1 << PREPROC_AEC))) {
913a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->apmSamplingRate = 32000;
914a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    } else
915a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (config->inputCfg.samplingRate >= 16000) {
916a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->apmSamplingRate = 16000;
917a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    } else if (config->inputCfg.samplingRate >= 8000) {
918a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->apmSamplingRate = 8000;
919a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
9209718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs
9219718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs    const webrtc::ProcessingConfig processing_config = {
9223f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs      {{static_cast<int>(session->apmSamplingRate), inCnl},
9233f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs       {static_cast<int>(session->apmSamplingRate), outCnl},
9243f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs       {static_cast<int>(session->apmSamplingRate), inCnl},
9253f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs       {static_cast<int>(session->apmSamplingRate), inCnl}}};
9269718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs    status = session->apm->Initialize(processing_config);
927a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (status < 0) {
928a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
929a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
930a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
931a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->samplingRate = config->inputCfg.samplingRate;
932a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->apmFrameCount = session->apmSamplingRate / 100;
933a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->samplingRate == session->apmSamplingRate) {
934a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->frameCount = session->apmFrameCount;
935a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    } else {
936a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->frameCount = (session->apmFrameCount * session->samplingRate) /
937a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                session->apmSamplingRate  + 1;
938a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
939a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->inChannelCount = inCnl;
940a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->outChannelCount = outCnl;
941de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh    session->procFrame->num_channels_ = inCnl;
942de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh    session->procFrame->sample_rate_hz_ = session->apmSamplingRate;
943a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
944a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->revChannelCount = inCnl;
945de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh    session->revFrame->num_channels_ = inCnl;
946de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh    session->revFrame->sample_rate_hz_ = session->apmSamplingRate;
947a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
9483f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    // force process buffer reallocation
9493f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    session->inBufSize = 0;
9503f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    session->outBufSize = 0;
9513f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    session->framesIn = 0;
9523f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    session->framesOut = 0;
9533f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
9543f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
955a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->inResampler != NULL) {
956a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        speex_resampler_destroy(session->inResampler);
957a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->inResampler = NULL;
958a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
959a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->outResampler != NULL) {
960a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        speex_resampler_destroy(session->outResampler);
961a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->outResampler = NULL;
962a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
963a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->revResampler != NULL) {
964a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        speex_resampler_destroy(session->revResampler);
965a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revResampler = NULL;
966a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
967a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->samplingRate != session->apmSamplingRate) {
968a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        int error;
969a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->inResampler = speex_resampler_init(session->inChannelCount,
970a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    session->samplingRate,
971a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    session->apmSamplingRate,
972a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    RESAMPLER_QUALITY,
973a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    &error);
974a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->inResampler == NULL) {
9755ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Session_SetConfig Cannot create speex resampler: %s",
976a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                 speex_resampler_strerror(error));
977a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return -EINVAL;
978a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
979a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->outResampler = speex_resampler_init(session->outChannelCount,
980a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    session->apmSamplingRate,
981a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    session->samplingRate,
982a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    RESAMPLER_QUALITY,
983a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    &error);
984a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->outResampler == NULL) {
9855ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Session_SetConfig Cannot create speex resampler: %s",
986a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                 speex_resampler_strerror(error));
987a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            speex_resampler_destroy(session->inResampler);
988a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->inResampler = NULL;
989a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return -EINVAL;
990a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
991a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revResampler = speex_resampler_init(session->inChannelCount,
992a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    session->samplingRate,
993a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    session->apmSamplingRate,
994a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    RESAMPLER_QUALITY,
995a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                    &error);
996a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->revResampler == NULL) {
9975ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Session_SetConfig Cannot create speex resampler: %s",
998a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                 speex_resampler_strerror(error));
999a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            speex_resampler_destroy(session->inResampler);
1000a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->inResampler = NULL;
1001a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            speex_resampler_destroy(session->outResampler);
1002a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->outResampler = NULL;
1003a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return -EINVAL;
1004a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1005a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1006a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1007a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->state = PREPROC_SESSION_STATE_CONFIG;
1008a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
1009a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1010a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
10113d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurentvoid Session_GetConfig(preproc_session_t *session, effect_config_t *config)
10123d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent{
10133d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    memset(config, 0, sizeof(effect_config_t));
10143d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    config->inputCfg.samplingRate = config->outputCfg.samplingRate = session->samplingRate;
10153d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    config->inputCfg.format = config->outputCfg.format = AUDIO_FORMAT_PCM_16_BIT;
1016ab334fd351ae5a0e18903da123d63e565b536874Glenn Kasten    config->inputCfg.channels = audio_channel_in_mask_from_count(session->inChannelCount);
1017ab334fd351ae5a0e18903da123d63e565b536874Glenn Kasten    // "out" doesn't mean output device, so this is the correct API to convert channel count to mask
1018ab334fd351ae5a0e18903da123d63e565b536874Glenn Kasten    config->outputCfg.channels = audio_channel_in_mask_from_count(session->outChannelCount);
10193d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    config->inputCfg.mask = config->outputCfg.mask =
10203d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            (EFFECT_CONFIG_SMP_RATE | EFFECT_CONFIG_CHANNELS | EFFECT_CONFIG_FORMAT);
10213d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent}
10223d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent
1023a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint Session_SetReverseConfig(preproc_session_t *session, effect_config_t *config)
1024a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1025a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (config->inputCfg.samplingRate != config->outputCfg.samplingRate ||
1026a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            config->inputCfg.format != config->outputCfg.format ||
1027a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            config->inputCfg.format != AUDIO_FORMAT_PCM_16_BIT) {
1028a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1029a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1030a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
10313856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Session_SetReverseConfig sr %d cnl %08x",
1032a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent         config->inputCfg.samplingRate, config->inputCfg.channels);
1033a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1034a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session->state < PREPROC_SESSION_STATE_CONFIG) {
1035a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -ENOSYS;
1036a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1037a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (config->inputCfg.samplingRate != session->samplingRate ||
1038a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            config->inputCfg.format != AUDIO_FORMAT_PCM_16_BIT) {
1039a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1040a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1041e541269be94f3a1072932d51537905b120ef4733Andy Hung    uint32_t inCnl = audio_channel_count_from_out_mask(config->inputCfg.channels);
10429718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs    const webrtc::ProcessingConfig processing_config = {
10433f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs       {{static_cast<int>(session->apmSamplingRate), session->inChannelCount},
10443f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs        {static_cast<int>(session->apmSamplingRate), session->outChannelCount},
10453f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs        {static_cast<int>(session->apmSamplingRate), inCnl},
10463f11ef01574219f323a7f9ee4216d610733df50fAlex Luebs        {static_cast<int>(session->apmSamplingRate), inCnl}}};
10479718b7d9cba77c739269462ac57d43cc6ae69134Alex Luebs    int status = session->apm->Initialize(processing_config);
1048a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (status < 0) {
1049a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1050a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1051a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->revChannelCount = inCnl;
1052de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh    session->revFrame->num_channels_ = inCnl;
1053de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh    session->revFrame->sample_rate_hz_ = session->apmSamplingRate;
10543f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    // force process buffer reallocation
10553f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    session->revBufSize = 0;
10563f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent    session->framesRev = 0;
10573f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
1058a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
1059a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1060a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
10613d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurentvoid Session_GetReverseConfig(preproc_session_t *session, effect_config_t *config)
10623d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent{
10633d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    memset(config, 0, sizeof(effect_config_t));
10643d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    config->inputCfg.samplingRate = config->outputCfg.samplingRate = session->samplingRate;
10653d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    config->inputCfg.format = config->outputCfg.format = AUDIO_FORMAT_PCM_16_BIT;
10663d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    config->inputCfg.channels = config->outputCfg.channels =
1067ab334fd351ae5a0e18903da123d63e565b536874Glenn Kasten            audio_channel_in_mask_from_count(session->revChannelCount);
10683d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent    config->inputCfg.mask = config->outputCfg.mask =
10693d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            (EFFECT_CONFIG_SMP_RATE | EFFECT_CONFIG_CHANNELS | EFFECT_CONFIG_FORMAT);
10703d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent}
10713d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent
1072a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentvoid Session_SetProcEnabled(preproc_session_t *session, uint32_t procId, bool enabled)
1073a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1074a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (enabled) {
1075a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if(session->enabledMsk == 0) {
1076a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesIn = 0;
1077a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->inResampler != NULL) {
1078a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_reset_mem(session->inResampler);
1079a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1080a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesOut = 0;
1081a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->outResampler != NULL) {
1082a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_reset_mem(session->outResampler);
1083a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1084a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1085a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->enabledMsk |= (1 << procId);
1086a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (HasReverseStream(procId)) {
1087a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesRev = 0;
1088a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->revResampler != NULL) {
1089a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_reset_mem(session->revResampler);
1090a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1091a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->revEnabledMsk |= (1 << procId);
1092a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1093a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    } else {
1094a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->enabledMsk &= ~(1 << procId);
1095a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (HasReverseStream(procId)) {
1096a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->revEnabledMsk &= ~(1 << procId);
1097a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1098a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
10993856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Session_SetProcEnabled proc %d, enabled %d enabledMsk %08x revEnabledMsk %08x",
1100a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent         procId, enabled, session->enabledMsk, session->revEnabledMsk);
1101a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->processedMsk = 0;
1102a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (HasReverseStream(procId)) {
1103a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->revProcessedMsk = 0;
1104a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1105a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1106a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1107a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
1108a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Bundle functions
1109a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
1110a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1111a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic int sInitStatus = 1;
1112a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentstatic preproc_session_t sSessions[PREPROC_NUM_SESSIONS];
1113a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1114a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentpreproc_session_t *PreProc_GetSession(int32_t procId, int32_t  sessionId, int32_t  ioId)
1115a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1116a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t i;
1117a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int free = -1;
1118a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    for (i = 0; i < PREPROC_NUM_SESSIONS; i++) {
1119a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (sSessions[i].io == ioId) {
1120a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (sSessions[i].createdMsk & (1 << procId)) {
1121a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return NULL;
1122a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1123a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return &sSessions[i];
1124a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1125a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1126a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    for (i = 0; i < PREPROC_NUM_SESSIONS; i++) {
1127a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (sSessions[i].io == 0) {
1128a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            sSessions[i].id = sessionId;
1129a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            sSessions[i].io = ioId;
1130a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return &sSessions[i];
1131a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1132a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1133a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return NULL;
1134a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1135a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1136a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1137a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint PreProc_Init() {
1138a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t i;
1139a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status = 0;
1140a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1141a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (sInitStatus <= 0) {
1142a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return sInitStatus;
1143a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1144a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    for (i = 0; i < PREPROC_NUM_SESSIONS && status == 0; i++) {
1145a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        status = Session_Init(&sSessions[i]);
1146a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1147a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    sInitStatus = status;
1148a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return sInitStatus;
1149a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1150a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
11515e92a7861196ddae14638d4b7a63fc4892b7ef59Glenn Kastenconst effect_descriptor_t *PreProc_GetDescriptor(const effect_uuid_t *uuid)
1152a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1153a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    size_t i;
1154a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    for (i = 0; i < PREPROC_NUM_EFFECTS; i++) {
1155a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (memcmp(&sDescriptors[i]->uuid, uuid, sizeof(effect_uuid_t)) == 0) {
1156a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return sDescriptors[i];
1157a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1158a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1159a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return NULL;
1160a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1161a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1162a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1163a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentextern "C" {
1164a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1165a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
1166a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Effect Control Interface Implementation
1167a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
1168a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1169a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint PreProcessingFx_Process(effect_handle_t     self,
1170a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            audio_buffer_t    *inBuffer,
1171a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            audio_buffer_t    *outBuffer)
1172a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1173a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_effect_t * effect = (preproc_effect_t *)self;
1174a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int    status = 0;
1175a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1176a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (effect == NULL){
11773856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("PreProcessingFx_Process() ERROR effect == NULL");
1178a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1179a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1180a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_session_t * session = (preproc_session_t *)effect->session;
1181a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1182a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (inBuffer == NULL  || inBuffer->raw == NULL  ||
1183a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            outBuffer == NULL || outBuffer->raw == NULL){
11845ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("PreProcessingFx_Process() ERROR bad pointer");
1185a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1186a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1187a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1188a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->processedMsk |= (1<<effect->procId);
1189a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
11903856b090cd04ba5dd4a59a12430ed724d5995909Steve Block//    ALOGV("PreProcessingFx_Process In %d frames enabledMsk %08x processedMsk %08x",
1191a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//         inBuffer->frameCount, session->enabledMsk, session->processedMsk);
1192a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1193a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if ((session->processedMsk & session->enabledMsk) == session->enabledMsk) {
1194a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        effect->session->processedMsk = 0;
1195a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        size_t framesRq = outBuffer->frameCount;
1196a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        size_t framesWr = 0;
1197a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->framesOut) {
1198a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            size_t fr = session->framesOut;
1199a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (outBuffer->frameCount < fr) {
1200a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                fr = outBuffer->frameCount;
1201a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1202a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(outBuffer->s16,
1203a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                  session->outBuf,
1204a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                  fr * session->outChannelCount * sizeof(int16_t));
1205a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(session->outBuf,
1206a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                  session->outBuf + fr * session->outChannelCount,
1207a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                  (session->framesOut - fr) * session->outChannelCount * sizeof(int16_t));
1208a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesOut -= fr;
1209a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            framesWr += fr;
1210a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1211a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        outBuffer->frameCount = framesWr;
1212a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (framesWr == framesRq) {
1213a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            inBuffer->frameCount = 0;
1214a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return 0;
1215a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1216a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1217a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->inResampler != NULL) {
1218a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            size_t fr = session->frameCount - session->framesIn;
1219a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (inBuffer->frameCount < fr) {
1220a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                fr = inBuffer->frameCount;
1221a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1222a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->inBufSize < session->framesIn + fr) {
1223a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                session->inBufSize = session->framesIn + fr;
1224a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                session->inBuf = (int16_t *)realloc(session->inBuf,
1225a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                 session->inBufSize * session->inChannelCount * sizeof(int16_t));
1226a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1227a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(session->inBuf + session->framesIn * session->inChannelCount,
1228a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   inBuffer->s16,
1229a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   fr * session->inChannelCount * sizeof(int16_t));
12303f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#ifdef DUAL_MIC_TEST
12313f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_lock(&gPcmDumpLock);
12323f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (gPcmDumpFh != NULL) {
12333f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                fwrite(inBuffer->raw,
12343f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                       fr * session->inChannelCount * sizeof(int16_t), 1, gPcmDumpFh);
12353f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
12363f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_unlock(&gPcmDumpLock);
12373f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#endif
1238a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1239a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesIn += fr;
1240a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            inBuffer->frameCount = fr;
1241a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->framesIn < session->frameCount) {
1242a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return 0;
1243a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1244377b2ec9a2885f9b6405b07ba900a9e3f4349c38Kévin PETIT            spx_uint32_t frIn = session->framesIn;
1245377b2ec9a2885f9b6405b07ba900a9e3f4349c38Kévin PETIT            spx_uint32_t frOut = session->apmFrameCount;
1246a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->inChannelCount == 1) {
1247a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_process_int(session->inResampler,
1248a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            0,
1249a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            session->inBuf,
1250a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            &frIn,
1251de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh                                            session->procFrame->data_,
1252a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            &frOut);
1253a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            } else {
1254a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_process_interleaved_int(session->inResampler,
1255a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                        session->inBuf,
1256a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                        &frIn,
1257de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh                                                        session->procFrame->data_,
1258a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                        &frOut);
1259a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1260a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(session->inBuf,
1261a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   session->inBuf + frIn * session->inChannelCount,
1262a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   (session->framesIn - frIn) * session->inChannelCount * sizeof(int16_t));
1263a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesIn -= frIn;
1264a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        } else {
1265a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            size_t fr = session->frameCount - session->framesIn;
1266a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (inBuffer->frameCount < fr) {
1267a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                fr = inBuffer->frameCount;
1268a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1269de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh            memcpy(session->procFrame->data_ + session->framesIn * session->inChannelCount,
1270a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   inBuffer->s16,
1271a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   fr * session->inChannelCount * sizeof(int16_t));
12723f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
12733f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#ifdef DUAL_MIC_TEST
12743f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_lock(&gPcmDumpLock);
12753f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (gPcmDumpFh != NULL) {
12763f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                fwrite(inBuffer->raw,
12773f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                       fr * session->inChannelCount * sizeof(int16_t), 1, gPcmDumpFh);
12783f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
12793f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_unlock(&gPcmDumpLock);
12803f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#endif
12813f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
1282a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesIn += fr;
1283a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            inBuffer->frameCount = fr;
1284a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->framesIn < session->frameCount) {
1285a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return 0;
1286a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1287a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesIn = 0;
1288a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1289de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh        session->procFrame->samples_per_channel_ =
1290a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                session->apmFrameCount * session->inChannelCount;
1291a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1292a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        effect->session->apm->ProcessStream(session->procFrame);
1293a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1294a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->outBufSize < session->framesOut + session->frameCount) {
1295a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->outBufSize = session->framesOut + session->frameCount;
1296a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->outBuf = (int16_t *)realloc(session->outBuf,
1297a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                              session->outBufSize * session->outChannelCount * sizeof(int16_t));
1298a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1299a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1300a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->outResampler != NULL) {
1301377b2ec9a2885f9b6405b07ba900a9e3f4349c38Kévin PETIT            spx_uint32_t frIn = session->apmFrameCount;
1302377b2ec9a2885f9b6405b07ba900a9e3f4349c38Kévin PETIT            spx_uint32_t frOut = session->frameCount;
1303a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->inChannelCount == 1) {
1304a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_process_int(session->outResampler,
1305a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    0,
1306de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh                                    session->procFrame->data_,
1307a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    &frIn,
1308a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    session->outBuf + session->framesOut * session->outChannelCount,
1309a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    &frOut);
1310a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            } else {
1311a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_process_interleaved_int(session->outResampler,
1312de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh                                    session->procFrame->data_,
1313a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    &frIn,
1314a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    session->outBuf + session->framesOut * session->outChannelCount,
1315a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                    &frOut);
1316a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1317a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesOut += frOut;
1318a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        } else {
1319a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(session->outBuf + session->framesOut * session->outChannelCount,
1320de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh                   session->procFrame->data_,
1321a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   session->frameCount * session->outChannelCount * sizeof(int16_t));
1322a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesOut += session->frameCount;
1323a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1324a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        size_t fr = session->framesOut;
1325a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (framesRq - framesWr < fr) {
1326a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            fr = framesRq - framesWr;
1327a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1328a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        memcpy(outBuffer->s16 + framesWr * session->outChannelCount,
1329a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent              session->outBuf,
1330a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent              fr * session->outChannelCount * sizeof(int16_t));
1331a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        memcpy(session->outBuf,
1332a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent              session->outBuf + fr * session->outChannelCount,
1333a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent              (session->framesOut - fr) * session->outChannelCount * sizeof(int16_t));
1334a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->framesOut -= fr;
1335a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        outBuffer->frameCount += fr;
1336a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1337a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return 0;
1338a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    } else {
1339a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -ENODATA;
1340a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1341a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1342a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1343a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint PreProcessingFx_Command(effect_handle_t  self,
1344a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            uint32_t            cmdCode,
1345a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            uint32_t            cmdSize,
1346a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            void                *pCmdData,
1347a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            uint32_t            *replySize,
1348a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            void                *pReplyData)
1349a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1350a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_effect_t * effect = (preproc_effect_t *) self;
1351a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int retsize;
1352a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status;
1353a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1354a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (effect == NULL){
1355a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1356a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1357a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
13583856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    //ALOGV("PreProcessingFx_Command: command %d cmdSize %d",cmdCode, cmdSize);
1359a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1360a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    switch (cmdCode){
1361a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_INIT:
1362a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (pReplyData == NULL || *replySize != sizeof(int)){
1363a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1364a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1365a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (effect->ops->init) {
1366a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                effect->ops->init(effect);
1367a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1368a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *(int *)pReplyData = 0;
1369a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
1370a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
13713f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case EFFECT_CMD_SET_CONFIG: {
1372a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (pCmdData    == NULL||
1373a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                cmdSize     != sizeof(effect_config_t)||
1374a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                pReplyData  == NULL||
1375a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                *replySize  != sizeof(int)){
13763856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: "
13773d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                        "EFFECT_CMD_SET_CONFIG: ERROR");
1378a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1379a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
13803f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#ifdef DUAL_MIC_TEST
13813f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            // make sure that the config command is accepted by making as if all effects were
13823f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            // disabled: this is OK for functional tests
13833f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            uint32_t enabledMsk = effect->session->enabledMsk;
13843f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (gDualMicEnabled) {
13853f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                effect->session->enabledMsk = 0;
13863f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
13873f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#endif
1388a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *(int *)pReplyData = Session_SetConfig(effect->session, (effect_config_t *)pCmdData);
13893f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#ifdef DUAL_MIC_TEST
13903f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (gDualMicEnabled) {
13913f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                effect->session->enabledMsk = enabledMsk;
13923f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
13933f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#endif
1394a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (*(int *)pReplyData != 0) {
1395a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                break;
1396a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
139776533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent            if (effect->state != PREPROC_EFFECT_STATE_ACTIVE) {
139876533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent                *(int *)pReplyData = Effect_SetState(effect, PREPROC_EFFECT_STATE_CONFIG);
139976533e94500e5d221ec4a1a5d9c97ea3d75860e4Eric Laurent            }
14003f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } break;
1401a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
14023d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent        case EFFECT_CMD_GET_CONFIG:
14033d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            if (pReplyData == NULL ||
14043d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                *replySize != sizeof(effect_config_t)) {
14053d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                ALOGV("\tLVM_ERROR : PreProcessingFx_Command cmdCode Case: "
14063d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                        "EFFECT_CMD_GET_CONFIG: ERROR");
14073d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                return -EINVAL;
14083d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            }
14093d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent
141094fef38613962c983c46ab0b97c58f5373412d2dEric Laurent            Session_GetConfig(effect->session, (effect_config_t *)pReplyData);
14113d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            break;
14123d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent
14133d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent        case EFFECT_CMD_SET_CONFIG_REVERSE:
14143d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            if (pCmdData == NULL ||
14153d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                cmdSize != sizeof(effect_config_t) ||
14163d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                pReplyData == NULL ||
14173d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                *replySize != sizeof(int)) {
14183856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: "
14193d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                        "EFFECT_CMD_SET_CONFIG_REVERSE: ERROR");
1420a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1421a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1422a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *(int *)pReplyData = Session_SetReverseConfig(effect->session,
1423a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                          (effect_config_t *)pCmdData);
1424a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (*(int *)pReplyData != 0) {
1425a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                break;
1426a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1427a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
1428a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
14293d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent        case EFFECT_CMD_GET_CONFIG_REVERSE:
14303d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            if (pReplyData == NULL ||
14313d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                *replySize != sizeof(effect_config_t)){
14323d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                ALOGV("PreProcessingFx_Command cmdCode Case: "
14333d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                        "EFFECT_CMD_GET_CONFIG_REVERSE: ERROR");
14343d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent                return -EINVAL;
14353d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            }
14363d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            Session_GetReverseConfig(effect->session, (effect_config_t *)pCmdData);
14373d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent            break;
14383d5188bd6abe55898f10a0edf3c05aff8aa2ef67Eric Laurent
1439a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_RESET:
1440a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (effect->ops->reset) {
1441a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                effect->ops->reset(effect);
1442a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1443a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
1444a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
14450f714a464d2425afe00d6450535e763131b40844Eric Laurent        case EFFECT_CMD_GET_PARAM: {
14460f714a464d2425afe00d6450535e763131b40844Eric Laurent            effect_param_t *p = (effect_param_t *)pCmdData;
14470f714a464d2425afe00d6450535e763131b40844Eric Laurent
14480f714a464d2425afe00d6450535e763131b40844Eric Laurent            if (pCmdData == NULL || cmdSize < sizeof(effect_param_t) ||
14490f714a464d2425afe00d6450535e763131b40844Eric Laurent                    cmdSize < (sizeof(effect_param_t) + p->psize) ||
14500f714a464d2425afe00d6450535e763131b40844Eric Laurent                    pReplyData == NULL || replySize == NULL ||
14510f714a464d2425afe00d6450535e763131b40844Eric Laurent                    *replySize < (sizeof(effect_param_t) + p->psize)){
14523856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: "
1453a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                        "EFFECT_CMD_GET_PARAM: ERROR");
1454a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1455a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1456a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1457a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(pReplyData, pCmdData, sizeof(effect_param_t) + p->psize);
1458a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1459a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            p = (effect_param_t *)pReplyData;
1460a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1461a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            int voffset = ((p->psize - 1) / sizeof(int32_t) + 1) * sizeof(int32_t);
1462a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1463a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (effect->ops->get_parameter) {
1464a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                p->status = effect->ops->get_parameter(effect, p->data,
1465b302bd5d288be2d3363b80053ca2392560b00b25Ashok Bhat                                                       &p->vsize,
1466a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                       p->data + voffset);
1467a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                *replySize = sizeof(effect_param_t) + voffset + p->vsize;
1468a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1469a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        } break;
1470a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1471a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_SET_PARAM:{
1472a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (pCmdData == NULL||
14730f714a464d2425afe00d6450535e763131b40844Eric Laurent                    cmdSize < sizeof(effect_param_t) ||
14740f714a464d2425afe00d6450535e763131b40844Eric Laurent                    pReplyData == NULL || replySize == NULL ||
1475a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                    *replySize != sizeof(int32_t)){
14763856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: "
1477a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                        "EFFECT_CMD_SET_PARAM: ERROR");
1478a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1479a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1480a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            effect_param_t *p = (effect_param_t *) pCmdData;
1481a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1482a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (p->psize != sizeof(int32_t)){
14833856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: "
1484a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                        "EFFECT_CMD_SET_PARAM: ERROR, psize is not sizeof(int32_t)");
1485a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1486a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1487a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (effect->ops->set_parameter) {
1488a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                *(int *)pReplyData = effect->ops->set_parameter(effect,
1489a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                                (void *)p->data,
1490a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                                p->data + p->psize);
1491a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1492a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        } break;
1493a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1494a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_ENABLE:
14950f714a464d2425afe00d6450535e763131b40844Eric Laurent            if (pReplyData == NULL || replySize == NULL || *replySize != sizeof(int)){
14963856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: EFFECT_CMD_ENABLE: ERROR");
1497a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1498a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1499a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *(int *)pReplyData = Effect_SetState(effect, PREPROC_EFFECT_STATE_ACTIVE);
1500a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
1501a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1502a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_DISABLE:
15030f714a464d2425afe00d6450535e763131b40844Eric Laurent            if (pReplyData == NULL || replySize == NULL || *replySize != sizeof(int)){
15043856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: EFFECT_CMD_DISABLE: ERROR");
1505a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1506a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1507a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            *(int *)pReplyData  = Effect_SetState(effect, PREPROC_EFFECT_STATE_CONFIG);
1508a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
1509a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1510a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_SET_DEVICE:
1511a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_SET_INPUT_DEVICE:
1512a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (pCmdData == NULL ||
1513a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                cmdSize != sizeof(uint32_t)) {
15143856b090cd04ba5dd4a59a12430ed724d5995909Steve Block                ALOGV("PreProcessingFx_Command cmdCode Case: EFFECT_CMD_SET_DEVICE: ERROR");
1515a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return -EINVAL;
1516a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1517a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1518a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (effect->ops->set_device) {
1519a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                effect->ops->set_device(effect, *(uint32_t *)pCmdData);
1520a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1521a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
1522a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1523a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_SET_VOLUME:
1524a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        case EFFECT_CMD_SET_AUDIO_MODE:
1525a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            break;
1526a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
15273f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#ifdef DUAL_MIC_TEST
15283f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        ///// test commands start
15293f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case PREPROC_CMD_DUAL_MIC_ENABLE: {
15303f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (pCmdData == NULL|| cmdSize != sizeof(uint32_t) ||
15313f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    pReplyData == NULL || replySize == NULL) {
15323f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGE("PreProcessingFx_Command cmdCode Case: "
15333f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "PREPROC_CMD_DUAL_MIC_ENABLE: ERROR");
15343f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *replySize = 0;
15353f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
15363f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
15373f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            gDualMicEnabled = *(bool *)pCmdData;
15383f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (gDualMicEnabled) {
15393f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                effect->aux_channels_on = sHasAuxChannels[effect->procId];
15403f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } else {
15413f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                effect->aux_channels_on = false;
15423f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
15433f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            effect->cur_channel_config = (effect->session->inChannelCount == 1) ?
15443f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    CHANNEL_CFG_MONO : CHANNEL_CFG_STEREO;
15453f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
15463f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            ALOGV("PREPROC_CMD_DUAL_MIC_ENABLE: %s", gDualMicEnabled ? "enabled" : "disabled");
15473f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *replySize = sizeof(int);
15483f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *(int *)pReplyData = 0;
15493f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } break;
15503f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case PREPROC_CMD_DUAL_MIC_PCM_DUMP_START: {
15513f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (pCmdData == NULL|| pReplyData == NULL || replySize == NULL) {
15523f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGE("PreProcessingFx_Command cmdCode Case: "
15533f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "PREPROC_CMD_DUAL_MIC_PCM_DUMP_START: ERROR");
15543f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *replySize = 0;
15553f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
15563f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
15573f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_lock(&gPcmDumpLock);
15583f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (gPcmDumpFh != NULL) {
15593f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                fclose(gPcmDumpFh);
15603f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                gPcmDumpFh = NULL;
15613f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
15623f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            char *path = strndup((char *)pCmdData, cmdSize);
15633f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            gPcmDumpFh = fopen((char *)path, "wb");
15643f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_unlock(&gPcmDumpLock);
15653f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            ALOGV("PREPROC_CMD_DUAL_MIC_PCM_DUMP_START: path %s gPcmDumpFh %p",
15663f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                  path, gPcmDumpFh);
15673f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            ALOGE_IF(gPcmDumpFh <= 0, "gPcmDumpFh open error %d %s", errno, strerror(errno));
15683f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            free(path);
15693f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *replySize = sizeof(int);
15703f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *(int *)pReplyData = 0;
15713f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } break;
15723f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case PREPROC_CMD_DUAL_MIC_PCM_DUMP_STOP: {
15733f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (pReplyData == NULL || replySize == NULL) {
15743f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGE("PreProcessingFx_Command cmdCode Case: "
15753f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "PREPROC_CMD_DUAL_MIC_PCM_DUMP_STOP: ERROR");
15763f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *replySize = 0;
15773f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
15783f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
15793f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_lock(&gPcmDumpLock);
15803f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (gPcmDumpFh != NULL) {
15813f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                fclose(gPcmDumpFh);
15823f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                gPcmDumpFh = NULL;
15833f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
15843f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            pthread_mutex_unlock(&gPcmDumpLock);
15853f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            ALOGV("PREPROC_CMD_DUAL_MIC_PCM_DUMP_STOP");
15863f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *replySize = sizeof(int);
15873f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *(int *)pReplyData = 0;
15883f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } break;
15893f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        ///// test commands end
15903f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
15913f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case EFFECT_CMD_GET_FEATURE_SUPPORTED_CONFIGS: {
15923f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if(!gDualMicEnabled) {
15933f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
15943f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
15953f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (pCmdData == NULL|| cmdSize != 2 * sizeof(uint32_t) ||
15963f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    pReplyData == NULL || replySize == NULL) {
15973f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGE("PreProcessingFx_Command cmdCode Case: "
15983f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "EFFECT_CMD_GET_FEATURE_SUPPORTED_CONFIGS: ERROR");
15993f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *replySize = 0;
16003f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
16013f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16023f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (*(uint32_t *)pCmdData != EFFECT_FEATURE_AUX_CHANNELS ||
16033f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                  !effect->aux_channels_on) {
16043f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGV("PreProcessingFx_Command feature EFFECT_FEATURE_AUX_CHANNELS not supported by"
16053f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        " fx %d", effect->procId);
16063f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *(uint32_t *)pReplyData = -ENOSYS;
16073f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *replySize = sizeof(uint32_t);
16083f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                break;
16093f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16103f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            size_t num_configs = *((uint32_t *)pCmdData + 1);
16113f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (*replySize < (2 * sizeof(uint32_t) +
16123f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                              num_configs * sizeof(channel_config_t))) {
16133f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *replySize = 0;
16143f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
16153f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16163f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
16173f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *((uint32_t *)pReplyData + 1) = CHANNEL_CFG_CNT;
16183f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (num_configs < CHANNEL_CFG_CNT ||
16193f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    *replySize < (2 * sizeof(uint32_t) +
16203f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                                     CHANNEL_CFG_CNT * sizeof(channel_config_t))) {
16213f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *(uint32_t *)pReplyData = -ENOMEM;
16223f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } else {
16233f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                num_configs = CHANNEL_CFG_CNT;
16243f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *(uint32_t *)pReplyData = 0;
16253f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16263f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            ALOGV("PreProcessingFx_Command EFFECT_CMD_GET_FEATURE_SUPPORTED_CONFIGS num config %d",
16273f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                  num_configs);
16283f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent
16293f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *replySize = 2 * sizeof(uint32_t) + num_configs * sizeof(channel_config_t);
16303f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *((uint32_t *)pReplyData + 1) = num_configs;
16313f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            memcpy((uint32_t *)pReplyData + 2, &sDualMicConfigs, num_configs * sizeof(channel_config_t));
16323f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } break;
16333f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case EFFECT_CMD_GET_FEATURE_CONFIG:
16343f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if(!gDualMicEnabled) {
16353f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
16363f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16373f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (pCmdData == NULL|| cmdSize != sizeof(uint32_t) ||
16383f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    pReplyData == NULL || replySize == NULL ||
16393f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    *replySize < sizeof(uint32_t) + sizeof(channel_config_t)) {
16403f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGE("PreProcessingFx_Command cmdCode Case: "
16413f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "EFFECT_CMD_GET_FEATURE_CONFIG: ERROR");
16423f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
16433f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16443f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (*(uint32_t *)pCmdData != EFFECT_FEATURE_AUX_CHANNELS || !effect->aux_channels_on) {
16453f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *(uint32_t *)pReplyData = -ENOSYS;
16463f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *replySize = sizeof(uint32_t);
16473f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                break;
16483f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16493f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            ALOGV("PreProcessingFx_Command EFFECT_CMD_GET_FEATURE_CONFIG");
16503f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *(uint32_t *)pReplyData = 0;
16513f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *replySize = sizeof(uint32_t) + sizeof(channel_config_t);
16523f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            memcpy((uint32_t *)pReplyData + 1,
16533f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                   &sDualMicConfigs[effect->cur_channel_config],
16543f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                   sizeof(channel_config_t));
16553f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            break;
16563f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent        case EFFECT_CMD_SET_FEATURE_CONFIG: {
16573f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            ALOGV("PreProcessingFx_Command EFFECT_CMD_SET_FEATURE_CONFIG: "
16583f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    "gDualMicEnabled %d effect->aux_channels_on %d",
16593f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                  gDualMicEnabled, effect->aux_channels_on);
16603f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if(!gDualMicEnabled) {
16613f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
16623f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16633f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (pCmdData == NULL|| cmdSize != (sizeof(uint32_t) + sizeof(channel_config_t)) ||
16643f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    pReplyData == NULL || replySize == NULL ||
16653f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    *replySize < sizeof(uint32_t)) {
16663f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGE("PreProcessingFx_Command cmdCode Case: "
16673f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "EFFECT_CMD_SET_FEATURE_CONFIG: ERROR\n"
16683f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "pCmdData %p cmdSize %d pReplyData %p replySize %p *replySize %d",
16693f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        pCmdData, cmdSize, pReplyData, replySize, replySize ? *replySize : -1);
16703f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                return -EINVAL;
16713f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16723f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            *replySize = sizeof(uint32_t);
16733f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (*(uint32_t *)pCmdData != EFFECT_FEATURE_AUX_CHANNELS || !effect->aux_channels_on) {
16743f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *(uint32_t *)pReplyData = -ENOSYS;
16753f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGV("PreProcessingFx_Command cmdCode Case: "
16763f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                                        "EFFECT_CMD_SET_FEATURE_CONFIG: ERROR\n"
16773f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                                        "CmdData %d effect->aux_channels_on %d",
16783f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                                        *(uint32_t *)pCmdData, effect->aux_channels_on);
16793f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                break;
16803f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16813f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            size_t i;
16823f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            for (i = 0; i < CHANNEL_CFG_CNT;i++) {
16833f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                if (memcmp((uint32_t *)pCmdData + 1,
16843f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                           &sDualMicConfigs[i], sizeof(channel_config_t)) == 0) {
16853f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                    break;
16863f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                }
16873f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16883f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            if (i == CHANNEL_CFG_CNT) {
16893f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *(uint32_t *)pReplyData = -EINVAL;
16903f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGW("PreProcessingFx_Command EFFECT_CMD_SET_FEATURE_CONFIG invalid config"
16913f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "[%08x].[%08x]", *((uint32_t *)pCmdData + 1), *((uint32_t *)pCmdData + 2));
16923f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } else {
16933f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                effect->cur_channel_config = i;
16943f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                *(uint32_t *)pReplyData = 0;
16953f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                ALOGV("PreProcessingFx_Command EFFECT_CMD_SET_FEATURE_CONFIG New config"
16963f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent                        "[%08x].[%08x]", sDualMicConfigs[i].main_channels, sDualMicConfigs[i].aux_channels);
16973f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            }
16983f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent            } break;
16993f9c84c0a5af83fceb8669390e2d71b75ec7b550Eric Laurent#endif
1700a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        default:
1701a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            return -EINVAL;
1702a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1703a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
1704a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1705a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1706a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1707a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint PreProcessingFx_GetDescriptor(effect_handle_t   self,
1708a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                  effect_descriptor_t *pDescriptor)
1709a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1710a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_effect_t * effect = (preproc_effect_t *) self;
1711a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1712a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (effect == NULL || pDescriptor == NULL) {
1713a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1714a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1715a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1716a189a6883ee55cf62da1d7bf5bf5a8ab501938a4Glenn Kasten    *pDescriptor = *sDescriptors[effect->procId];
1717a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1718a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
1719a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1720a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1721a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint PreProcessingFx_ProcessReverse(effect_handle_t     self,
1722a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                   audio_buffer_t    *inBuffer,
17230f714a464d2425afe00d6450535e763131b40844Eric Laurent                                   audio_buffer_t    *outBuffer __unused)
1724a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1725a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_effect_t * effect = (preproc_effect_t *)self;
1726a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int    status = 0;
1727a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1728a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (effect == NULL){
17295ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("PreProcessingFx_ProcessReverse() ERROR effect == NULL");
1730a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1731a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1732a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_session_t * session = (preproc_session_t *)effect->session;
1733a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1734a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (inBuffer == NULL  || inBuffer->raw == NULL){
17355ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("PreProcessingFx_ProcessReverse() ERROR bad pointer");
1736a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1737a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1738a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1739a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session->revProcessedMsk |= (1<<effect->procId);
1740a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
17413856b090cd04ba5dd4a59a12430ed724d5995909Steve Block//    ALOGV("PreProcessingFx_ProcessReverse In %d frames revEnabledMsk %08x revProcessedMsk %08x",
1742a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//         inBuffer->frameCount, session->revEnabledMsk, session->revProcessedMsk);
1743a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1744a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1745a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if ((session->revProcessedMsk & session->revEnabledMsk) == session->revEnabledMsk) {
1746a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        effect->session->revProcessedMsk = 0;
1747a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        if (session->revResampler != NULL) {
1748a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            size_t fr = session->frameCount - session->framesRev;
1749a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (inBuffer->frameCount < fr) {
1750a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                fr = inBuffer->frameCount;
1751a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1752a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->revBufSize < session->framesRev + fr) {
1753a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                session->revBufSize = session->framesRev + fr;
1754a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                session->revBuf = (int16_t *)realloc(session->revBuf,
1755a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                  session->revBufSize * session->inChannelCount * sizeof(int16_t));
1756a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1757a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(session->revBuf + session->framesRev * session->inChannelCount,
1758a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   inBuffer->s16,
1759a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   fr * session->inChannelCount * sizeof(int16_t));
1760a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1761a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesRev += fr;
1762a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            inBuffer->frameCount = fr;
1763a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->framesRev < session->frameCount) {
1764a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return 0;
1765a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1766377b2ec9a2885f9b6405b07ba900a9e3f4349c38Kévin PETIT            spx_uint32_t frIn = session->framesRev;
1767377b2ec9a2885f9b6405b07ba900a9e3f4349c38Kévin PETIT            spx_uint32_t frOut = session->apmFrameCount;
1768a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->inChannelCount == 1) {
1769a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_process_int(session->revResampler,
1770a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            0,
1771a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            session->revBuf,
1772a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            &frIn,
1773de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh                                            session->revFrame->data_,
1774a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                            &frOut);
1775a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            } else {
1776a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                speex_resampler_process_interleaved_int(session->revResampler,
1777a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                        session->revBuf,
1778a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                        &frIn,
1779de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh                                                        session->revFrame->data_,
1780a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                                        &frOut);
1781a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1782a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            memcpy(session->revBuf,
1783a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   session->revBuf + frIn * session->inChannelCount,
1784a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   (session->framesRev - frIn) * session->inChannelCount * sizeof(int16_t));
1785a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesRev -= frIn;
1786a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        } else {
1787a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            size_t fr = session->frameCount - session->framesRev;
1788a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (inBuffer->frameCount < fr) {
1789a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                fr = inBuffer->frameCount;
1790a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1791de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh            memcpy(session->revFrame->data_ + session->framesRev * session->inChannelCount,
1792a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   inBuffer->s16,
1793a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                   fr * session->inChannelCount * sizeof(int16_t));
1794a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesRev += fr;
1795a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            inBuffer->frameCount = fr;
1796a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            if (session->framesRev < session->frameCount) {
1797a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                return 0;
1798a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            }
1799a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent            session->framesRev = 0;
1800a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        }
1801de7fa31668bd90dba80ff2c0c462492fed873c3dChih-Hung Hsieh        session->revFrame->samples_per_channel_ =
1802a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                session->apmFrameCount * session->inChannelCount;
1803a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        effect->session->apm->AnalyzeReverseStream(session->revFrame);
1804a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return 0;
1805a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    } else {
1806a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -ENODATA;
1807a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1808a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1809a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1810a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1811a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// effect_handle_t interface implementation for effect
1812a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentconst struct effect_interface_s sEffectInterface = {
1813a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PreProcessingFx_Process,
1814a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PreProcessingFx_Command,
1815a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PreProcessingFx_GetDescriptor,
1816a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    NULL
1817a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
1818a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1819a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentconst struct effect_interface_s sEffectInterfaceReverse = {
1820a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PreProcessingFx_Process,
1821a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PreProcessingFx_Command,
1822a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PreProcessingFx_GetDescriptor,
1823a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    PreProcessingFx_ProcessReverse
1824a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
1825a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1826a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
1827a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent// Effect Library Interface Implementation
1828a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent//------------------------------------------------------------------------------
1829a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
18305e92a7861196ddae14638d4b7a63fc4892b7ef59Glenn Kastenint PreProcessingLib_Create(const effect_uuid_t *uuid,
1831a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            int32_t             sessionId,
1832a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            int32_t             ioId,
1833a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                            effect_handle_t  *pInterface)
1834a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
18353856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("EffectCreate: uuid: %08x session %d IO: %d", uuid->timeLow, sessionId, ioId);
1836a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1837a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status;
1838a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    const effect_descriptor_t *desc;
1839a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_session_t *session;
1840a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    uint32_t procId;
1841a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1842a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (PreProc_Init() != 0) {
1843a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return sInitStatus;
1844a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1845a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    desc =  PreProc_GetDescriptor(uuid);
1846a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (desc == NULL) {
18475ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("EffectCreate: fx not found uuid: %08x", uuid->timeLow);
1848a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1849a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1850a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    procId = UuidToProcId(&desc->type);
1851a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1852a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    session = PreProc_GetSession(procId, sessionId, ioId);
1853a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (session == NULL) {
18545ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block        ALOGW("EffectCreate: no more session available");
1855a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1856a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1857a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1858a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    status = Session_CreateEffect(session, procId, pInterface);
1859a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1860a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (status < 0 && session->createdMsk == 0) {
1861a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        session->io = 0;
1862a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1863a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return status;
1864a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1865a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1866a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentint PreProcessingLib_Release(effect_handle_t interface)
1867a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent{
1868a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    int status;
18693856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("EffectRelease start %p", interface);
1870a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (PreProc_Init() != 0) {
1871a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return sInitStatus;
1872a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1873a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1874a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    preproc_effect_t *fx = (preproc_effect_t *)interface;
1875a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1876a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (fx->session->io == 0) {
1877a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1878a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1879a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return Session_ReleaseEffect(fx->session, fx);
1880a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1881a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
18825e92a7861196ddae14638d4b7a63fc4892b7ef59Glenn Kastenint PreProcessingLib_GetDescriptor(const effect_uuid_t *uuid,
1883a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent                                   effect_descriptor_t *pDescriptor) {
1884a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1885a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (pDescriptor == NULL || uuid == NULL){
1886a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return -EINVAL;
1887a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1888a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1889a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    const effect_descriptor_t *desc = PreProc_GetDescriptor(uuid);
1890a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    if (desc == NULL) {
18913856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("PreProcessingLib_GetDescriptor() not found");
1892a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent        return  -EINVAL;
1893a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    }
1894a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
18953856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("PreProcessingLib_GetDescriptor() got fx %s", desc->name);
1896a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1897a189a6883ee55cf62da1d7bf5bf5a8ab501938a4Glenn Kasten    *pDescriptor = *desc;
1898a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent    return 0;
1899a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}
1900a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
19017f16b197c76fbae9399242f055a7ee16dcd0fd6dMarco Nelissen// This is the only symbol that needs to be exported
19027f16b197c76fbae9399242f055a7ee16dcd0fd6dMarco Nelissen__attribute__ ((visibility ("default")))
1903a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurentaudio_effect_library_t AUDIO_EFFECT_LIBRARY_INFO_SYM = {
1904c9d8ea7f8f9a1ca8ecd266695e3cac423790b2f9synergydev    .tag = AUDIO_EFFECT_LIBRARY_TAG,
1905c9d8ea7f8f9a1ca8ecd266695e3cac423790b2f9synergydev    .version = EFFECT_LIBRARY_API_VERSION,
1906c9d8ea7f8f9a1ca8ecd266695e3cac423790b2f9synergydev    .name = "Audio Preprocessing Library",
1907c9d8ea7f8f9a1ca8ecd266695e3cac423790b2f9synergydev    .implementor = "The Android Open Source Project",
1908c9d8ea7f8f9a1ca8ecd266695e3cac423790b2f9synergydev    .create_effect = PreProcessingLib_Create,
1909c9d8ea7f8f9a1ca8ecd266695e3cac423790b2f9synergydev    .release_effect = PreProcessingLib_Release,
1910c9d8ea7f8f9a1ca8ecd266695e3cac423790b2f9synergydev    .get_descriptor = PreProcessingLib_GetDescriptor
1911a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent};
1912a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent
1913a9390d4d571817cdb1d659b4b22b04130dc77a48Eric Laurent}; // extern "C"
1914