/* * 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 #include "IOProfile.h" #include "HwModule.h" #include "TypeConverter.h" namespace android { bool IOProfile::isCompatibleProfile(const DeviceVector &devices, uint32_t samplingRate, uint32_t *updatedSamplingRate, audio_format_t format, audio_format_t *updatedFormat, audio_channel_mask_t channelMask, audio_channel_mask_t *updatedChannelMask, // FIXME type punning here uint32_t flags, bool exactMatchRequiredForInputFlags) const { const bool isPlaybackThread = getType() == AUDIO_PORT_TYPE_MIX && getRole() == AUDIO_PORT_ROLE_SOURCE; const bool isRecordThread = getType() == AUDIO_PORT_TYPE_MIX && getRole() == AUDIO_PORT_ROLE_SINK; ALOG_ASSERT(isPlaybackThread != isRecordThread); if (!devices.isEmpty()) { if (!mSupportedDevices.containsAllDevices(devices)) { return false; } } if (!audio_is_valid_format(format) || (isPlaybackThread && (samplingRate == 0 || !audio_is_output_channel(channelMask))) || (isRecordThread && (!audio_is_input_channel(channelMask)))) { return false; } audio_format_t myUpdatedFormat = format; audio_channel_mask_t myUpdatedChannelMask = channelMask; uint32_t myUpdatedSamplingRate = samplingRate; const struct audio_port_config config = { .config_mask = AUDIO_PORT_CONFIG_ALL & ~AUDIO_PORT_CONFIG_GAIN, .sample_rate = samplingRate, .channel_mask = channelMask, .format = format, }; if (isRecordThread) { if ((flags & AUDIO_INPUT_FLAG_MMAP_NOIRQ) != 0) { if (checkExactAudioProfile(&config) != NO_ERROR) { return false; } } else if (checkCompatibleAudioProfile( myUpdatedSamplingRate, myUpdatedChannelMask, myUpdatedFormat) != NO_ERROR) { return false; } } else { if (checkExactAudioProfile(&config) != NO_ERROR) { return false; } } const uint32_t mustMatchOutputFlags = AUDIO_OUTPUT_FLAG_DIRECT|AUDIO_OUTPUT_FLAG_HW_AV_SYNC|AUDIO_OUTPUT_FLAG_MMAP_NOIRQ; if (isPlaybackThread && (((getFlags() ^ flags) & mustMatchOutputFlags) || (getFlags() & 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 && ((getFlags() ^ flags) & ~(exactMatchRequiredForInputFlags ? AUDIO_INPUT_FLAG_NONE : 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(String8 *dst) const { std::string portStr; AudioPort::dump(&portStr, 4); dst->append(portStr.c_str()); dst->appendFormat(" - flags: 0x%04x", getFlags()); std::string flagsLiteral; if (getRole() == AUDIO_PORT_ROLE_SINK) { InputFlagConverter::maskToString(getFlags(), flagsLiteral); } else if (getRole() == AUDIO_PORT_ROLE_SOURCE) { OutputFlagConverter::maskToString(getFlags(), flagsLiteral); } if (!flagsLiteral.empty()) { dst->appendFormat(" (%s)", flagsLiteral.c_str()); } dst->append("\n"); mSupportedDevices.dump(dst, String8("Supported"), 4, false); dst->appendFormat("\n - maxOpenCount: %u - curOpenCount: %u\n", maxOpenCount, curOpenCount); dst->appendFormat(" - maxActiveCount: %u - curActiveCount: %u\n", maxActiveCount, curActiveCount); } void IOProfile::log() { // @TODO: forward log to AudioPort } } // namespace android