1 /* AudioStreamInALSA.cpp
2  **
3  ** Copyright 2008-2009 Wind River Systems
4  ** Copyright (c) 2011-2012, Code Aurora Forum. All rights reserved.
5  **
6  ** Licensed under the Apache License, Version 2.0 (the "License");
7  ** you may not use this file except in compliance with the License.
8  ** You may obtain a copy of the License at
9  **
10  **     http://www.apache.org/licenses/LICENSE-2.0
11  **
12  ** Unless required by applicable law or agreed to in writing, software
13  ** distributed under the License is distributed on an "AS IS" BASIS,
14  ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  ** See the License for the specific language governing permissions and
16  ** limitations under the License.
17  */
18 
19 #include <errno.h>
20 #include <stdarg.h>
21 #include <sys/stat.h>
22 #include <fcntl.h>
23 #include <stdlib.h>
24 #include <unistd.h>
25 #include <dlfcn.h>
26 
27 #define LOG_TAG "AudioStreamInALSA"
28 //#define LOG_NDEBUG 0
29 #define LOG_NDDEBUG 0
30 #include <utils/Log.h>
31 #include <utils/String8.h>
32 
33 #include <cutils/properties.h>
34 #include <media/AudioRecord.h>
35 #include <hardware_legacy/power.h>
36 
37 #include "AudioHardwareALSA.h"
38 
39 extern "C" {
40 #ifdef QCOM_CSDCLIENT_ENABLED
41 static int (*csd_start_record)(int);
42 static int (*csd_stop_record)(void);
43 #endif
44 
45 #ifdef QCOM_SSR_ENABLED
46 #include "surround_filters_interface.h"
47 #endif
48 }
49 
50 namespace android_audio_legacy
51 {
52 #ifdef QCOM_SSR_ENABLED
53 #define SURROUND_FILE_1R "/system/etc/surround_sound/filter1r.pcm"
54 #define SURROUND_FILE_2R "/system/etc/surround_sound/filter2r.pcm"
55 #define SURROUND_FILE_3R "/system/etc/surround_sound/filter3r.pcm"
56 #define SURROUND_FILE_4R "/system/etc/surround_sound/filter4r.pcm"
57 
58 #define SURROUND_FILE_1I "/system/etc/surround_sound/filter1i.pcm"
59 #define SURROUND_FILE_2I "/system/etc/surround_sound/filter2i.pcm"
60 #define SURROUND_FILE_3I "/system/etc/surround_sound/filter3i.pcm"
61 #define SURROUND_FILE_4I "/system/etc/surround_sound/filter4i.pcm"
62 
63 // Use AAC/DTS channel mapping as default channel mapping: C,FL,FR,Ls,Rs,LFE
64 const int chanMap[] = { 1, 2, 4, 3, 0, 5 };
65 #endif
66 
67 AudioStreamInALSA::AudioStreamInALSA(AudioHardwareALSA *parent,
68         alsa_handle_t *handle,
69         AudioSystem::audio_in_acoustics audio_acoustics) :
70     ALSAStreamOps(parent, handle),
71     mFramesLost(0),
72     mAcoustics(audio_acoustics),
73     mParent(parent)
74 #ifdef QCOM_SSR_ENABLED
75     , mFp_4ch(NULL),
76     mFp_6ch(NULL),
77     mRealCoeffs(NULL),
78     mImagCoeffs(NULL),
79     mSurroundObj(NULL),
80     mSurroundOutputBuffer(NULL),
81     mSurroundInputBuffer(NULL),
82     mSurroundOutputBufferIdx(0),
83     mSurroundInputBufferIdx(0)
84 #endif
85 {
86 #ifdef QCOM_SSR_ENABLED
87     char c_multi_ch_dump[128] = {0};
88     status_t err = NO_ERROR;
89 
90     // Call surround sound library init if device is Surround Sound
91     if ( handle->channels == 6) {
92         if (!strncmp(handle->useCase, SND_USE_CASE_VERB_HIFI_REC, strlen(SND_USE_CASE_VERB_HIFI_REC))
93             || !strncmp(handle->useCase, SND_USE_CASE_MOD_CAPTURE_MUSIC, strlen(SND_USE_CASE_MOD_CAPTURE_MUSIC))) {
94 
95             err = initSurroundSoundLibrary(handle->bufferSize);
96             if ( NO_ERROR != err) {
97                 ALOGE("initSurroundSoundLibrary failed: %d  handle->bufferSize:%d", err,handle->bufferSize);
98             }
99 
100             property_get("ssr.pcmdump",c_multi_ch_dump,"0");
101             if (0 == strncmp("true",c_multi_ch_dump, sizeof("ssr.dump-pcm"))) {
102                 //Remember to change file system permission of data(e.g. chmod 777 data/),
103                 //otherwise, fopen may fail.
104                 if ( !mFp_4ch)
105                     mFp_4ch = fopen("/data/4ch_ssr.pcm", "wb");
106                 if ( !mFp_6ch)
107                     mFp_6ch = fopen("/data/6ch_ssr.pcm", "wb");
108                 if ((!mFp_4ch) || (!mFp_6ch))
109                     ALOGE("mfp_4ch or mfp_6ch open failed: mfp_4ch:%p mfp_6ch:%p",mFp_4ch,mFp_6ch);
110             }
111         }
112     }
113 #endif
114 }
115 
116 AudioStreamInALSA::~AudioStreamInALSA()
117 {
118     close();
119 }
120 
121 status_t AudioStreamInALSA::setGain(float gain)
122 {
123     return 0; //mixer() ? mixer()->setMasterGain(gain) : (status_t)NO_INIT;
124 }
125 
126 ssize_t AudioStreamInALSA::read(void *buffer, ssize_t bytes)
127 {
128     int period_size;
129 
130     ALOGV("read:: buffer %p, bytes %d", buffer, bytes);
131 
132     int n;
133     status_t          err;
134     ssize_t            read = 0;
135     char *use_case;
136     int newMode = mParent->mode();
137 
138     if((mHandle->handle == NULL) && (mHandle->rxHandle == NULL) &&
139          (strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) &&
140          (strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) {
141         mParent->mLock.lock();
142         snd_use_case_get(mHandle->ucMgr, "_verb", (const char **)&use_case);
143         if ((use_case != NULL) && (strcmp(use_case, SND_USE_CASE_VERB_INACTIVE))) {
144             if ((mHandle->devices == AudioSystem::DEVICE_IN_VOICE_CALL) &&
145                 (newMode == AudioSystem::MODE_IN_CALL)) {
146                 ALOGD("read:: mParent->mIncallMode=%d", mParent->mIncallMode);
147                 if ((mParent->mIncallMode & AudioSystem::CHANNEL_IN_VOICE_UPLINK) &&
148                     (mParent->mIncallMode & AudioSystem::CHANNEL_IN_VOICE_DNLINK)) {
149 #ifdef QCOM_CSDCLIENT_ENABLED
150                     if (mParent->mFusion3Platform) {
151                         mParent->mALSADevice->setVocRecMode(INCALL_REC_STEREO);
152                         strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_VOICE,
153                                 sizeof(mHandle->useCase));
154                         start_csd_record(INCALL_REC_STEREO);
155                     } else
156 #endif
157                     {
158                         strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_VOICE_UL_DL,
159                                 sizeof(mHandle->useCase));
160                     }
161                 } else if (mParent->mIncallMode & AudioSystem::CHANNEL_IN_VOICE_DNLINK) {
162 #ifdef QCOM_CSDCLIENT_ENABLED
163                     if (mParent->mFusion3Platform) {
164                         mParent->mALSADevice->setVocRecMode(INCALL_REC_MONO);
165                         strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_VOICE,
166                                 sizeof(mHandle->useCase));
167                         start_csd_record(INCALL_REC_MONO);
168                     } else
169 #endif
170                     {
171                         strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_VOICE_DL,
172                                 sizeof(mHandle->useCase));
173                     }
174                 }
175 #ifdef QCOM_FM_ENABLED
176             } else if(mHandle->devices == AudioSystem::DEVICE_IN_FM_RX) {
177                 strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_FM, sizeof(mHandle->useCase));
178             } else if (mHandle->devices == AudioSystem::DEVICE_IN_FM_RX_A2DP) {
179                 strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_A2DP_FM, sizeof(mHandle->useCase));
180 #endif
181             } else if(!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP)) {
182                 strlcpy(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP, sizeof(mHandle->useCase));
183             } else {
184                     char value[128];
185                     property_get("persist.audio.lowlatency.rec",value,"0");
186                     if (!strcmp("true", value)) {
187                         strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_LOWLATENCY_MUSIC, sizeof(mHandle->useCase));
188                     } else {
189                         strlcpy(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_MUSIC, sizeof(mHandle->useCase));
190                     }
191             }
192         } else {
193             if ((mHandle->devices == AudioSystem::DEVICE_IN_VOICE_CALL) &&
194                 (newMode == AudioSystem::MODE_IN_CALL)) {
195                 ALOGD("read:: ---- mParent->mIncallMode=%d", mParent->mIncallMode);
196                 if ((mParent->mIncallMode & AudioSystem::CHANNEL_IN_VOICE_UPLINK) &&
197                     (mParent->mIncallMode & AudioSystem::CHANNEL_IN_VOICE_DNLINK)) {
198 #ifdef QCOM_CSDCLIENT_ENABLED
199                     if (mParent->mFusion3Platform) {
200                         mParent->mALSADevice->setVocRecMode(INCALL_REC_STEREO);
201                         strlcpy(mHandle->useCase, SND_USE_CASE_VERB_INCALL_REC,
202                                 sizeof(mHandle->useCase));
203                         start_csd_record(INCALL_REC_STEREO);
204                     } else
205 #endif
206                     {
207                         strlcpy(mHandle->useCase, SND_USE_CASE_VERB_UL_DL_REC,
208                                 sizeof(mHandle->useCase));
209                     }
210                 } else if (mParent->mIncallMode & AudioSystem::CHANNEL_IN_VOICE_DNLINK) {
211 #ifdef QCOM_CSDCLIENT_ENABLED
212                    if (mParent->mFusion3Platform) {
213                        mParent->mALSADevice->setVocRecMode(INCALL_REC_MONO);
214                        strlcpy(mHandle->useCase, SND_USE_CASE_VERB_INCALL_REC,
215                                sizeof(mHandle->useCase));
216                        start_csd_record(INCALL_REC_MONO);
217                    } else
218 #endif
219                    {
220                        strlcpy(mHandle->useCase, SND_USE_CASE_VERB_DL_REC,
221                                sizeof(mHandle->useCase));
222                    }
223                 }
224 #ifdef QCOM_FM_ENABLED
225             } else if(mHandle->devices == AudioSystem::DEVICE_IN_FM_RX) {
226                 strlcpy(mHandle->useCase, SND_USE_CASE_VERB_FM_REC, sizeof(mHandle->useCase));
227         } else if (mHandle->devices == AudioSystem::DEVICE_IN_FM_RX_A2DP) {
228                 strlcpy(mHandle->useCase, SND_USE_CASE_VERB_FM_A2DP_REC, sizeof(mHandle->useCase));
229 #endif
230             } else if(!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)){
231                     strlcpy(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL, sizeof(mHandle->useCase));
232             } else {
233                     char value[128];
234                     property_get("persist.audio.lowlatency.rec",value,"0");
235                     if (!strcmp("true", value)) {
236                         strlcpy(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOWLATENCY_REC, sizeof(mHandle->useCase));
237                     } else {
238                         strlcpy(mHandle->useCase, SND_USE_CASE_VERB_HIFI_REC, sizeof(mHandle->useCase));
239                     }
240             }
241         }
242         if (mHandle->channelMask == AUDIO_CHANNEL_IN_FRONT_BACK) {
243             mHandle->module->setFlags(mParent->mDevSettingsFlag | DMIC_FLAG);
244         }
245         free(use_case);
246         if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) ||
247             (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) {
248 #ifdef QCOM_USBAUDIO_ENABLED
249             if((mDevices & AudioSystem::DEVICE_IN_ANLG_DOCK_HEADSET) ||
250                (mDevices & AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET)) {
251                 mHandle->module->route(mHandle, (mDevices | AudioSystem::DEVICE_IN_PROXY) , AudioSystem::MODE_IN_COMMUNICATION);
252             }else
253 #endif
254             {
255                 mHandle->module->route(mHandle, mDevices , AudioSystem::MODE_IN_COMMUNICATION);
256             }
257         } else {
258 #ifdef QCOM_USBAUDIO_ENABLED
259             if((mHandle->devices == AudioSystem::DEVICE_IN_ANLG_DOCK_HEADSET)||
260                (mHandle->devices == AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET)){
261                 mHandle->module->route(mHandle, AudioSystem::DEVICE_IN_PROXY , mParent->mode());
262             } else
263 #endif
264             {
265 
266                 mHandle->module->route(mHandle, mDevices , mParent->mode());
267             }
268         }
269         if (!strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_REC) ||
270             !strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOWLATENCY_REC) ||
271             !strcmp(mHandle->useCase, SND_USE_CASE_VERB_FM_REC) ||
272             !strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL) ||
273             !strcmp(mHandle->useCase, SND_USE_CASE_VERB_FM_A2DP_REC) ||
274             !strcmp(mHandle->useCase, SND_USE_CASE_VERB_UL_DL_REC) ||
275             !strcmp(mHandle->useCase, SND_USE_CASE_VERB_DL_REC) ||
276             !strcmp(mHandle->useCase, SND_USE_CASE_VERB_INCALL_REC)) {
277             snd_use_case_set(mHandle->ucMgr, "_verb", mHandle->useCase);
278         } else {
279             snd_use_case_set(mHandle->ucMgr, "_enamod", mHandle->useCase);
280         }
281        if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) ||
282            (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) {
283             err = mHandle->module->startVoipCall(mHandle);
284         }
285         else
286             mHandle->module->open(mHandle);
287         if(mHandle->handle == NULL) {
288             ALOGE("read:: PCM device open failed");
289             mParent->mLock.unlock();
290 
291             return 0;
292         }
293 #ifdef QCOM_USBAUDIO_ENABLED
294         if((mHandle->devices == AudioSystem::DEVICE_IN_ANLG_DOCK_HEADSET)||
295            (mHandle->devices == AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET)){
296             if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) ||
297                (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) {
298                 mParent->musbRecordingState |= USBRECBIT_VOIPCALL;
299             } else {
300                 mParent->startUsbRecordingIfNotStarted();
301                 mParent->musbRecordingState |= USBRECBIT_REC;
302             }
303         }
304 #endif
305         mParent->mLock.unlock();
306     }
307 #ifdef QCOM_USBAUDIO_ENABLED
308     if(((mDevices & AudioSystem::DEVICE_IN_ANLG_DOCK_HEADSET) ||
309        (mDevices & AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET)) &&
310        (!mParent->musbRecordingState)) {
311         mParent->mLock.lock();
312         ALOGD("Starting UsbRecording thread");
313         mParent->startUsbRecordingIfNotStarted();
314         if(!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL) ||
315            !strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP)) {
316             ALOGD("Enabling voip recording bit");
317             mParent->musbRecordingState |= USBRECBIT_VOIPCALL;
318         }else{
319             ALOGD("Enabling HiFi Recording bit");
320             mParent->musbRecordingState |= USBRECBIT_REC;
321         }
322         mParent->mLock.unlock();
323     }
324 #endif
325     period_size = mHandle->periodSize;
326     int read_pending = bytes;
327 
328 #ifdef QCOM_SSR_ENABLED
329     if (mSurroundObj) {
330         int processed = 0;
331         int processed_pending;
332         int samples = bytes >> 1;
333         void *buffer_start = buffer;
334         int period_bytes = mHandle->handle->period_size;
335         int period_samples = period_bytes >> 1;
336 
337         do {
338             if (mSurroundOutputBufferIdx > 0) {
339                 ALOGV("AudioStreamInALSA::read() - copy processed output "
340                      "to buffer, mSurroundOutputBufferIdx = %d",
341                      mSurroundOutputBufferIdx);
342                 // Copy processed output to buffer
343                 processed_pending = mSurroundOutputBufferIdx;
344                 if (processed_pending > (samples - processed)) {
345                     processed_pending = (samples - processed);
346                 }
347                 memcpy(buffer, mSurroundOutputBuffer, processed_pending * sizeof(Word16));
348                 buffer += processed_pending * sizeof(Word16);
349                 processed += processed_pending;
350                 if (mSurroundOutputBufferIdx > processed_pending) {
351                     // Shift leftover samples to beginning of the buffer
352                     memcpy(&mSurroundOutputBuffer[0],
353                            &mSurroundOutputBuffer[processed_pending],
354                            (mSurroundOutputBufferIdx - processed_pending) * sizeof(Word16));
355                 }
356                 mSurroundOutputBufferIdx -= processed_pending;
357             }
358 
359             if (processed >= samples) {
360                 ALOGV("AudioStreamInALSA::read() - done processing buffer, "
361                      "processed = %d", processed);
362                 // Done processing this buffer
363                 break;
364             }
365 
366             // Fill input buffer until there is enough to process
367             read_pending = SSR_INPUT_FRAME_SIZE - mSurroundInputBufferIdx;
368             read = mSurroundInputBufferIdx;
369             while (mHandle->handle && read_pending > 0) {
370                 n = pcm_read(mHandle->handle, &mSurroundInputBuffer[read],
371                              period_bytes);
372                 ALOGV("pcm_read() returned n = %d buffer:%p size:%d", n, &mSurroundInputBuffer[read], period_bytes);
373                 if (n && n != -EAGAIN) {
374                     //Recovery part of pcm_read. TODO:split recovery.
375                     return static_cast<ssize_t>(n);
376                 }
377                 else if (n < 0) {
378                     // Recovery is part of pcm_write. TODO split is later.
379                     return static_cast<ssize_t>(n);
380                 }
381                 else {
382                     read_pending -= period_samples;
383                     read += period_samples;
384                 }
385             }
386 
387 
388             if (mFp_4ch) {
389                 fwrite( mSurroundInputBuffer, 1,
390                         SSR_INPUT_FRAME_SIZE * sizeof(Word16), mFp_4ch);
391             }
392 
393             //apply ssr libs to conver 4ch to 6ch
394             surround_filters_intl_process(mSurroundObj,
395                 &mSurroundOutputBuffer[mSurroundOutputBufferIdx],
396                 (Word16 *)mSurroundInputBuffer);
397 
398             // Shift leftover samples to beginning of input buffer
399             if (read_pending < 0) {
400                 memcpy(&mSurroundInputBuffer[0],
401                        &mSurroundInputBuffer[SSR_INPUT_FRAME_SIZE],
402                        (-read_pending) * sizeof(Word16));
403             }
404             mSurroundInputBufferIdx = -read_pending;
405 
406             if (mFp_6ch) {
407                 fwrite( &mSurroundOutputBuffer[mSurroundOutputBufferIdx],
408                         1, SSR_OUTPUT_FRAME_SIZE * sizeof(Word16), mFp_6ch);
409             }
410 
411             mSurroundOutputBufferIdx += SSR_OUTPUT_FRAME_SIZE;
412             ALOGV("do_while loop: processed=%d, samples=%d\n", processed, samples);
413         } while (mHandle->handle && processed < samples);
414         read = processed * sizeof(Word16);
415         buffer = buffer_start;
416     } else
417 #endif
418     {
419 
420         do {
421             if (read_pending < period_size) {
422                 read_pending = period_size;
423             }
424 
425             n = pcm_read(mHandle->handle, buffer,
426                 period_size);
427             ALOGV("pcm_read() returned n = %d", n);
428             if (n && (n == -EIO || n == -EAGAIN || n == -EPIPE || n == -EBADFD)) {
429                 mParent->mLock.lock();
430                 ALOGW("pcm_read() returned error n %d, Recovering from error\n", n);
431                 pcm_close(mHandle->handle);
432                 mHandle->handle = NULL;
433                 if((!strncmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL, strlen(SND_USE_CASE_VERB_IP_VOICECALL))) ||
434                 (!strncmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP, strlen(SND_USE_CASE_MOD_PLAY_VOIP)))) {
435                     pcm_close(mHandle->rxHandle);
436                     mHandle->rxHandle = NULL;
437                     mHandle->module->startVoipCall(mHandle);
438                 }
439                 else
440                     mHandle->module->open(mHandle);
441 
442                 if(mHandle->handle == NULL) {
443                    ALOGE("read:: PCM device re-open failed");
444                    mParent->mLock.unlock();
445                    return 0;
446                 }
447 
448                 mParent->mLock.unlock();
449                 continue;
450             }
451             else if (n < 0) {
452                 ALOGD("pcm_read() returned n < 0");
453                 return static_cast<ssize_t>(n);
454             }
455             else {
456                 read += static_cast<ssize_t>((period_size));
457                 read_pending -= period_size;
458                 //Set mute by cleanning buffers read
459                 if (mParent->mMicMute) {
460                     memset(buffer, 0, period_size);
461                 }
462                 buffer = ((uint8_t *)buffer) + period_size;
463             }
464 
465         } while (mHandle->handle && read < bytes);
466     }
467 
468     return read;
469 }
470 
471 status_t AudioStreamInALSA::dump(int fd, const Vector<String16>& args)
472 {
473     return NO_ERROR;
474 }
475 
476 status_t AudioStreamInALSA::open(int mode)
477 {
478     Mutex::Autolock autoLock(mParent->mLock);
479 
480     status_t status = ALSAStreamOps::open(mode);
481 
482     return status;
483 }
484 
485 status_t AudioStreamInALSA::close()
486 {
487     Mutex::Autolock autoLock(mParent->mLock);
488 
489     ALOGD("close");
490     if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) ||
491         (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) {
492         if((mParent->mVoipStreamCount)) {
493 #ifdef QCOM_USBAUDIO_ENABLED
494             ALOGD("musbRecordingState: %d, mVoipStreamCount:%d",mParent->musbRecordingState,
495                   mParent->mVoipStreamCount );
496             if(mParent->mVoipStreamCount == 1) {
497                 ALOGD("Deregistering VOIP Call bit, musbPlaybackState:%d,"
498                        "musbRecordingState:%d", mParent->musbPlaybackState, mParent->musbRecordingState);
499                 mParent->musbPlaybackState &= ~USBPLAYBACKBIT_VOIPCALL;
500                 mParent->musbRecordingState &= ~USBRECBIT_VOIPCALL;
501                 mParent->closeUsbRecordingIfNothingActive();
502                 mParent->closeUsbPlaybackIfNothingActive();
503             }
504 #endif
505                return NO_ERROR;
506         }
507         mParent->mVoipStreamCount = 0;
508 #ifdef QCOM_USBAUDIO_ENABLED
509     } else {
510         ALOGD("Deregistering REC bit, musbRecordingState:%d", mParent->musbRecordingState);
511         mParent->musbRecordingState &= ~USBRECBIT_REC;
512 #endif
513      }
514 #ifdef QCOM_CSDCLIENT_ENABLED
515     if (mParent->mFusion3Platform) {
516        if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_INCALL_REC)) ||
517            (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_VOICE))) {
518            stop_csd_record();
519        }
520     }
521 #endif
522     ALOGD("close");
523 #ifdef QCOM_USBAUDIO_ENABLED
524     mParent->closeUsbRecordingIfNothingActive();
525 #endif
526 
527     ALSAStreamOps::close();
528 
529 #ifdef QCOM_SSR_ENABLED
530     if (mSurroundObj) {
531         surround_filters_release(mSurroundObj);
532         if (mSurroundObj)
533             free(mSurroundObj);
534         mSurroundObj = NULL;
535         if (mRealCoeffs){
536             for (int i =0; i<COEFF_ARRAY_SIZE; i++ ) {
537                 if (mRealCoeffs[i]) {
538                     free(mRealCoeffs[i]);
539                     mRealCoeffs[i] = NULL;
540                 }
541             }
542             free(mRealCoeffs);
543             mRealCoeffs = NULL;
544         }
545         if (mImagCoeffs){
546             for (int i =0; i<COEFF_ARRAY_SIZE; i++ ) {
547                 if (mImagCoeffs[i]) {
548                     free(mImagCoeffs[i]);
549                     mImagCoeffs[i] = NULL;
550                 }
551             }
552             free(mImagCoeffs);
553             mImagCoeffs = NULL;
554         }
555         if (mSurroundOutputBuffer){
556             free(mSurroundOutputBuffer);
557             mSurroundOutputBuffer = NULL;
558         }
559         if (mSurroundInputBuffer) {
560             free(mSurroundInputBuffer);
561             mSurroundInputBuffer = NULL;
562         }
563 
564         if ( mFp_4ch ) fclose(mFp_4ch);
565         if ( mFp_6ch ) fclose(mFp_6ch);
566 
567     }
568 #endif
569 
570     return NO_ERROR;
571 }
572 
573 status_t AudioStreamInALSA::standby()
574 {
575     Mutex::Autolock autoLock(mParent->mLock);
576 
577     ALOGD("standby");
578 
579     if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) ||
580         (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) {
581          return NO_ERROR;
582     }
583 
584 #ifdef QCOM_CSDCLIENT_ENABLED
585     ALOGD("standby");
586     if (mParent->mFusion3Platform) {
587        if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_INCALL_REC)) ||
588            (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_CAPTURE_VOICE))) {
589            ALOGD(" into standby, stop record");
590            stop_csd_record();
591        }
592     }
593 #endif
594     mHandle->module->standby(mHandle);
595 
596 #ifdef QCOM_USBAUDIO_ENABLED
597     ALOGD("Checking for musbRecordingState %d", mParent->musbRecordingState);
598     mParent->musbRecordingState &= ~USBRECBIT_REC;
599     mParent->closeUsbRecordingIfNothingActive();
600 #endif
601 
602     if (mHandle->channelMask == AUDIO_CHANNEL_IN_FRONT_BACK) {
603         mHandle->module->setFlags(mParent->mDevSettingsFlag);
604     }
605 
606     return NO_ERROR;
607 }
608 
609 void AudioStreamInALSA::resetFramesLost()
610 {
611     mFramesLost = 0;
612 }
613 
614 unsigned int AudioStreamInALSA::getInputFramesLost() const
615 {
616     unsigned int count = mFramesLost;
617     // Stupid interface wants us to have a side effect of clearing the count
618     // but is defined as a const to prevent such a thing.
619     ((AudioStreamInALSA *)this)->resetFramesLost();
620     return count;
621 }
622 
623 status_t AudioStreamInALSA::setAcousticParams(void *params)
624 {
625     Mutex::Autolock autoLock(mParent->mLock);
626 
627     return (status_t)NO_ERROR;
628 }
629 
630 #ifdef QCOM_SSR_ENABLED
631 status_t AudioStreamInALSA::initSurroundSoundLibrary(unsigned long buffersize)
632 {
633     int subwoofer = 0;  // subwoofer channel assignment: default as first microphone input channel
634     int low_freq = 4;   // frequency upper bound for subwoofer: frequency=(low_freq-1)/FFT_SIZE*samplingRate, default as 4
635     int high_freq = 100;    // frequency upper bound for spatial processing: frequency=(high_freq-1)/FFT_SIZE*samplingRate, default as 100
636     int ret = 0;
637 
638     mSurroundInputBufferIdx = 0;
639     mSurroundOutputBufferIdx = 0;
640 
641     if ( mSurroundObj ) {
642         ALOGE("ola filter library is already initialized");
643         return ALREADY_EXISTS;
644     }
645 
646     // Allocate memory for input buffer
647     mSurroundInputBuffer = (Word16 *) calloc(2 * SSR_INPUT_FRAME_SIZE,
648                                               sizeof(Word16));
649     if ( !mSurroundInputBuffer ) {
650        ALOGE("Memory allocation failure. Not able to allocate memory for surroundInputBuffer");
651        goto init_fail;
652     }
653 
654     // Allocate memory for output buffer
655     mSurroundOutputBuffer = (Word16 *) calloc(2 * SSR_OUTPUT_FRAME_SIZE,
656                                                sizeof(Word16));
657     if ( !mSurroundOutputBuffer ) {
658        ALOGE("Memory allocation failure. Not able to allocate memory for surroundOutputBuffer");
659        goto init_fail;
660     }
661 
662     // Allocate memory for real and imag coeffs array
663     mRealCoeffs = (Word16 **) calloc(COEFF_ARRAY_SIZE, sizeof(Word16 *));
664     if ( !mRealCoeffs ) {
665         ALOGE("Memory allocation failure during real Coefficient array");
666         goto init_fail;
667     }
668 
669     mImagCoeffs = (Word16 **) calloc(COEFF_ARRAY_SIZE, sizeof(Word16 *));
670     if ( !mImagCoeffs ) {
671         ALOGE("Memory allocation failure during imaginary Coefficient array");
672         goto init_fail;
673     }
674 
675     if( readCoeffsFromFile() != NO_ERROR) {
676         ALOGE("Error while loading coeffs from file");
677         goto init_fail;
678     }
679 
680     //calculate the size of data to allocate for mSurroundObj
681     ret = surround_filters_init(NULL,
682                   6, // Num output channel
683                   4,     // Num input channel
684                   mRealCoeffs,       // Coeffs hardcoded in header
685                   mImagCoeffs,       // Coeffs hardcoded in header
686                   subwoofer,
687                   low_freq,
688                   high_freq,
689                   NULL);
690 
691     if ( ret > 0 ) {
692         ALOGV("Allocating surroundObj size is %d", ret);
693         mSurroundObj = (void *)malloc(ret);
694         memset(mSurroundObj,0,ret);
695         if (NULL != mSurroundObj) {
696             //initialize after allocating the memory for mSurroundObj
697             ret = surround_filters_init(mSurroundObj,
698                         6,
699                         4,
700                         mRealCoeffs,
701                         mImagCoeffs,
702                         subwoofer,
703                         low_freq,
704                         high_freq,
705                         NULL);
706             if (0 != ret) {
707                ALOGE("surround_filters_init failed with ret:%d",ret);
708                surround_filters_release(mSurroundObj);
709                goto init_fail;
710             }
711         } else {
712             ALOGE("Allocationg mSurroundObj failed");
713             goto init_fail;
714         }
715     } else {
716         ALOGE("surround_filters_init(mSurroundObj=Null) failed with ret: %d",ret);
717         goto init_fail;
718     }
719 
720     (void) surround_filters_set_channel_map(mSurroundObj, chanMap);
721 
722     return NO_ERROR;
723 
724 init_fail:
725     if (mSurroundObj) {
726         free(mSurroundObj);
727         mSurroundObj = NULL;
728     }
729     if (mSurroundOutputBuffer) {
730         free(mSurroundOutputBuffer);
731         mSurroundOutputBuffer = NULL;
732     }
733     if (mSurroundInputBuffer) {
734         free(mSurroundInputBuffer);
735         mSurroundInputBuffer = NULL;
736     }
737     if (mRealCoeffs){
738         for (int i =0; i<COEFF_ARRAY_SIZE; i++ ) {
739             if (mRealCoeffs[i]) {
740                 free(mRealCoeffs[i]);
741                 mRealCoeffs[i] = NULL;
742             }
743         }
744         free(mRealCoeffs);
745         mRealCoeffs = NULL;
746     }
747     if (mImagCoeffs){
748         for (int i =0; i<COEFF_ARRAY_SIZE; i++ ) {
749             if (mImagCoeffs[i]) {
750                 free(mImagCoeffs[i]);
751                 mImagCoeffs[i] = NULL;
752             }
753         }
754         free(mImagCoeffs);
755         mImagCoeffs = NULL;
756     }
757 
758     return NO_MEMORY;
759 
760 }
761 
762 
763 // Helper function to read coeffs from File and updates real and imaginary
764 // coeff array member variable
765 status_t AudioStreamInALSA::readCoeffsFromFile()
766 {
767     FILE    *flt1r;
768     FILE    *flt2r;
769     FILE    *flt3r;
770     FILE    *flt4r;
771     FILE    *flt1i;
772     FILE    *flt2i;
773     FILE    *flt3i;
774     FILE    *flt4i;
775 
776     if ( (flt1r = fopen(SURROUND_FILE_1R, "rb")) == NULL ) {
777         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_1R);
778         return NAME_NOT_FOUND;
779     }
780 
781     if ( (flt2r = fopen(SURROUND_FILE_2R, "rb")) == NULL ) {
782         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_2R);
783         return NAME_NOT_FOUND;
784     }
785 
786     if ( (flt3r = fopen(SURROUND_FILE_3R, "rb")) == NULL ) {
787         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_3R);
788         return  NAME_NOT_FOUND;
789     }
790 
791     if ( (flt4r = fopen(SURROUND_FILE_4R, "rb")) == NULL ) {
792         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_4R);
793         return  NAME_NOT_FOUND;
794     }
795 
796     if ( (flt1i = fopen(SURROUND_FILE_1I, "rb")) == NULL ) {
797         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_1I);
798         return NAME_NOT_FOUND;
799     }
800 
801     if ( (flt2i = fopen(SURROUND_FILE_2I, "rb")) == NULL ) {
802         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_2I);
803         return NAME_NOT_FOUND;
804     }
805 
806     if ( (flt3i = fopen(SURROUND_FILE_3I, "rb")) == NULL ) {
807         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_3I);
808         return NAME_NOT_FOUND;
809     }
810 
811     if ( (flt4i = fopen(SURROUND_FILE_4I, "rb")) == NULL ) {
812         ALOGE("Cannot open filter co-efficient file %s", SURROUND_FILE_4I);
813         return NAME_NOT_FOUND;
814     }
815     ALOGV("readCoeffsFromFile all filter files opened");
816 
817     for (int i=0; i<COEFF_ARRAY_SIZE; i++) {
818         mRealCoeffs[i] = (Word16 *)calloc(FILT_SIZE, sizeof(Word16));
819     }
820     for (int i=0; i<COEFF_ARRAY_SIZE; i++) {
821         mImagCoeffs[i] = (Word16 *)calloc(FILT_SIZE, sizeof(Word16));
822     }
823 
824     // Read real co-efficients
825     if (NULL != mRealCoeffs[0]) {
826         fread(mRealCoeffs[0], sizeof(int16), FILT_SIZE, flt1r);
827     }
828     if (NULL != mRealCoeffs[0]) {
829         fread(mRealCoeffs[1], sizeof(int16), FILT_SIZE, flt2r);
830     }
831     if (NULL != mRealCoeffs[0]) {
832         fread(mRealCoeffs[2], sizeof(int16), FILT_SIZE, flt3r);
833     }
834     if (NULL != mRealCoeffs[0]) {
835         fread(mRealCoeffs[3], sizeof(int16), FILT_SIZE, flt4r);
836     }
837 
838     // read imaginary co-efficients
839     if (NULL != mImagCoeffs[0]) {
840         fread(mImagCoeffs[0], sizeof(int16), FILT_SIZE, flt1i);
841     }
842     if (NULL != mImagCoeffs[0]) {
843         fread(mImagCoeffs[1], sizeof(int16), FILT_SIZE, flt2i);
844     }
845     if (NULL != mImagCoeffs[0]) {
846         fread(mImagCoeffs[2], sizeof(int16), FILT_SIZE, flt3i);
847     }
848     if (NULL != mImagCoeffs[0]) {
849         fread(mImagCoeffs[3], sizeof(int16), FILT_SIZE, flt4i);
850     }
851 
852     fclose(flt1r);
853     fclose(flt2r);
854     fclose(flt3r);
855     fclose(flt4r);
856     fclose(flt1i);
857     fclose(flt2i);
858     fclose(flt3i);
859     fclose(flt4i);
860 
861     return NO_ERROR;
862 }
863 #endif
864 
865 #ifdef QCOM_CSDCLIENT_ENABLED
866 int AudioStreamInALSA::start_csd_record(int param)
867 {
868     int err = NO_ERROR;
869 
870     if (mParent->mCsdHandle != NULL) {
871         csd_start_record = (int (*)(int))::dlsym(mParent->mCsdHandle,"csd_client_start_record");
872         if (csd_start_record == NULL) {
873             ALOGE("dlsym:Error:%s Loading csd_client_start_record", dlerror());
874         } else {
875             err = csd_start_record(param);
876         }
877     }
878     return err;
879 }
880 
881 int AudioStreamInALSA::stop_csd_record()
882 {
883     int err = NO_ERROR;
884     if (mParent->mCsdHandle != NULL) {
885         csd_stop_record = (int (*)())::dlsym(mParent->mCsdHandle,"csd_client_stop_record");
886         if (csd_start_record == NULL) {
887             ALOGE("dlsym:Error:%s Loading csd_client_start_record", dlerror());
888         } else {
889             csd_stop_record();
890         }
891     }
892     return err;
893 }
894 #endif
895 
896 }       // namespace android_audio_legacy
897