/* AudioStreamOutALSA.cpp ** ** Copyright 2008-2009 Wind River Systems ** Copyright (c) 2011, Code Aurora Forum. All rights reserved. ** ** Licensed under the Apache License, Version 2.0 (the "License"); ** you may not use this file except in compliance with the License. ** You may obtain a copy of the License at ** ** http://www.apache.org/licenses/LICENSE-2.0 ** ** Unless required by applicable law or agreed to in writing, software ** distributed under the License is distributed on an "AS IS" BASIS, ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. ** See the License for the specific language governing permissions and ** limitations under the License. */ #include #include #include #include #include #include #include #include #define LOG_TAG "AudioStreamOutALSA" //#define LOG_NDEBUG 0 #define LOG_NDDEBUG 0 #include #include #include #include #include #include "AudioHardwareALSA.h" #ifndef ALSA_DEFAULT_SAMPLE_RATE #define ALSA_DEFAULT_SAMPLE_RATE 44100 // in Hz #endif namespace android_audio_legacy { // ---------------------------------------------------------------------------- static const int DEFAULT_SAMPLE_RATE = ALSA_DEFAULT_SAMPLE_RATE; // ---------------------------------------------------------------------------- AudioStreamOutALSA::AudioStreamOutALSA(AudioHardwareALSA *parent, alsa_handle_t *handle) : ALSAStreamOps(parent, handle), mParent(parent), mFrameCount(0) { } AudioStreamOutALSA::~AudioStreamOutALSA() { close(); } uint32_t AudioStreamOutALSA::channels() const { int c = ALSAStreamOps::channels(); return c; } status_t AudioStreamOutALSA::setVolume(float left, float right) { int vol; float volume; status_t status = NO_ERROR; volume = (left + right) / 2; if (volume < 0.0) { ALOGW("AudioSessionOutALSA::setVolume(%f) under 0.0, assuming 0.0\n", volume); volume = 0.0; } else if (volume > 1.0) { ALOGW("AudioSessionOutALSA::setVolume(%f) over 1.0, assuming 1.0\n", volume); volume = 1.0; } vol = lrint((volume * 0x2000)+0.5); if(!strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOW_POWER) || !strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_LPA)) { ALOGV("setLpaVolume(%f)\n", volume); ALOGV("Setting LPA volume to %d (available range is 0 to 100)\n", vol); mHandle->module->setLpaVolume(vol); return status; } else if(!strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_TUNNEL) || !strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_TUNNEL)) { ALOGV("setCompressedVolume(%f)\n", volume); ALOGV("Setting Compressed volume to %d (available range is 0 to 100)\n", vol); mHandle->module->setCompressedVolume(vol); return status; } else if(!strncmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL, sizeof(mHandle->useCase)) || !strncmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP, sizeof(mHandle->useCase))) { ALOGV("Avoid Software volume by returning success\n"); return status; } return INVALID_OPERATION; } ssize_t AudioStreamOutALSA::write(const void *buffer, size_t bytes) { int period_size; char *use_case; ALOGV("write:: buffer %p, bytes %d", buffer, bytes); snd_pcm_sframes_t n = 0; size_t sent = 0; status_t err; int write_pending = bytes; if((mHandle->handle == NULL) && (mHandle->rxHandle == NULL) && (strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) && (strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) { mParent->mLock.lock(); ALOGD("mHandle->useCase: %s", mHandle->useCase); snd_use_case_get(mHandle->ucMgr, "_verb", (const char **)&use_case); if ((use_case == NULL) || (!strcmp(use_case, SND_USE_CASE_VERB_INACTIVE))) { if(!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP)){ strlcpy(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL, sizeof(SND_USE_CASE_VERB_IP_VOICECALL)); } else if(!strcmp(mHandle->useCase,SND_USE_CASE_MOD_PLAY_MUSIC2)) { strlcpy(mHandle->useCase, SND_USE_CASE_VERB_HIFI2, sizeof(SND_USE_CASE_MOD_PLAY_MUSIC2)); } else if (!strcmp(mHandle->useCase,SND_USE_CASE_MOD_PLAY_MUSIC)){ strlcpy(mHandle->useCase, SND_USE_CASE_VERB_HIFI, sizeof(SND_USE_CASE_MOD_PLAY_MUSIC)); } else if(!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_LOWLATENCY_MUSIC)) { strlcpy(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOWLATENCY_MUSIC, sizeof(SND_USE_CASE_MOD_PLAY_LOWLATENCY_MUSIC)); } } else { if(!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)){ strlcpy(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP, sizeof(SND_USE_CASE_MOD_PLAY_VOIP)); } else if(!strcmp(mHandle->useCase,SND_USE_CASE_VERB_HIFI2)) { strlcpy(mHandle->useCase, SND_USE_CASE_MOD_PLAY_MUSIC2, sizeof(SND_USE_CASE_MOD_PLAY_MUSIC2)); } else if (!strcmp(mHandle->useCase,SND_USE_CASE_VERB_HIFI)){ strlcpy(mHandle->useCase, SND_USE_CASE_MOD_PLAY_MUSIC, sizeof(SND_USE_CASE_MOD_PLAY_MUSIC)); } else if(!strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOWLATENCY_MUSIC)) { strlcpy(mHandle->useCase, SND_USE_CASE_MOD_PLAY_LOWLATENCY_MUSIC, sizeof(SND_USE_CASE_MOD_PLAY_LOWLATENCY_MUSIC)); } } free(use_case); if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) || (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) { #ifdef QCOM_USBAUDIO_ENABLED if((mDevices & AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET)|| (mDevices & AudioSystem::DEVICE_OUT_DGTL_DOCK_HEADSET)|| (mDevices & AudioSystem::DEVICE_OUT_PROXY)) { mDevices |= AudioSystem::DEVICE_OUT_PROXY; mHandle->module->route(mHandle, mDevices , mParent->mode()); }else #endif { mHandle->module->route(mHandle, mDevices , AudioSystem::MODE_IN_COMMUNICATION); } #ifdef QCOM_USBAUDIO_ENABLED } else if((mDevices & AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET)|| (mDevices & AudioSystem::DEVICE_OUT_DGTL_DOCK_HEADSET)|| (mDevices & AudioSystem::DEVICE_OUT_PROXY)) { mDevices |= AudioSystem::DEVICE_OUT_PROXY; mHandle->module->route(mHandle, mDevices , mParent->mode()); #endif } else { mHandle->module->route(mHandle, mDevices , mParent->mode()); } if (!strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI) || !strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI2) || !strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOWLATENCY_MUSIC) || !strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) { snd_use_case_set(mHandle->ucMgr, "_verb", mHandle->useCase); } else { snd_use_case_set(mHandle->ucMgr, "_enamod", mHandle->useCase); } if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) || (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) { err = mHandle->module->startVoipCall(mHandle); } else mHandle->module->open(mHandle); if(mHandle->handle == NULL) { ALOGE("write:: device open failed"); mParent->mLock.unlock(); return bytes; } #ifdef QCOM_USBAUDIO_ENABLED if((mHandle->devices == AudioSystem::DEVICE_IN_ANLG_DOCK_HEADSET)|| (mHandle->devices == AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET)){ if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) || (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) { mParent->musbPlaybackState |= USBPLAYBACKBIT_VOIPCALL; } else { mParent->startUsbPlaybackIfNotStarted(); mParent->musbPlaybackState |= USBPLAYBACKBIT_MUSIC; } } #endif mParent->mLock.unlock(); } #ifdef QCOM_USBAUDIO_ENABLED if(((mDevices & AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET) || (mDevices & AudioSystem::DEVICE_OUT_DGTL_DOCK_HEADSET)) && (!mParent->musbPlaybackState)) { mParent->mLock.lock(); mParent->startUsbPlaybackIfNotStarted(); ALOGV("Starting playback on USB"); if(!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL) || !strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP)) { ALOGD("Setting VOIPCALL bit here, musbPlaybackState %d", mParent->musbPlaybackState); mParent->musbPlaybackState |= USBPLAYBACKBIT_VOIPCALL; }else{ ALOGV("enabling music, musbPlaybackState: %d ", mParent->musbPlaybackState); mParent->musbPlaybackState |= USBPLAYBACKBIT_MUSIC; } mParent->mLock.unlock(); } #endif period_size = mHandle->periodSize; do { if (write_pending < period_size) { write_pending = period_size; } if((mParent->mVoipStreamCount) && (mHandle->rxHandle != 0)) { n = pcm_write(mHandle->rxHandle, (char *)buffer + sent, period_size); } else if (mHandle->handle != 0){ n = pcm_write(mHandle->handle, (char *)buffer + sent, period_size); } if (n < 0) { mParent->mLock.lock(); if (mHandle->handle != NULL) { ALOGE("pcm_write returned error %d, trying to recover\n", n); pcm_close(mHandle->handle); mHandle->handle = NULL; if((!strncmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL, strlen(SND_USE_CASE_VERB_IP_VOICECALL))) || (!strncmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP, strlen(SND_USE_CASE_MOD_PLAY_VOIP)))) { pcm_close(mHandle->rxHandle); mHandle->rxHandle = NULL; mHandle->module->startVoipCall(mHandle); } else mHandle->module->open(mHandle); if(mHandle->handle == NULL) { ALOGE("write:: device re-open failed"); mParent->mLock.unlock(); return bytes; } } mParent->mLock.unlock(); continue; } else { mFrameCount += n; sent += static_cast((period_size)); write_pending -= period_size; } } while ((mHandle->handle||(mHandle->rxHandle && mParent->mVoipStreamCount)) && sent < bytes); return sent; } status_t AudioStreamOutALSA::dump(int fd, const Vector& args) { return NO_ERROR; } status_t AudioStreamOutALSA::open(int mode) { Mutex::Autolock autoLock(mParent->mLock); return ALSAStreamOps::open(mode); } status_t AudioStreamOutALSA::close() { Mutex::Autolock autoLock(mParent->mLock); ALOGV("close"); if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) || (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) { if((mParent->mVoipStreamCount)) { #ifdef QCOM_USBAUDIO_ENABLED if(mParent->mVoipStreamCount == 1) { ALOGV("Deregistering VOIP Call bit, musbPlaybackState:%d, musbRecordingState: %d", mParent->musbPlaybackState, mParent->musbRecordingState); mParent->musbPlaybackState &= ~USBPLAYBACKBIT_VOIPCALL; mParent->musbRecordingState &= ~USBRECBIT_VOIPCALL; mParent->closeUsbPlaybackIfNothingActive(); mParent->closeUsbRecordingIfNothingActive(); } #endif return NO_ERROR; } mParent->mVoipStreamCount = 0; } #ifdef QCOM_USBAUDIO_ENABLED else if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOW_POWER)) || (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_LPA))) { mParent->musbPlaybackState &= ~USBPLAYBACKBIT_LPA; } else { mParent->musbPlaybackState &= ~USBPLAYBACKBIT_MUSIC; } mParent->closeUsbPlaybackIfNothingActive(); #endif ALSAStreamOps::close(); return NO_ERROR; } status_t AudioStreamOutALSA::standby() { Mutex::Autolock autoLock(mParent->mLock); ALOGV("standby"); if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_IP_VOICECALL)) || (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_VOIP))) { return NO_ERROR; } #ifdef QCOM_USBAUDIO_ENABLED if((!strcmp(mHandle->useCase, SND_USE_CASE_VERB_HIFI_LOW_POWER)) || (!strcmp(mHandle->useCase, SND_USE_CASE_MOD_PLAY_LPA))) { ALOGV("Deregistering LPA bit"); mParent->musbPlaybackState &= ~USBPLAYBACKBIT_LPA; } else { ALOGV("Deregistering MUSIC bit, musbPlaybackState: %d", mParent->musbPlaybackState); mParent->musbPlaybackState &= ~USBPLAYBACKBIT_MUSIC; } #endif mHandle->module->standby(mHandle); #ifdef QCOM_USBAUDIO_ENABLED mParent->closeUsbPlaybackIfNothingActive(); #endif mFrameCount = 0; return NO_ERROR; } #define USEC_TO_MSEC(x) ((x + 999) / 1000) uint32_t AudioStreamOutALSA::latency() const { // Android wants latency in milliseconds. return USEC_TO_MSEC (mHandle->latency); } // return the number of audio frames written by the audio dsp to DAC since // the output has exited standby status_t AudioStreamOutALSA::getRenderPosition(uint32_t *dspFrames) { *dspFrames = mFrameCount; return NO_ERROR; } } // namespace android_audio_legacy