blob: e1265cf9aaadec06e7947eda2dc3725e35aa445f [file] [log] [blame]
/*
* Copyright (C) 2019 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.
*/
#include <sstream>
#include <string>
#include <media/AudioContainers.h>
namespace android {
const DeviceTypeSet& getAudioDeviceOutAllSet() {
static const DeviceTypeSet audioDeviceOutAllSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_OUT_ALL_ARRAY),
std::end(AUDIO_DEVICE_OUT_ALL_ARRAY));
return audioDeviceOutAllSet;
}
const DeviceTypeSet& getAudioDeviceOutAllA2dpSet() {
static const DeviceTypeSet audioDeviceOutAllA2dpSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_OUT_ALL_A2DP_ARRAY),
std::end(AUDIO_DEVICE_OUT_ALL_A2DP_ARRAY));
return audioDeviceOutAllA2dpSet;
}
const DeviceTypeSet& getAudioDeviceOutAllScoSet() {
static const DeviceTypeSet audioDeviceOutAllScoSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_OUT_ALL_SCO_ARRAY),
std::end(AUDIO_DEVICE_OUT_ALL_SCO_ARRAY));
return audioDeviceOutAllScoSet;
}
const DeviceTypeSet& getAudioDeviceOutAllUsbSet() {
static const DeviceTypeSet audioDeviceOutAllUsbSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_OUT_ALL_USB_ARRAY),
std::end(AUDIO_DEVICE_OUT_ALL_USB_ARRAY));
return audioDeviceOutAllUsbSet;
}
const DeviceTypeSet& getAudioDeviceInAllSet() {
static const DeviceTypeSet audioDeviceInAllSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_IN_ALL_ARRAY),
std::end(AUDIO_DEVICE_IN_ALL_ARRAY));
return audioDeviceInAllSet;
}
const DeviceTypeSet& getAudioDeviceInAllUsbSet() {
static const DeviceTypeSet audioDeviceInAllUsbSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_IN_ALL_USB_ARRAY),
std::end(AUDIO_DEVICE_IN_ALL_USB_ARRAY));
return audioDeviceInAllUsbSet;
}
const DeviceTypeSet& getAudioDeviceOutAllBleSet() {
static const DeviceTypeSet audioDeviceOutAllBleSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_OUT_ALL_BLE_ARRAY),
std::end(AUDIO_DEVICE_OUT_ALL_BLE_ARRAY));
return audioDeviceOutAllBleSet;
}
const DeviceTypeSet& getAudioDeviceOutLeAudioUnicastSet() {
static const DeviceTypeSet audioDeviceOutLeAudioUnicastSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_OUT_BLE_UNICAST_ARRAY),
std::end(AUDIO_DEVICE_OUT_BLE_UNICAST_ARRAY));
return audioDeviceOutLeAudioUnicastSet;
}
const DeviceTypeSet& getAudioDeviceOutLeAudioBroadcastSet() {
static const DeviceTypeSet audioDeviceOutLeAudioUnicastSet = DeviceTypeSet(
std::begin(AUDIO_DEVICE_OUT_BLE_BROADCAST_ARRAY),
std::end(AUDIO_DEVICE_OUT_BLE_BROADCAST_ARRAY));
return audioDeviceOutLeAudioUnicastSet;
}
std::string deviceTypesToString(const DeviceTypeSet &deviceTypes) {
if (deviceTypes.empty()) {
return "Empty device types";
}
std::stringstream ss;
for (auto it = deviceTypes.begin(); it != deviceTypes.end(); ++it) {
if (it != deviceTypes.begin()) {
ss << ", ";
}
const char* strType = audio_device_to_string(*it);
if (strlen(strType) != 0) {
ss << strType;
} else {
ss << "unknown type:0x" << std::hex << *it;
}
}
return ss.str();
}
bool deviceTypesToString(const DeviceTypeSet &deviceTypes, std::string &str) {
str = deviceTypesToString(deviceTypes);
return true;
}
std::string dumpDeviceTypes(const DeviceTypeSet &deviceTypes) {
std::stringstream ss;
for (auto it = deviceTypes.begin(); it != deviceTypes.end(); ++it) {
if (it != deviceTypes.begin()) {
ss << ", ";
}
ss << "0x" << std::hex << (*it);
}
return ss.str();
}
std::string dumpMixerBehaviors(const MixerBehaviorSet& mixerBehaviors) {
std::stringstream ss;
for (auto it = mixerBehaviors.begin(); it != mixerBehaviors.end(); ++it) {
if (it != mixerBehaviors.begin()) {
ss << ", ";
}
ss << (*it);
}
return ss.str();
}
AudioProfileAttributesMultimap createAudioProfilesAttrMap(audio_profile profiles[],
uint32_t first,
uint32_t last) {
AudioProfileAttributesMultimap result;
for (uint32_t i = first; i < last; ++i) {
SampleRateSet sampleRates(profiles[i].sample_rates,
profiles[i].sample_rates + profiles[i].num_sample_rates);
ChannelMaskSet channelMasks(profiles[i].channel_masks,
profiles[i].channel_masks + profiles[i].num_channel_masks);
result.emplace(profiles[i].format, std::make_pair(sampleRates, channelMasks));
}
return result;
}
namespace {
void populateAudioProfile(audio_format_t format,
const ChannelMaskSet& channelMasks,
const SampleRateSet& samplingRates,
audio_profile* profile) {
profile->format = format;
profile->num_channel_masks = 0;
for (auto it = channelMasks.begin();
it != channelMasks.end() && profile->num_channel_masks < AUDIO_PORT_MAX_CHANNEL_MASKS;
++it) {
profile->channel_masks[profile->num_channel_masks++] = *it;
}
profile->num_sample_rates = 0;
for (auto it = samplingRates.begin();
it != samplingRates.end() && profile->num_sample_rates < AUDIO_PORT_MAX_SAMPLING_RATES;
++it) {
profile->sample_rates[profile->num_sample_rates++] = *it;
}
}
} // namespace
void populateAudioProfiles(const AudioProfileAttributesMultimap& profileAttrs,
audio_format_t format,
ChannelMaskSet allChannelMasks,
SampleRateSet allSampleRates,
audio_profile audioProfiles[],
uint32_t* numAudioProfiles,
uint32_t maxAudioProfiles) {
if (*numAudioProfiles >= maxAudioProfiles) {
return;
}
const auto lower= profileAttrs.lower_bound(format);
const auto upper = profileAttrs.upper_bound(format);
SampleRateSet sampleRatesPresent;
ChannelMaskSet channelMasksPresent;
for (auto it = lower; it != upper && *numAudioProfiles < maxAudioProfiles; ++it) {
SampleRateSet srs;
std::set_intersection(it->second.first.begin(), it->second.first.end(),
allSampleRates.begin(), allSampleRates.end(),
std::inserter(srs, srs.begin()));
if (srs.empty()) {
continue;
}
ChannelMaskSet cms;
std::set_intersection(it->second.second.begin(), it->second.second.end(),
allChannelMasks.begin(), allChannelMasks.end(),
std::inserter(cms, cms.begin()));
if (cms.empty()) {
continue;
}
sampleRatesPresent.insert(srs.begin(), srs.end());
channelMasksPresent.insert(cms.begin(), cms.end());
populateAudioProfile(it->first, cms, srs,
&audioProfiles[(*numAudioProfiles)++]);
}
if (*numAudioProfiles >= maxAudioProfiles) {
ALOGW("%s, too many audio profiles", __func__);
return;
}
SampleRateSet srs;
std::set_difference(allSampleRates.begin(), allSampleRates.end(),
sampleRatesPresent.begin(), sampleRatesPresent.end(),
std::inserter(srs, srs.begin()));
if (!srs.empty()) {
populateAudioProfile(format, allChannelMasks, srs,
&audioProfiles[(*numAudioProfiles)++]);
}
if (*numAudioProfiles >= maxAudioProfiles) {
ALOGW("%s, too many audio profiles", __func__);
return;
}
ChannelMaskSet cms;
std::set_difference(allChannelMasks.begin(), allChannelMasks.end(),
channelMasksPresent.begin(), channelMasksPresent.end(),
std::inserter(cms, cms.begin()));
if (!cms.empty()) {
populateAudioProfile(format, cms, allSampleRates,
&audioProfiles[(*numAudioProfiles)++]);
}
}
} // namespace android