blob: bee105bf2eea348f1efc41156c489b1d21533137 [file] [log] [blame]
/*
* Copyright (C) 2015 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 "APM::IOProfile"
//#define LOG_NDEBUG 0
#include "IOProfile.h"
#include "HwModule.h"
#include "AudioGain.h"
namespace android {
IOProfile::IOProfile(const String8& name, audio_port_role_t role)
: AudioPort(name, AUDIO_PORT_TYPE_MIX, role)
{
}
IOProfile::~IOProfile()
{
}
// checks if the IO profile is compatible with specified parameters.
// Sampling rate, format and channel mask must be specified in order to
// get a valid a match
bool IOProfile::isCompatibleProfile(audio_devices_t device,
const String8& address,
uint32_t samplingRate,
uint32_t *updatedSamplingRate,
audio_format_t format,
audio_format_t *updatedFormat,
audio_channel_mask_t channelMask,
audio_channel_mask_t *updatedChannelMask,
uint32_t flags) const
{
const bool isPlaybackThread = mType == AUDIO_PORT_TYPE_MIX && mRole == AUDIO_PORT_ROLE_SOURCE;
const bool isRecordThread = mType == AUDIO_PORT_TYPE_MIX && mRole == AUDIO_PORT_ROLE_SINK;
ALOG_ASSERT(isPlaybackThread != isRecordThread);
if (device != AUDIO_DEVICE_NONE) {
// just check types if multiple devices are selected
if (popcount(device & ~AUDIO_DEVICE_BIT_IN) > 1) {
if ((mSupportedDevices.types() & device) != device) {
return false;
}
} else if (mSupportedDevices.getDevice(device, address) == 0) {
return false;
}
}
if (samplingRate == 0) {
return false;
}
uint32_t myUpdatedSamplingRate = samplingRate;
if (isPlaybackThread && checkExactSamplingRate(samplingRate) != NO_ERROR) {
return false;
}
if (isRecordThread && checkCompatibleSamplingRate(samplingRate, &myUpdatedSamplingRate) !=
NO_ERROR) {
return false;
}
if (!audio_is_valid_format(format)) {
return false;
}
if (isPlaybackThread && checkExactFormat(format) != NO_ERROR) {
return false;
}
audio_format_t myUpdatedFormat = format;
if (isRecordThread && checkCompatibleFormat(format, &myUpdatedFormat) != NO_ERROR) {
return false;
}
if (isPlaybackThread && (!audio_is_output_channel(channelMask) ||
checkExactChannelMask(channelMask) != NO_ERROR)) {
return false;
}
audio_channel_mask_t myUpdatedChannelMask = channelMask;
if (isRecordThread && (!audio_is_input_channel(channelMask) ||
checkCompatibleChannelMask(channelMask, &myUpdatedChannelMask) != NO_ERROR)) {
return false;
}
if (isPlaybackThread && (mFlags & flags) != flags) {
return false;
}
// The only input flag that is allowed to be different is the fast flag.
// An existing fast stream is compatible with a normal track request.
// An existing normal stream is compatible with a fast track request,
// but the fast request will be denied by AudioFlinger and converted to normal track.
if (isRecordThread && ((mFlags ^ flags) &
~AUDIO_INPUT_FLAG_FAST)) {
return false;
}
if (updatedSamplingRate != NULL) {
*updatedSamplingRate = myUpdatedSamplingRate;
}
if (updatedFormat != NULL) {
*updatedFormat = myUpdatedFormat;
}
if (updatedChannelMask != NULL) {
*updatedChannelMask = myUpdatedChannelMask;
}
return true;
}
void IOProfile::dump(int fd)
{
const size_t SIZE = 256;
char buffer[SIZE];
String8 result;
AudioPort::dump(fd, 4);
snprintf(buffer, SIZE, " - flags: 0x%04x\n", mFlags);
result.append(buffer);
snprintf(buffer, SIZE, " - devices:\n");
result.append(buffer);
write(fd, result.string(), result.size());
for (size_t i = 0; i < mSupportedDevices.size(); i++) {
mSupportedDevices[i]->dump(fd, 6, i);
}
}
void IOProfile::log()
{
const size_t SIZE = 256;
char buffer[SIZE];
String8 result;
ALOGV(" - sampling rates: ");
for (size_t i = 0; i < mSamplingRates.size(); i++) {
ALOGV(" %d", mSamplingRates[i]);
}
ALOGV(" - channel masks: ");
for (size_t i = 0; i < mChannelMasks.size(); i++) {
ALOGV(" 0x%04x", mChannelMasks[i]);
}
ALOGV(" - formats: ");
for (size_t i = 0; i < mFormats.size(); i++) {
ALOGV(" 0x%08x", mFormats[i]);
}
ALOGV(" - devices: 0x%04x\n", mSupportedDevices.types());
ALOGV(" - flags: 0x%04x\n", mFlags);
}
}; // namespace android