/* * 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. */ #define LOG_TAG "DeviceDescriptorBase" //#define LOG_NDEBUG 0 #include #include #include #include #include namespace android { DeviceDescriptorBase::DeviceDescriptorBase(audio_devices_t type) : DeviceDescriptorBase(type, "") { } DeviceDescriptorBase::DeviceDescriptorBase( audio_devices_t type, const std::string& address, const FormatVector &encodedFormats) : DeviceDescriptorBase(AudioDeviceTypeAddr(type, address), encodedFormats) { } DeviceDescriptorBase::DeviceDescriptorBase( const AudioDeviceTypeAddr &deviceTypeAddr, const FormatVector &encodedFormats) : AudioPort("", AUDIO_PORT_TYPE_DEVICE, audio_is_output_device(deviceTypeAddr.mType) ? AUDIO_PORT_ROLE_SINK : AUDIO_PORT_ROLE_SOURCE), mDeviceTypeAddr(deviceTypeAddr), mEncodedFormats(encodedFormats) { if (mDeviceTypeAddr.address().empty() && audio_is_remote_submix_device(mDeviceTypeAddr.mType)) { mDeviceTypeAddr.setAddress("0"); } } void DeviceDescriptorBase::setAddress(const std::string &address) { mDeviceTypeAddr.setAddress(address); } void DeviceDescriptorBase::toAudioPortConfig(struct audio_port_config *dstConfig, const struct audio_port_config *srcConfig) const { dstConfig->config_mask = AUDIO_PORT_CONFIG_GAIN; if (mSamplingRate != 0) { dstConfig->config_mask |= AUDIO_PORT_CONFIG_SAMPLE_RATE; } if (mChannelMask != AUDIO_CHANNEL_NONE) { dstConfig->config_mask |= AUDIO_PORT_CONFIG_CHANNEL_MASK; } if (mFormat != AUDIO_FORMAT_INVALID) { dstConfig->config_mask |= AUDIO_PORT_CONFIG_FORMAT; } if (srcConfig != NULL) { dstConfig->config_mask |= srcConfig->config_mask; } AudioPortConfig::toAudioPortConfig(dstConfig, srcConfig); dstConfig->role = audio_is_output_device(mDeviceTypeAddr.mType) ? AUDIO_PORT_ROLE_SINK : AUDIO_PORT_ROLE_SOURCE; dstConfig->type = AUDIO_PORT_TYPE_DEVICE; dstConfig->ext.device.type = mDeviceTypeAddr.mType; (void)audio_utils_strlcpy_zerofill(dstConfig->ext.device.address, mDeviceTypeAddr.getAddress()); } void DeviceDescriptorBase::toAudioPort(struct audio_port *port) const { ALOGV("DeviceDescriptorBase::toAudioPort() handle %d type %08x", mId, mDeviceTypeAddr.mType); toAudioPortInternal(port); } void DeviceDescriptorBase::toAudioPort(struct audio_port_v7 *port) const { ALOGV("DeviceDescriptorBase::toAudioPort() v7 handle %d type %08x", mId, mDeviceTypeAddr.mType); toAudioPortInternal(port); } status_t DeviceDescriptorBase::setEncapsulationModes(uint32_t encapsulationModes) { if ((encapsulationModes & ~AUDIO_ENCAPSULATION_MODE_ALL_POSITION_BITS) != 0) { return BAD_VALUE; } mEncapsulationModes = encapsulationModes & ~(1 << AUDIO_ENCAPSULATION_MODE_NONE); return NO_ERROR; } status_t DeviceDescriptorBase::setEncapsulationMetadataTypes(uint32_t encapsulationMetadataTypes) { if ((encapsulationMetadataTypes & ~AUDIO_ENCAPSULATION_METADATA_TYPE_ALL_POSITION_BITS) != 0) { return BAD_VALUE; } mEncapsulationMetadataTypes = encapsulationMetadataTypes & ~(1 << AUDIO_ENCAPSULATION_METADATA_TYPE_NONE); return NO_ERROR; } void DeviceDescriptorBase::dump(std::string *dst, int spaces, const char* extraInfo, bool verbose) const { if (mId != 0) { dst->append(base::StringPrintf("Port ID: %d; ", mId)); } if (extraInfo != nullptr) { dst->append(base::StringPrintf("%s; ", extraInfo)); } dst->append(base::StringPrintf("{%s}\n", mDeviceTypeAddr.toString(true /*includeSensitiveInfo*/).c_str())); dst->append(base::StringPrintf( "%*sEncapsulation modes: %u, metadata types: %u\n", spaces, "", mEncapsulationModes, mEncapsulationMetadataTypes)); AudioPort::dump(dst, spaces, nullptr, verbose); } std::string DeviceDescriptorBase::toString(bool includeSensitiveInfo) const { return mDeviceTypeAddr.toString(includeSensitiveInfo); } void DeviceDescriptorBase::log() const { ALOGI("Device id:%d type:0x%08X:%s, addr:%s", mId, mDeviceTypeAddr.mType, ::android::toString(mDeviceTypeAddr.mType).c_str(), mDeviceTypeAddr.getAddress()); AudioPort::log(" "); } template bool checkEqual(const T& f1, const T& f2) { std::set s1(f1.begin(), f1.end()); std::set s2(f2.begin(), f2.end()); return s1 == s2; } bool DeviceDescriptorBase::equals(const sp &other) const { return other != nullptr && static_cast(this)->equals(other) && static_cast(this)->equals(other, useInputChannelMask()) && mDeviceTypeAddr.equals(other->mDeviceTypeAddr) && checkEqual(mEncodedFormats, other->mEncodedFormats); } bool DeviceDescriptorBase::supportsFormat(audio_format_t format) { if (mEncodedFormats.empty()) { return true; } for (const auto& devFormat : mEncodedFormats) { if (devFormat == format) { return true; } } return false; } status_t DeviceDescriptorBase::writeToParcelable(media::AudioPort* parcelable) const { AudioPort::writeToParcelable(parcelable); AudioPortConfig::writeToParcelable(&parcelable->sys.activeConfig.hal, useInputChannelMask()); parcelable->hal.id = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_port_handle_t_int32_t(mId)); parcelable->sys.activeConfig.hal.portId = parcelable->hal.id; media::audio::common::AudioPortDeviceExt deviceExt; deviceExt.device = VALUE_OR_RETURN_STATUS( legacy2aidl_AudioDeviceTypeAddress(mDeviceTypeAddr)); deviceExt.encodedFormats = VALUE_OR_RETURN_STATUS( convertContainer>( mEncodedFormats, legacy2aidl_audio_format_t_AudioFormatDescription)); UNION_SET(parcelable->hal.ext, device, deviceExt); media::AudioPortDeviceExtSys deviceSys; deviceSys.encapsulationModes = VALUE_OR_RETURN_STATUS( legacy2aidl_AudioEncapsulationMode_mask(mEncapsulationModes)); deviceSys.encapsulationMetadataTypes = VALUE_OR_RETURN_STATUS( legacy2aidl_AudioEncapsulationMetadataType_mask(mEncapsulationMetadataTypes)); UNION_SET(parcelable->sys.ext, device, deviceSys); return OK; } status_t DeviceDescriptorBase::readFromParcelable(const media::AudioPort& parcelable) { if (parcelable.sys.type != media::AudioPortType::DEVICE) { return BAD_VALUE; } status_t status = AudioPort::readFromParcelable(parcelable) ?: AudioPortConfig::readFromParcelable( parcelable.sys.activeConfig.hal, useInputChannelMask()); if (status != OK) { return status; } media::audio::common::AudioPortDeviceExt deviceExt = VALUE_OR_RETURN_STATUS( UNION_GET(parcelable.hal.ext, device)); mDeviceTypeAddr = VALUE_OR_RETURN_STATUS( aidl2legacy_AudioDeviceTypeAddress(deviceExt.device)); mEncodedFormats = VALUE_OR_RETURN_STATUS( convertContainer(deviceExt.encodedFormats, aidl2legacy_AudioFormatDescription_audio_format_t)); media::AudioPortDeviceExtSys deviceSys = VALUE_OR_RETURN_STATUS( UNION_GET(parcelable.sys.ext, device)); mEncapsulationModes = VALUE_OR_RETURN_STATUS( aidl2legacy_AudioEncapsulationMode_mask(deviceSys.encapsulationModes)); mEncapsulationMetadataTypes = VALUE_OR_RETURN_STATUS( aidl2legacy_AudioEncapsulationMetadataType_mask(deviceSys.encapsulationMetadataTypes)); return OK; } std::string toString(const DeviceDescriptorBaseVector& devices) { std::string ret; for (const auto& device : devices) { if (device != *devices.begin()) { ret += ";"; } ret += device->toString(); } return ret; } AudioDeviceTypeAddrVector deviceTypeAddrsFromDescriptors(const DeviceDescriptorBaseVector& devices) { AudioDeviceTypeAddrVector deviceTypeAddrs; for (const auto& device : devices) { deviceTypeAddrs.push_back(device->getDeviceTypeAddr()); } return deviceTypeAddrs; } ConversionResult> aidl2legacy_DeviceDescriptorBase(const media::AudioPort& aidl) { sp result = new DeviceDescriptorBase(AUDIO_DEVICE_NONE); status_t status = result->readFromParcelable(aidl); if (status != OK) { return base::unexpected(status); } return result; } ConversionResult legacy2aidl_DeviceDescriptorBase(const sp& legacy) { media::AudioPort aidl; status_t status = legacy->writeToParcelable(&aidl); if (status != OK) { return base::unexpected(status); } return aidl; } } // namespace android