blob: f3a95185c5285bafd766f8f710d4c7933a1b7545 [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.
*/
#pragma once
#include "DeviceDescriptor.h"
#include "PolicyAudioPort.h"
#include "policy.h"
#include <media/AudioContainers.h>
#include <utils/String8.h>
#include <system/audio.h>
namespace android {
class HwModule;
// the IOProfile class describes the capabilities of an output or input stream.
// It is currently assumed that all combination of listed parameters are supported.
// It is used by the policy manager to determine if an output or input is suitable for
// a given use case, open/close it accordingly and connect/disconnect audio tracks
// to/from it.
class IOProfile : public AudioPort, public PolicyAudioPort
{
public:
IOProfile(const std::string &name, audio_port_role_t role);
virtual ~IOProfile() = default;
// For a Profile aka MixPort, tag name and name are equivalent.
virtual const std::string getTagName() const { return getName(); }
virtual void addAudioProfile(const sp<AudioProfile> &profile) {
addAudioProfileAndSort(mProfiles, profile);
}
virtual sp<AudioPort> asAudioPort() const {
return static_cast<AudioPort*>(const_cast<IOProfile*>(this));
}
// FIXME: this is needed because shared MMAP stream clients use the same audio session.
// Once capture clients are tracked individually and not per session this can be removed
// MMAP no IRQ input streams do not have the default limitation of one active client
// max as they can be used in shared mode by the same application.
// NOTE: Please consider moving to AudioPort when addressing the FIXME
// NOTE: this works for explicit values set in audio_policy_configuration.xml because
// flags are parsed before maxActiveCount by the serializer.
void setFlags(uint32_t flags) override
{
AudioPort::setFlags(flags);
if (getRole() == AUDIO_PORT_ROLE_SINK && (flags & AUDIO_INPUT_FLAG_MMAP_NOIRQ) != 0) {
maxActiveCount = 0;
}
refreshMixerBehaviors();
}
const MixerBehaviorSet& getMixerBehaviors() const {
return mMixerBehaviors;
}
/**
* @brief isCompatibleProfile: This method is used for input and direct output,
* and is not used for other output.
* Checks if the IO profile is compatible with specified parameters.
* For input, flags is interpreted as audio_input_flags_t.
* TODO: merge audio_output_flags_t and audio_input_flags_t.
*
* @param devices vector of devices to be checked for compatibility
* @param samplingRate to be checked for compatibility. Must be specified
* @param updatedSamplingRate if non-NULL, it is assigned the actual sample rate.
* @param format to be checked for compatibility. Must be specified
* @param updatedFormat if non-NULL, it is assigned the actual format
* @param channelMask to be checked for compatibility. Must be specified
* @param updatedChannelMask if non-NULL, it is assigned the actual channel mask
* @param flags to be checked for compatibility
* @param exactMatchRequiredForInputFlags true if exact match is required on flags
* @return true if the profile is compatible, false otherwise.
*/
bool 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 parameter type
uint32_t flags,
bool exactMatchRequiredForInputFlags = false) const;
/**
* @brief areAllDevicesSupported: Checks if the given devices are supported by the IO profile.
*
* @param devices vector of devices to be checked for compatibility
* @return true if all devices are supported, false otherwise.
*/
bool areAllDevicesSupported(const DeviceVector &devices) const;
/**
* @brief isCompatibleProfileForFlags: Checks if the IO profile is compatible with
* specified flags.
*
* @param flags to be checked for compatibility
* @param exactMatchRequiredForInputFlags true if exact match is required on flags
* @return true if the profile is compatible, false otherwise.
*/
bool isCompatibleProfileForFlags(uint32_t flags,
bool exactMatchRequiredForInputFlags = false) const;
void dump(String8 *dst, int spaces) const;
void log();
bool hasSupportedDevices() const { return !mSupportedDevices.isEmpty(); }
bool supportsDeviceTypes(const DeviceTypeSet& deviceTypes) const
{
const bool areOutputDevices = Intersection(deviceTypes, getAudioDeviceInAllSet()).empty();
const bool devicesSupported = !mSupportedDevices.getDevicesFromTypes(deviceTypes).empty();
return devicesSupported &&
(!areOutputDevices || devicesSupportEncodedFormats(deviceTypes));
}
/**
* @brief getTag
* @param deviceTypes to be considered
* @return tagName of first matching device for the considered types, empty string otherwise.
*/
std::string getTag(const DeviceTypeSet& deviceTypes) const
{
if (supportsDeviceTypes(deviceTypes)) {
return mSupportedDevices.getDevicesFromTypes(deviceTypes).itemAt(0)->getTagName();
}
return {};
}
/**
* @brief supportsDevice
* @param device to be checked against
* forceCheckOnAddress if true, check on type and address whatever the type, otherwise
* the address enforcement is limited to "offical devices" that distinguishe on address
* @return true if the device is supported by type (for non bus / remote submix devices),
* true if the device is supported (both type and address) for bus / remote submix
* false otherwise
*/
bool supportsDevice(const sp<DeviceDescriptor> &device, bool forceCheckOnAddress = false) const
{
if (!device_distinguishes_on_address(device->type()) && !forceCheckOnAddress) {
return supportsDeviceTypes(DeviceTypeSet({device->type()}));
}
return mSupportedDevices.contains(device);
}
bool devicesSupportEncodedFormats(DeviceTypeSet deviceTypes) const
{
if (deviceTypes.empty()) {
return true; // required for getOffloadSupport() check
}
DeviceVector deviceList =
mSupportedDevices.getDevicesFromTypes(deviceTypes);
for (const auto& device : deviceList) {
if (device->hasCurrentEncodedFormat()) {
return true;
}
}
return false;
}
bool containsSingleDeviceSupportingEncodedFormats(const sp<DeviceDescriptor>& device) const;
void clearSupportedDevices() { mSupportedDevices.clear(); }
void addSupportedDevice(const sp<DeviceDescriptor> &device)
{
mSupportedDevices.add(device);
}
void removeSupportedDevice(const sp<DeviceDescriptor> &device)
{
ssize_t ret = mSupportedDevices.indexOf(device);
if (ret >= 0 && !mSupportedDevices.itemAt(ret)->isDynamic()) {
// devices equality checks only type, address, name and format
// Prevents from removing non dynamically added devices
return;
}
mSupportedDevices.remove(device);
}
void setSupportedDevices(const DeviceVector &devices)
{
mSupportedDevices = devices;
}
const DeviceVector &getSupportedDevices() const { return mSupportedDevices; }
bool canOpenNewIo() {
if (maxOpenCount == 0 || curOpenCount < maxOpenCount) {
return true;
}
return false;
}
bool canStartNewIo() {
if (maxActiveCount == 0 || curActiveCount < maxActiveCount) {
return true;
}
return false;
}
void toSupportedMixerAttributes(std::vector<audio_mixer_attributes_t>* mixerAttributes) const;
status_t readFromParcelable(const media::AudioPortFw& parcelable);
void importAudioPort(const audio_port_v7& port) override;
// Number of streams currently opened for this profile.
uint32_t curOpenCount;
// Number of streams currently active for this profile. This is not the number of active clients
// (AudioTrack or AudioRecord) but the number of active HAL streams.
uint32_t curActiveCount;
private:
void refreshMixerBehaviors();
DeviceVector mSupportedDevices; // supported devices: this input/output can be routed from/to
MixerBehaviorSet mMixerBehaviors;
};
class InputProfile : public IOProfile
{
public:
explicit InputProfile(const std::string &name) : IOProfile(name, AUDIO_PORT_ROLE_SINK) {}
};
class OutputProfile : public IOProfile
{
public:
explicit OutputProfile(const std::string &name) : IOProfile(name, AUDIO_PORT_ROLE_SOURCE) {}
};
} // namespace android