/*
**
** Copyright 2012, The Android Open Source Project
**
** 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.
*/
#define LOG_TAG "AudioHAL:AudioHardwareInput"
#include <utils/Log.h>
#include <fcntl.h>
#include <sys/eventfd.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <utils/String8.h>
#include "AudioHardwareInput.h"
#include "AudioHotplugThread.h"
#include "AudioStreamIn.h"
namespace android {
// Global singleton.
AudioHardwareInput gAudioHardwareInput;
AudioHardwareInput::AudioHardwareInput()
: mMicMute(false)
{
mHotplugThread = new AudioHotplugThread(*this);
if (mHotplugThread == NULL) {
ALOGE("Unable to create ATV Remote audio hotplug thread. "
"Pluggable audio input devices will not function.");
} else if (!mHotplugThread->start()) {
ALOGE("Unable to start ATV Remote audio hotplug thread. "
"Pluggable audio input devices will not function.");
mHotplugThread.clear();
}
for (int i=0; i<kMaxDevices; i++) {
mDeviceInfos[i].valid = false;
}
}
AudioHardwareInput::~AudioHardwareInput()
{
if (mHotplugThread != NULL) {
mHotplugThread->shutdown();
mHotplugThread.clear();
}
closeAllInputStreams();
}
status_t AudioHardwareInput::setMicMute(bool mute)
{
mMicMute = mute;
return NO_ERROR;
}
status_t AudioHardwareInput::getMicMute(bool* mute)
{
*mute = mMicMute;
return NO_ERROR;
}
// milliseconds per ALSA period
const uint32_t AudioHardwareInput::kPeriodMsec = 20;
size_t AudioHardwareInput::calculateInputBufferSize(uint32_t outputSampleRate,
audio_format_t format,
uint32_t channelCount)
{
size_t size;
// AudioFlinger expects audio buffers to be a multiple of 16 frames
size = (kPeriodMsec * outputSampleRate) / 1000;
size = ((size + 15) / 16) * 16;
return size * channelCount * audio_bytes_per_sample(format);
}
status_t AudioHardwareInput::getInputBufferSize(const audio_config* config)
{
size_t size = calculateInputBufferSize(config->sample_rate,
config->format,
audio_channel_count_from_in_mask(config->channel_mask));
return size;
}
AudioStreamIn* AudioHardwareInput::openInputStream(uint32_t devices,
audio_format_t* format, uint32_t* channelMask, uint32_t* sampleRate,
status_t* status)
{
(void) devices;
Mutex::Autolock _l(mLock);
AudioStreamIn* in;
in = new AudioStreamIn(*this);
if (in == NULL) {
*status = NO_MEMORY;
return NULL;
}
*status = in->set(format, channelMask, sampleRate);
if (*status != NO_ERROR) {
delete in;
return NULL;
}
mInputStreams.add(in);
return in;
}
void AudioHardwareInput::closeInputStream(AudioStreamIn* in)
{
Mutex::Autolock _l(mLock);
for (size_t i = 0; i < mInputStreams.size(); i++) {
if (in == mInputStreams[i]) {
mInputStreams.removeAt(i);
in->standby();
delete in;
break;
}
}
}
void AudioHardwareInput::closeAllInputStreams()
{
while (mInputStreams.size() != 0) {
AudioStreamIn* in = mInputStreams[0];
mInputStreams.removeAt(0);
in->standby();
delete in;
}
}
void AudioHardwareInput::standbyAllInputStreams(const AudioHotplugThread::DeviceInfo* deviceInfo)
{
for (size_t i = 0; i < mInputStreams.size(); i++) {
if (deviceInfo == NULL || deviceInfo == mInputStreams[i]->getDeviceInfo()) {
mInputStreams[i]->standby();
}
}
}
#define DUMP(a...) \
snprintf(buffer, SIZE, a); \
buffer[SIZE - 1] = 0; \
result.append(buffer);
#define B2STR(b) b ? "true" : "false"
status_t AudioHardwareInput::dump(int fd)
{
const size_t SIZE = 256;
char buffer[SIZE];
String8 result;
DUMP("\nAudioHardwareInput::dump\n");
for (int i=0; i<kMaxDevices; i++) {
if (mDeviceInfos[i].valid) {
DUMP("device[%d] is valid\n", i);
DUMP("\tcapture card: %d\n", mDeviceInfos[i].pcmCard);
DUMP("\tcapture device: %d\n", mDeviceInfos[i].pcmDevice);
}
}
::write(fd, result.string(), result.size());
{
Mutex::Autolock _l(mLock);
for (size_t i = 0; i < mInputStreams.size(); i++) {
mInputStreams[i]->dump(fd);
}
}
return NO_ERROR;
}
#undef DUMP
#undef B2STR
// called on the audio hotplug thread
void AudioHardwareInput::onDeviceFound(
const AudioHotplugThread::DeviceInfo& devInfo)
{
bool foundSlot = false;
Mutex::Autolock _l(mLock);
ALOGD("AudioHardwareInput::onDeviceFound pcmCard = %d", devInfo.pcmCard);
for (int i=0; i<kMaxDevices; i++) {
if (mDeviceInfos[i].valid) {
if ((mDeviceInfos[i].pcmCard == devInfo.pcmCard)
&& (mDeviceInfos[i].pcmDevice == devInfo.pcmDevice)) {
ALOGW("AudioHardwareInput::onDeviceFound already has %d:%d",
devInfo.pcmCard, devInfo.pcmDevice);
return; // Got it already so no action needed.
}
}
}
// New device so find an empty slot and save it.
for (int i=0; i<kMaxDevices; i++) {
if (!mDeviceInfos[i].valid) {
ALOGD("AudioHardwareInput::onDeviceFound saving as device #%d", i);
mDeviceInfos[i] = devInfo;
mDeviceInfos[i].valid = true;
foundSlot = true;
/* Restart any currently running streams. */
standbyAllInputStreams(NULL);
break;
}
}
if (!foundSlot) {
ALOGW("AudioHardwareInput::onDeviceFound found more devices than expected! Dropped");
}
}
// called on the audio hotplug thread
void AudioHardwareInput::onDeviceRemoved(unsigned int pcmCard, unsigned int pcmDevice)
{
Mutex::Autolock _l(mLock);
ALOGD("AudioHardwareInput::onDeviceRemoved pcmCard = %d", pcmCard);
// Find matching DeviceInfo.
for (int i=0; i<kMaxDevices; i++) {
if (mDeviceInfos[i].valid) {
if ((mDeviceInfos[i].pcmCard == pcmCard) && (mDeviceInfos[i].pcmDevice == pcmDevice)) {
ALOGD("AudioHardwareInput::onDeviceRemoved matches #%d", i);
mDeviceInfos[i].valid = false;
/* If currently active stream is using this device then restart. */
standbyAllInputStreams(&mDeviceInfos[i]);
break;
}
}
}
}
const AudioHotplugThread::DeviceInfo* AudioHardwareInput::getBestDevice(int inputSource)
{
bool doVoiceRecognition = (inputSource == AUDIO_SOURCE_VOICE_RECOGNITION);
const bool favorNoVoiceRecognition = (inputSource == AUDIO_SOURCE_UNPROCESSED);
int chosenDeviceIndex = -1;
Mutex::Autolock _l(mLock);
ALOGD("AudioHardwareInput::getBestDevice inputSource = %d, doVoiceRecognition = %d",
inputSource, (doVoiceRecognition ? 1 : 0));
// RemoteControl is the only input device usable for voice recognition
// and no other devices are used for voice recognition.
// Currently the RemoteControl is the only device marked with forVoiceRecognition=true.
// A connected USB mic could be used for anything but voice recognition.
// For UNPROCESSED source, a connected USB microphone will be favored over the remote mic.
for (int i=0; i<kMaxDevices; i++) {
if (mDeviceInfos[i].valid) {
if (favorNoVoiceRecognition) {
if (mDeviceInfos[i].forVoiceRecognition) {
chosenDeviceIndex = i;
//continue matching
} else {
chosenDeviceIndex = i;
break;
}
} else if (mDeviceInfos[i].forVoiceRecognition == doVoiceRecognition) {
chosenDeviceIndex = i;
break;
}
}
}
if (chosenDeviceIndex < 0) {
ALOGE("ERROR AudioHardwareInput::getBestDevice, none for source %d", inputSource);
} else {
ALOGD("AudioHardwareInput::getBestDevice chose #%d", chosenDeviceIndex);
}
return (chosenDeviceIndex >= 0) ? &mDeviceInfos[chosenDeviceIndex] : NULL;
}
}; // namespace android