blob: 05b7412aef255f6b133858b7607a66d5faadece9 [file] [log] [blame]
/*
* Copyright 2016 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.
*/
/******************************************************************************
*
* Utility functions to help build and parse the AAC Codec Information
* Element and Media Payload.
*
******************************************************************************/
#define LOG_TAG "a2dp_aac"
#include "a2dp_aac.h"
#include <bluetooth/log.h>
#include <string.h>
#include "a2dp_aac_decoder.h"
#include "a2dp_aac_encoder.h"
#include "include/check.h"
#include "internal_include/bt_trace.h"
#include "os/log.h"
#include "osi/include/osi.h"
#include "osi/include/properties.h"
#include "stack/include/bt_hdr.h"
#define A2DP_AAC_DEFAULT_BITRATE 320000 // 320 kbps
#define A2DP_AAC_MIN_BITRATE 64000 // 64 kbps
using namespace bluetooth;
// data type for the AAC Codec Information Element */
// NOTE: bits_per_sample is needed only for AAC encoder initialization.
typedef struct {
uint8_t objectType; /* Object Type */
uint16_t sampleRate; /* Sampling Frequency */
uint8_t channelMode; /* STEREO/MONO */
uint8_t variableBitRateSupport; /* Variable Bit Rate Support*/
uint32_t bitRate; /* Bit rate */
btav_a2dp_codec_bits_per_sample_t bits_per_sample;
} tA2DP_AAC_CIE;
static bool aac_source_caps_configured = false;
static tA2DP_AAC_CIE a2dp_aac_source_caps = {};
/* AAC Source codec capabilities */
static const tA2DP_AAC_CIE a2dp_aac_cbr_source_caps = {
// objectType
A2DP_AAC_OBJECT_TYPE_MPEG2_LC,
// sampleRate
// TODO: AAC 48.0kHz sampling rate should be added back - see b/62301376
A2DP_AAC_SAMPLING_FREQ_44100,
// channelMode
A2DP_AAC_CHANNEL_MODE_STEREO,
// variableBitRateSupport
A2DP_AAC_VARIABLE_BIT_RATE_DISABLED,
// bitRate
A2DP_AAC_DEFAULT_BITRATE,
// bits_per_sample
BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16};
/* AAC Source codec capabilities */
static const tA2DP_AAC_CIE a2dp_aac_vbr_source_caps = {
// objectType
A2DP_AAC_OBJECT_TYPE_MPEG2_LC,
// sampleRate
// TODO: AAC 48.0kHz sampling rate should be added back - see b/62301376
A2DP_AAC_SAMPLING_FREQ_44100,
// channelMode
A2DP_AAC_CHANNEL_MODE_STEREO,
// variableBitRateSupport
A2DP_AAC_VARIABLE_BIT_RATE_ENABLED,
// bitRate
A2DP_AAC_DEFAULT_BITRATE,
// bits_per_sample
BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16};
/* AAC Sink codec capabilities */
static const tA2DP_AAC_CIE a2dp_aac_sink_caps = {
// objectType
A2DP_AAC_OBJECT_TYPE_MPEG2_LC,
// sampleRate
A2DP_AAC_SAMPLING_FREQ_44100 | A2DP_AAC_SAMPLING_FREQ_48000,
// channelMode
A2DP_AAC_CHANNEL_MODE_MONO | A2DP_AAC_CHANNEL_MODE_STEREO,
// variableBitRateSupport
A2DP_AAC_VARIABLE_BIT_RATE_ENABLED,
// bitRate
A2DP_AAC_DEFAULT_BITRATE,
// bits_per_sample
BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16};
/* Default AAC codec configuration */
static const tA2DP_AAC_CIE a2dp_aac_default_config = {
A2DP_AAC_OBJECT_TYPE_MPEG2_LC, // objectType
A2DP_AAC_SAMPLING_FREQ_44100, // sampleRate
A2DP_AAC_CHANNEL_MODE_STEREO, // channelMode
A2DP_AAC_VARIABLE_BIT_RATE_DISABLED, // variableBitRateSupport
A2DP_AAC_DEFAULT_BITRATE, // bitRate
BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16 // bits_per_sample
};
static const tA2DP_ENCODER_INTERFACE a2dp_encoder_interface_aac = {
a2dp_aac_encoder_init,
a2dp_aac_encoder_cleanup,
a2dp_aac_feeding_reset,
a2dp_aac_feeding_flush,
a2dp_aac_get_encoder_interval_ms,
a2dp_aac_get_effective_frame_size,
a2dp_aac_send_frames,
nullptr // set_transmit_queue_length
};
static const tA2DP_DECODER_INTERFACE a2dp_decoder_interface_aac = {
a2dp_aac_decoder_init,
a2dp_aac_decoder_cleanup,
a2dp_aac_decoder_decode_packet,
nullptr, // decoder_start
nullptr, // decoder_suspend
nullptr, // decoder_configure
};
UNUSED_ATTR static tA2DP_STATUS A2DP_CodecInfoMatchesCapabilityAac(
const tA2DP_AAC_CIE* p_cap, const uint8_t* p_codec_info,
bool is_capability);
// Builds the AAC Media Codec Capabilities byte sequence beginning from the
// LOSC octet. |media_type| is the media type |AVDT_MEDIA_TYPE_*|.
// |p_ie| is a pointer to the AAC Codec Information Element information.
// The result is stored in |p_result|. Returns A2DP_SUCCESS on success,
// otherwise the corresponding A2DP error status code.
static tA2DP_STATUS A2DP_BuildInfoAac(uint8_t media_type,
const tA2DP_AAC_CIE* p_ie,
uint8_t* p_result) {
if (p_ie == NULL || p_result == NULL) {
return A2DP_INVALID_PARAMS;
}
*p_result++ = A2DP_AAC_CODEC_LEN;
*p_result++ = (media_type << 4);
*p_result++ = A2DP_MEDIA_CT_AAC;
// Object Type
if (p_ie->objectType == 0) return A2DP_INVALID_PARAMS;
*p_result++ = p_ie->objectType;
// Sampling Frequency
if (p_ie->sampleRate == 0) return A2DP_INVALID_PARAMS;
*p_result++ = (uint8_t)(p_ie->sampleRate & A2DP_AAC_SAMPLING_FREQ_MASK0);
*p_result = (uint8_t)((p_ie->sampleRate & A2DP_AAC_SAMPLING_FREQ_MASK1) >> 8);
// Channel Mode
if (p_ie->channelMode == 0) return A2DP_INVALID_PARAMS;
*p_result++ |= (p_ie->channelMode & A2DP_AAC_CHANNEL_MODE_MASK);
// Variable Bit Rate Support
*p_result = (p_ie->variableBitRateSupport & A2DP_AAC_VARIABLE_BIT_RATE_MASK);
// Bit Rate
*p_result++ |= (uint8_t)((p_ie->bitRate & A2DP_AAC_BIT_RATE_MASK0) >> 16);
*p_result++ = (uint8_t)((p_ie->bitRate & A2DP_AAC_BIT_RATE_MASK1) >> 8);
*p_result++ = (uint8_t)(p_ie->bitRate & A2DP_AAC_BIT_RATE_MASK2);
return A2DP_SUCCESS;
}
// Parses the AAC Media Codec Capabilities byte sequence beginning from the
// LOSC octet. The result is stored in |p_ie|. The byte sequence to parse is
// |p_codec_info|. If |is_capability| is true, the byte sequence is
// codec capabilities, otherwise is codec configuration.
// Returns A2DP_SUCCESS on success, otherwise the corresponding A2DP error
// status code.
static tA2DP_STATUS A2DP_ParseInfoAac(tA2DP_AAC_CIE* p_ie,
const uint8_t* p_codec_info,
bool is_capability) {
uint8_t losc;
uint8_t media_type;
tA2DP_CODEC_TYPE codec_type;
if (p_ie == NULL || p_codec_info == NULL) return A2DP_INVALID_PARAMS;
// Check the codec capability length
losc = *p_codec_info++;
if (losc != A2DP_AAC_CODEC_LEN) return A2DP_WRONG_CODEC;
media_type = (*p_codec_info++) >> 4;
codec_type = *p_codec_info++;
/* Check the Media Type and Media Codec Type */
if (media_type != AVDT_MEDIA_TYPE_AUDIO || codec_type != A2DP_MEDIA_CT_AAC) {
return A2DP_WRONG_CODEC;
}
p_ie->objectType = *p_codec_info++;
p_ie->sampleRate = (*p_codec_info & A2DP_AAC_SAMPLING_FREQ_MASK0) |
(*(p_codec_info + 1) << 8 & A2DP_AAC_SAMPLING_FREQ_MASK1);
p_codec_info++;
p_ie->channelMode = *p_codec_info & A2DP_AAC_CHANNEL_MODE_MASK;
p_codec_info++;
p_ie->variableBitRateSupport =
*p_codec_info & A2DP_AAC_VARIABLE_BIT_RATE_MASK;
p_ie->bitRate = ((*p_codec_info) << 16 & A2DP_AAC_BIT_RATE_MASK0) |
(*(p_codec_info + 1) << 8 & A2DP_AAC_BIT_RATE_MASK1) |
(*(p_codec_info + 2) & A2DP_AAC_BIT_RATE_MASK2);
p_codec_info += 3;
if (is_capability) {
// NOTE: The checks here are very liberal. We should be using more
// pedantic checks specific to the SRC or SNK as specified in the spec.
if (A2DP_BitsSet(p_ie->objectType) == A2DP_SET_ZERO_BIT)
return A2DP_BAD_OBJ_TYPE;
if (A2DP_BitsSet(p_ie->sampleRate) == A2DP_SET_ZERO_BIT)
return A2DP_BAD_SAMP_FREQ;
if (A2DP_BitsSet(p_ie->channelMode) == A2DP_SET_ZERO_BIT)
return A2DP_BAD_CH_MODE;
return A2DP_SUCCESS;
}
if (A2DP_BitsSet(p_ie->objectType) != A2DP_SET_ONE_BIT)
return A2DP_BAD_OBJ_TYPE;
if (A2DP_BitsSet(p_ie->sampleRate) != A2DP_SET_ONE_BIT)
return A2DP_BAD_SAMP_FREQ;
if (A2DP_BitsSet(p_ie->channelMode) != A2DP_SET_ONE_BIT)
return A2DP_BAD_CH_MODE;
return A2DP_SUCCESS;
}
bool A2DP_IsSourceCodecValidAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE cfg_cie;
/* Use a liberal check when parsing the codec info */
return (A2DP_ParseInfoAac(&cfg_cie, p_codec_info, false) == A2DP_SUCCESS) ||
(A2DP_ParseInfoAac(&cfg_cie, p_codec_info, true) == A2DP_SUCCESS);
}
bool A2DP_IsSinkCodecValidAac(UNUSED_ATTR const uint8_t* p_codec_info) {
tA2DP_AAC_CIE cfg_cie;
/* Use a liberal check when parsing the codec info */
return (A2DP_ParseInfoAac(&cfg_cie, p_codec_info, false) == A2DP_SUCCESS) ||
(A2DP_ParseInfoAac(&cfg_cie, p_codec_info, true) == A2DP_SUCCESS);
}
bool A2DP_IsPeerSourceCodecValidAac(UNUSED_ATTR const uint8_t* p_codec_info) {
tA2DP_AAC_CIE cfg_cie;
/* Use a liberal check when parsing the codec info */
return (A2DP_ParseInfoAac(&cfg_cie, p_codec_info, false) == A2DP_SUCCESS) ||
(A2DP_ParseInfoAac(&cfg_cie, p_codec_info, true) == A2DP_SUCCESS);
}
bool A2DP_IsPeerSinkCodecValidAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE cfg_cie;
/* Use a liberal check when parsing the codec info */
return (A2DP_ParseInfoAac(&cfg_cie, p_codec_info, false) == A2DP_SUCCESS) ||
(A2DP_ParseInfoAac(&cfg_cie, p_codec_info, true) == A2DP_SUCCESS);
}
bool A2DP_IsSinkCodecSupportedAac(const uint8_t* p_codec_info) {
return A2DP_CodecInfoMatchesCapabilityAac(&a2dp_aac_sink_caps, p_codec_info,
false) == A2DP_SUCCESS;
}
bool A2DP_IsPeerSourceCodecSupportedAac(const uint8_t* p_codec_info) {
return A2DP_CodecInfoMatchesCapabilityAac(&a2dp_aac_sink_caps, p_codec_info,
true) == A2DP_SUCCESS;
}
// Checks whether A2DP AAC codec configuration matches with a device's codec
// capabilities. |p_cap| is the AAC codec configuration. |p_codec_info| is
// the device's codec capabilities. |is_capability| is true if
// |p_codec_info| contains A2DP codec capability.
// Returns A2DP_SUCCESS if the codec configuration matches with capabilities,
// otherwise the corresponding A2DP error status code.
static tA2DP_STATUS A2DP_CodecInfoMatchesCapabilityAac(
const tA2DP_AAC_CIE* p_cap, const uint8_t* p_codec_info,
bool is_capability) {
tA2DP_STATUS status;
tA2DP_AAC_CIE cfg_cie;
/* parse configuration */
status = A2DP_ParseInfoAac(&cfg_cie, p_codec_info, is_capability);
if (status != A2DP_SUCCESS) {
log::error("parsing failed {}", status);
return status;
}
/* verify that each parameter is in range */
log::verbose("Object Type peer: 0x{:x}, capability 0x{:x}",
cfg_cie.objectType, p_cap->objectType);
log::verbose("Sample Rate peer: {}, capability {}", cfg_cie.sampleRate,
p_cap->sampleRate);
log::verbose("Channel Mode peer: 0x{:x}, capability 0x{:x}",
cfg_cie.channelMode, p_cap->channelMode);
log::verbose("Variable Bit Rate Support peer: 0x{:x}, capability 0x{:x}",
cfg_cie.variableBitRateSupport, p_cap->variableBitRateSupport);
log::verbose("Bit Rate peer: {}, capability {}", cfg_cie.bitRate,
p_cap->bitRate);
/* Object Type */
if ((cfg_cie.objectType & p_cap->objectType) == 0) return A2DP_BAD_OBJ_TYPE;
/* Sample Rate */
if ((cfg_cie.sampleRate & p_cap->sampleRate) == 0) return A2DP_BAD_SAMP_FREQ;
/* Channel Mode */
if ((cfg_cie.channelMode & p_cap->channelMode) == 0) return A2DP_NS_CH_MODE;
return A2DP_SUCCESS;
}
bool A2DP_UsesRtpHeaderAac(UNUSED_ATTR bool content_protection_enabled,
UNUSED_ATTR const uint8_t* p_codec_info) {
return true;
}
const char* A2DP_CodecNameAac(UNUSED_ATTR const uint8_t* p_codec_info) {
return "AAC";
}
bool A2DP_CodecTypeEqualsAac(const uint8_t* p_codec_info_a,
const uint8_t* p_codec_info_b) {
tA2DP_AAC_CIE aac_cie_a;
tA2DP_AAC_CIE aac_cie_b;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status =
A2DP_ParseInfoAac(&aac_cie_a, p_codec_info_a, true);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return false;
}
a2dp_status = A2DP_ParseInfoAac(&aac_cie_b, p_codec_info_b, true);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return false;
}
return true;
}
bool A2DP_CodecEqualsAac(const uint8_t* p_codec_info_a,
const uint8_t* p_codec_info_b) {
tA2DP_AAC_CIE aac_cie_a;
tA2DP_AAC_CIE aac_cie_b;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status =
A2DP_ParseInfoAac(&aac_cie_a, p_codec_info_a, true);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return false;
}
a2dp_status = A2DP_ParseInfoAac(&aac_cie_b, p_codec_info_b, true);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return false;
}
return (aac_cie_a.objectType == aac_cie_b.objectType) &&
(aac_cie_a.sampleRate == aac_cie_b.sampleRate) &&
(aac_cie_a.channelMode == aac_cie_b.channelMode) &&
(aac_cie_a.variableBitRateSupport ==
aac_cie_b.variableBitRateSupport) &&
(aac_cie_a.bitRate == aac_cie_b.bitRate);
}
int A2DP_GetTrackSampleRateAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
switch (aac_cie.sampleRate) {
case A2DP_AAC_SAMPLING_FREQ_8000:
return 8000;
case A2DP_AAC_SAMPLING_FREQ_11025:
return 11025;
case A2DP_AAC_SAMPLING_FREQ_12000:
return 12000;
case A2DP_AAC_SAMPLING_FREQ_16000:
return 16000;
case A2DP_AAC_SAMPLING_FREQ_22050:
return 22050;
case A2DP_AAC_SAMPLING_FREQ_24000:
return 24000;
case A2DP_AAC_SAMPLING_FREQ_32000:
return 32000;
case A2DP_AAC_SAMPLING_FREQ_44100:
return 44100;
case A2DP_AAC_SAMPLING_FREQ_48000:
return 48000;
case A2DP_AAC_SAMPLING_FREQ_64000:
return 64000;
case A2DP_AAC_SAMPLING_FREQ_88200:
return 88200;
case A2DP_AAC_SAMPLING_FREQ_96000:
return 96000;
}
return -1;
}
int A2DP_GetTrackBitsPerSampleAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
// NOTE: The bits per sample never changes for AAC
return 16;
}
int A2DP_GetTrackChannelCountAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
switch (aac_cie.channelMode) {
case A2DP_AAC_CHANNEL_MODE_MONO:
return 1;
case A2DP_AAC_CHANNEL_MODE_STEREO:
return 2;
}
return -1;
}
int A2DP_GetSinkTrackChannelTypeAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
switch (aac_cie.channelMode) {
case A2DP_AAC_CHANNEL_MODE_MONO:
return 1;
case A2DP_AAC_CHANNEL_MODE_STEREO:
return 3;
}
return -1;
}
int A2DP_GetObjectTypeCodeAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
switch (aac_cie.objectType) {
case A2DP_AAC_OBJECT_TYPE_MPEG2_LC:
case A2DP_AAC_OBJECT_TYPE_MPEG4_LC:
case A2DP_AAC_OBJECT_TYPE_MPEG4_LTP:
case A2DP_AAC_OBJECT_TYPE_MPEG4_SCALABLE:
return aac_cie.objectType;
default:
break;
}
return -1;
}
int A2DP_GetChannelModeCodeAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
switch (aac_cie.channelMode) {
case A2DP_AAC_CHANNEL_MODE_MONO:
case A2DP_AAC_CHANNEL_MODE_STEREO:
return aac_cie.channelMode;
default:
break;
}
return -1;
}
int A2DP_GetVariableBitRateSupportAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
switch (aac_cie.variableBitRateSupport) {
case A2DP_AAC_VARIABLE_BIT_RATE_ENABLED:
case A2DP_AAC_VARIABLE_BIT_RATE_DISABLED:
return aac_cie.variableBitRateSupport;
default:
break;
}
return -1;
}
int A2DP_GetBitRateAac(const uint8_t* p_codec_info) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
return aac_cie.bitRate;
}
int A2DP_ComputeMaxBitRateAac(const uint8_t* p_codec_info, uint16_t mtu) {
tA2DP_AAC_CIE aac_cie;
// Check whether the codec info contains valid data
tA2DP_STATUS a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, false);
if (a2dp_status != A2DP_SUCCESS) {
log::error("cannot decode codec information: {}", a2dp_status);
return -1;
}
int sampling_freq = A2DP_GetTrackSampleRateAac(p_codec_info);
if (sampling_freq == -1) return -1;
int pcm_channel_samples_per_frame = 0;
switch (aac_cie.objectType) {
case A2DP_AAC_OBJECT_TYPE_MPEG2_LC:
case A2DP_AAC_OBJECT_TYPE_MPEG4_LC:
pcm_channel_samples_per_frame = 1024;
break;
case A2DP_AAC_OBJECT_TYPE_MPEG4_LTP:
case A2DP_AAC_OBJECT_TYPE_MPEG4_SCALABLE:
// TODO: The MPEG documentation doesn't specify the value.
break;
default:
break;
}
if (pcm_channel_samples_per_frame == 0) return -1;
// See Section 3.2.1 Estimating Average Frame Size from
// the aacEncoder.pdf document included with the AAC source code.
return (8 * mtu * sampling_freq) / pcm_channel_samples_per_frame;
}
bool A2DP_GetPacketTimestampAac(const uint8_t* p_codec_info,
const uint8_t* p_data, uint32_t* p_timestamp) {
// TODO: Is this function really codec-specific?
*p_timestamp = *(const uint32_t*)p_data;
return true;
}
bool A2DP_BuildCodecHeaderAac(UNUSED_ATTR const uint8_t* p_codec_info,
UNUSED_ATTR BT_HDR* p_buf,
UNUSED_ATTR uint16_t frames_per_packet) {
return true;
}
std::string A2DP_CodecInfoStringAac(const uint8_t* p_codec_info) {
std::stringstream res;
std::string field;
tA2DP_STATUS a2dp_status;
tA2DP_AAC_CIE aac_cie;
a2dp_status = A2DP_ParseInfoAac(&aac_cie, p_codec_info, true);
if (a2dp_status != A2DP_SUCCESS) {
res << "A2DP_ParseInfoAac fail: " << loghex(a2dp_status);
return res.str();
}
res << "\tname: AAC\n";
// Object type
field.clear();
AppendField(&field, (aac_cie.objectType == 0), "NONE");
AppendField(&field, (aac_cie.objectType & A2DP_AAC_OBJECT_TYPE_MPEG2_LC),
"(MPEG-2 AAC LC)");
AppendField(&field, (aac_cie.objectType & A2DP_AAC_OBJECT_TYPE_MPEG4_LC),
"(MPEG-4 AAC LC)");
AppendField(&field, (aac_cie.objectType & A2DP_AAC_OBJECT_TYPE_MPEG4_LTP),
"(MPEG-4 AAC LTP)");
AppendField(&field,
(aac_cie.objectType & A2DP_AAC_OBJECT_TYPE_MPEG4_SCALABLE),
"(MPEG-4 AAC Scalable)");
res << "\tobjectType: " << field << " (" << loghex(aac_cie.objectType)
<< ")\n";
// Sample frequency
field.clear();
AppendField(&field, (aac_cie.sampleRate == 0), "NONE");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_8000),
"8000");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_11025),
"11025");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_12000),
"12000");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_16000),
"16000");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_22050),
"22050");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_24000),
"24000");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_32000),
"32000");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_44100),
"44100");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_48000),
"48000");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_64000),
"64000");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_88200),
"88200");
AppendField(&field, (aac_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_96000),
"96000");
res << "\tsamp_freq: " << field << " (" << loghex(aac_cie.sampleRate)
<< ")\n";
// Channel mode
field.clear();
AppendField(&field, (aac_cie.channelMode == 0), "NONE");
AppendField(&field, (aac_cie.channelMode == A2DP_AAC_CHANNEL_MODE_MONO),
"Mono");
AppendField(&field, (aac_cie.channelMode == A2DP_AAC_CHANNEL_MODE_STEREO),
"Stereo");
res << "\tch_mode: " << field << " (" << loghex(aac_cie.channelMode) << ")\n";
// Variable bit rate support
res << "\tvariableBitRateSupport: " << std::boolalpha
<< (aac_cie.variableBitRateSupport != 0) << "\n";
// Bit rate
res << "\tbitRate: " << std::to_string(aac_cie.bitRate) << "\n";
return res.str();
}
const tA2DP_ENCODER_INTERFACE* A2DP_GetEncoderInterfaceAac(
const uint8_t* p_codec_info) {
if (!A2DP_IsSourceCodecValidAac(p_codec_info)) return NULL;
return &a2dp_encoder_interface_aac;
}
const tA2DP_DECODER_INTERFACE* A2DP_GetDecoderInterfaceAac(
const uint8_t* p_codec_info) {
if (!A2DP_IsSinkCodecValidAac(p_codec_info)) return NULL;
return &a2dp_decoder_interface_aac;
}
bool A2DP_AdjustCodecAac(uint8_t* p_codec_info) {
tA2DP_AAC_CIE cfg_cie;
// Nothing to do: just verify the codec info is valid
if (A2DP_ParseInfoAac(&cfg_cie, p_codec_info, true) != A2DP_SUCCESS)
return false;
return true;
}
btav_a2dp_codec_index_t A2DP_SourceCodecIndexAac(
UNUSED_ATTR const uint8_t* p_codec_info) {
return BTAV_A2DP_CODEC_INDEX_SOURCE_AAC;
}
btav_a2dp_codec_index_t A2DP_SinkCodecIndexAac(
UNUSED_ATTR const uint8_t* p_codec_info) {
return BTAV_A2DP_CODEC_INDEX_SINK_AAC;
}
const char* A2DP_CodecIndexStrAac(void) { return "AAC"; }
const char* A2DP_CodecIndexStrAacSink(void) { return "AAC SINK"; }
void aac_source_caps_initialize() {
if (aac_source_caps_configured) {
return;
}
a2dp_aac_source_caps =
osi_property_get_bool("persist.bluetooth.a2dp_aac.vbr_supported", false)
? a2dp_aac_vbr_source_caps
: a2dp_aac_cbr_source_caps;
aac_source_caps_configured = true;
}
bool A2DP_InitCodecConfigAac(AvdtpSepConfig* p_cfg) {
aac_source_caps_initialize();
if (A2DP_BuildInfoAac(AVDT_MEDIA_TYPE_AUDIO, &a2dp_aac_source_caps,
p_cfg->codec_info) != A2DP_SUCCESS) {
return false;
}
return true;
}
bool A2DP_InitCodecConfigAacSink(AvdtpSepConfig* p_cfg) {
return A2DP_BuildInfoAac(AVDT_MEDIA_TYPE_AUDIO, &a2dp_aac_sink_caps,
p_cfg->codec_info) == A2DP_SUCCESS;
}
UNUSED_ATTR static void build_codec_config(const tA2DP_AAC_CIE& config_cie,
btav_a2dp_codec_config_t* result) {
if (config_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_44100)
result->sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_44100;
if (config_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_48000)
result->sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_48000;
if (config_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_88200)
result->sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_88200;
if (config_cie.sampleRate & A2DP_AAC_SAMPLING_FREQ_96000)
result->sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_96000;
result->bits_per_sample = config_cie.bits_per_sample;
if (config_cie.channelMode & A2DP_AAC_CHANNEL_MODE_MONO)
result->channel_mode |= BTAV_A2DP_CODEC_CHANNEL_MODE_MONO;
if (config_cie.channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
result->channel_mode |= BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO;
}
}
A2dpCodecConfigAacSource::A2dpCodecConfigAacSource(
btav_a2dp_codec_priority_t codec_priority)
: A2dpCodecConfigAacBase(BTAV_A2DP_CODEC_INDEX_SOURCE_AAC,
A2DP_CodecIndexStrAac(), codec_priority, true) {
aac_source_caps_initialize();
// Compute the local capability
if (a2dp_aac_source_caps.sampleRate & A2DP_AAC_SAMPLING_FREQ_44100) {
codec_local_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_44100;
}
if (a2dp_aac_source_caps.sampleRate & A2DP_AAC_SAMPLING_FREQ_48000) {
codec_local_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_48000;
}
if (a2dp_aac_source_caps.sampleRate & A2DP_AAC_SAMPLING_FREQ_88200) {
codec_local_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_88200;
}
if (a2dp_aac_source_caps.sampleRate & A2DP_AAC_SAMPLING_FREQ_96000) {
codec_local_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_96000;
}
codec_local_capability_.bits_per_sample =
a2dp_aac_source_caps.bits_per_sample;
if (a2dp_aac_source_caps.channelMode & A2DP_AAC_CHANNEL_MODE_MONO) {
codec_local_capability_.channel_mode |= BTAV_A2DP_CODEC_CHANNEL_MODE_MONO;
}
if (a2dp_aac_source_caps.channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
codec_local_capability_.channel_mode |= BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO;
}
}
A2dpCodecConfigAacSource::~A2dpCodecConfigAacSource() {}
bool A2dpCodecConfigAacSource::init() {
if (!isValid()) return false;
// Load the encoder
if (!A2DP_LoadEncoderAac()) {
log::error("cannot load the encoder");
return false;
}
return true;
}
bool A2dpCodecConfigAacSource::useRtpHeaderMarkerBit() const { return true; }
//
// Selects the best sample rate from |sampleRate|.
// The result is stored in |p_result| and |p_codec_config|.
// Returns true if a selection was made, otherwise false.
//
static bool select_best_sample_rate(uint16_t sampleRate,
tA2DP_AAC_CIE* p_result,
btav_a2dp_codec_config_t* p_codec_config) {
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_96000) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_96000;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_96000;
return true;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_88200) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_88200;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_88200;
return true;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_48000) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_48000;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_48000;
return true;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_44100) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_44100;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_44100;
return true;
}
return false;
}
//
// Selects the audio sample rate from |p_codec_audio_config|.
// |sampleRate| contains the capability.
// The result is stored in |p_result| and |p_codec_config|.
// Returns true if a selection was made, otherwise false.
//
static bool select_audio_sample_rate(
const btav_a2dp_codec_config_t* p_codec_audio_config, uint16_t sampleRate,
tA2DP_AAC_CIE* p_result, btav_a2dp_codec_config_t* p_codec_config) {
switch (p_codec_audio_config->sample_rate) {
case BTAV_A2DP_CODEC_SAMPLE_RATE_44100:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_44100) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_44100;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_44100;
return true;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_48000:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_48000) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_48000;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_48000;
return true;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_88200:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_88200) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_88200;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_88200;
return true;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_96000:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_96000) {
p_result->sampleRate = A2DP_AAC_SAMPLING_FREQ_96000;
p_codec_config->sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_96000;
return true;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_176400:
case BTAV_A2DP_CODEC_SAMPLE_RATE_192000:
case BTAV_A2DP_CODEC_SAMPLE_RATE_16000:
case BTAV_A2DP_CODEC_SAMPLE_RATE_24000:
case BTAV_A2DP_CODEC_SAMPLE_RATE_NONE:
break;
}
return false;
}
//
// Selects the best bits per sample from |bits_per_sample|.
// |bits_per_sample| contains the capability.
// The result is stored in |p_result| and |p_codec_config|.
// Returns true if a selection was made, otherwise false.
//
static bool select_best_bits_per_sample(
btav_a2dp_codec_bits_per_sample_t bits_per_sample, tA2DP_AAC_CIE* p_result,
btav_a2dp_codec_config_t* p_codec_config) {
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32) {
p_codec_config->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32;
p_result->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32;
return true;
}
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24) {
p_codec_config->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24;
p_result->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24;
return true;
}
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16) {
p_codec_config->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16;
p_result->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16;
return true;
}
return false;
}
//
// Selects the audio bits per sample from |p_codec_audio_config|.
// |bits_per_sample| contains the capability.
// The result is stored in |p_result| and |p_codec_config|.
// Returns true if a selection was made, otherwise false.
//
static bool select_audio_bits_per_sample(
const btav_a2dp_codec_config_t* p_codec_audio_config,
btav_a2dp_codec_bits_per_sample_t bits_per_sample, tA2DP_AAC_CIE* p_result,
btav_a2dp_codec_config_t* p_codec_config) {
switch (p_codec_audio_config->bits_per_sample) {
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16:
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16) {
p_codec_config->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16;
p_result->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16;
return true;
}
break;
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24:
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24) {
p_codec_config->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24;
p_result->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24;
return true;
}
break;
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32:
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32) {
p_codec_config->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32;
p_result->bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32;
return true;
}
break;
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE:
break;
}
return false;
}
//
// Selects the best channel mode from |channelMode|.
// The result is stored in |p_result| and |p_codec_config|.
// Returns true if a selection was made, otherwise false.
//
static bool select_best_channel_mode(uint8_t channelMode,
tA2DP_AAC_CIE* p_result,
btav_a2dp_codec_config_t* p_codec_config) {
if (channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
p_result->channelMode = A2DP_AAC_CHANNEL_MODE_STEREO;
p_codec_config->channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO;
return true;
}
if (channelMode & A2DP_AAC_CHANNEL_MODE_MONO) {
p_result->channelMode = A2DP_AAC_CHANNEL_MODE_MONO;
p_codec_config->channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_MONO;
return true;
}
return false;
}
//
// Selects the audio channel mode from |p_codec_audio_config|.
// |channelMode| contains the capability.
// The result is stored in |p_result| and |p_codec_config|.
// Returns true if a selection was made, otherwise false.
//
static bool select_audio_channel_mode(
const btav_a2dp_codec_config_t* p_codec_audio_config, uint8_t channelMode,
tA2DP_AAC_CIE* p_result, btav_a2dp_codec_config_t* p_codec_config) {
switch (p_codec_audio_config->channel_mode) {
case BTAV_A2DP_CODEC_CHANNEL_MODE_MONO:
if (channelMode & A2DP_AAC_CHANNEL_MODE_MONO) {
p_result->channelMode = A2DP_AAC_CHANNEL_MODE_MONO;
p_codec_config->channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_MONO;
return true;
}
break;
case BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO:
if (channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
p_result->channelMode = A2DP_AAC_CHANNEL_MODE_STEREO;
p_codec_config->channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO;
return true;
}
break;
case BTAV_A2DP_CODEC_CHANNEL_MODE_NONE:
break;
}
return false;
}
bool A2dpCodecConfigAacBase::setCodecConfig(const uint8_t* p_peer_codec_info,
bool is_capability,
uint8_t* p_result_codec_config) {
std::lock_guard<std::recursive_mutex> lock(codec_mutex_);
tA2DP_AAC_CIE peer_info_cie;
tA2DP_AAC_CIE result_config_cie;
uint8_t channelMode;
uint16_t sampleRate;
btav_a2dp_codec_bits_per_sample_t bits_per_sample;
const tA2DP_AAC_CIE* p_a2dp_aac_caps =
(is_source_) ? &a2dp_aac_source_caps : &a2dp_aac_sink_caps;
// Save the internal state
btav_a2dp_codec_config_t saved_codec_config = codec_config_;
btav_a2dp_codec_config_t saved_codec_capability = codec_capability_;
btav_a2dp_codec_config_t saved_codec_selectable_capability =
codec_selectable_capability_;
btav_a2dp_codec_config_t saved_codec_user_config = codec_user_config_;
btav_a2dp_codec_config_t saved_codec_audio_config = codec_audio_config_;
uint8_t saved_ota_codec_config[AVDT_CODEC_SIZE];
uint8_t saved_ota_codec_peer_capability[AVDT_CODEC_SIZE];
uint8_t saved_ota_codec_peer_config[AVDT_CODEC_SIZE];
memcpy(saved_ota_codec_config, ota_codec_config_, sizeof(ota_codec_config_));
memcpy(saved_ota_codec_peer_capability, ota_codec_peer_capability_,
sizeof(ota_codec_peer_capability_));
memcpy(saved_ota_codec_peer_config, ota_codec_peer_config_,
sizeof(ota_codec_peer_config_));
tA2DP_STATUS status =
A2DP_ParseInfoAac(&peer_info_cie, p_peer_codec_info, is_capability);
if (status != A2DP_SUCCESS) {
log::error("can't parse peer's capabilities: error = {}", status);
goto fail;
}
//
// Build the preferred configuration
//
memset(&result_config_cie, 0, sizeof(result_config_cie));
// NOTE: Always assign the Object Type and Variable Bit Rate Support.
result_config_cie.objectType = p_a2dp_aac_caps->objectType;
// The Variable Bit Rate Support is disabled if either side disables it
result_config_cie.variableBitRateSupport =
p_a2dp_aac_caps->variableBitRateSupport &
peer_info_cie.variableBitRateSupport;
if (result_config_cie.variableBitRateSupport !=
A2DP_AAC_VARIABLE_BIT_RATE_DISABLED) {
codec_config_.codec_specific_1 =
static_cast<int64_t>(AacEncoderBitrateMode::AACENC_BR_MODE_VBR_5);
} else {
codec_config_.codec_specific_1 =
static_cast<int64_t>(AacEncoderBitrateMode::AACENC_BR_MODE_CBR);
}
// Set the bit rate as follows:
// 1. If the remote device reports a bogus bit rate
// (bitRate < A2DP_AAC_MIN_BITRATE), then use the bit rate from our
// configuration. Examples of observed bogus bit rates are zero
// and 24576.
// 2. If the remote device reports valid bit rate
// (bitRate >= A2DP_AAC_MIN_BITRATE), then use the smaller
// of the remote device's bit rate and the bit rate from our configuration.
// In either case, the actual streaming bit rate will also consider the MTU.
if (peer_info_cie.bitRate < A2DP_AAC_MIN_BITRATE) {
// Bogus bit rate
result_config_cie.bitRate = p_a2dp_aac_caps->bitRate;
} else {
result_config_cie.bitRate =
std::min(p_a2dp_aac_caps->bitRate, peer_info_cie.bitRate);
}
//
// Select the sample frequency
//
sampleRate = p_a2dp_aac_caps->sampleRate & peer_info_cie.sampleRate;
codec_config_.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_NONE;
switch (codec_user_config_.sample_rate) {
case BTAV_A2DP_CODEC_SAMPLE_RATE_44100:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_44100) {
result_config_cie.sampleRate = A2DP_AAC_SAMPLING_FREQ_44100;
codec_capability_.sample_rate = codec_user_config_.sample_rate;
codec_config_.sample_rate = codec_user_config_.sample_rate;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_48000:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_48000) {
result_config_cie.sampleRate = A2DP_AAC_SAMPLING_FREQ_48000;
codec_capability_.sample_rate = codec_user_config_.sample_rate;
codec_config_.sample_rate = codec_user_config_.sample_rate;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_88200:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_88200) {
result_config_cie.sampleRate = A2DP_AAC_SAMPLING_FREQ_88200;
codec_capability_.sample_rate = codec_user_config_.sample_rate;
codec_config_.sample_rate = codec_user_config_.sample_rate;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_96000:
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_96000) {
result_config_cie.sampleRate = A2DP_AAC_SAMPLING_FREQ_96000;
codec_capability_.sample_rate = codec_user_config_.sample_rate;
codec_config_.sample_rate = codec_user_config_.sample_rate;
}
break;
case BTAV_A2DP_CODEC_SAMPLE_RATE_176400:
case BTAV_A2DP_CODEC_SAMPLE_RATE_192000:
case BTAV_A2DP_CODEC_SAMPLE_RATE_16000:
case BTAV_A2DP_CODEC_SAMPLE_RATE_24000:
case BTAV_A2DP_CODEC_SAMPLE_RATE_NONE:
codec_capability_.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_NONE;
codec_config_.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_NONE;
break;
}
// Select the sample frequency if there is no user preference
do {
// Compute the selectable capability
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_44100) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_44100;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_48000) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_48000;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_88200) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_88200;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_96000) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_96000;
}
if (codec_config_.sample_rate != BTAV_A2DP_CODEC_SAMPLE_RATE_NONE) break;
// Compute the common capability
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_44100)
codec_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_44100;
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_48000)
codec_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_48000;
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_88200)
codec_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_88200;
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_96000)
codec_capability_.sample_rate |= BTAV_A2DP_CODEC_SAMPLE_RATE_96000;
// No user preference - try the codec audio config
if (select_audio_sample_rate(&codec_audio_config_, sampleRate,
&result_config_cie, &codec_config_)) {
break;
}
// No user preference - try the default config
if (select_best_sample_rate(
a2dp_aac_default_config.sampleRate & peer_info_cie.sampleRate,
&result_config_cie, &codec_config_)) {
break;
}
// No user preference - use the best match
if (select_best_sample_rate(sampleRate, &result_config_cie,
&codec_config_)) {
break;
}
} while (false);
if (codec_config_.sample_rate == BTAV_A2DP_CODEC_SAMPLE_RATE_NONE) {
log::error(
"cannot match sample frequency: source caps = 0x{:x} peer info = "
"0x{:x}",
p_a2dp_aac_caps->sampleRate, peer_info_cie.sampleRate);
goto fail;
}
//
// Select the bits per sample
//
// NOTE: this information is NOT included in the AAC A2DP codec description
// that is sent OTA.
bits_per_sample = p_a2dp_aac_caps->bits_per_sample;
codec_config_.bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE;
switch (codec_user_config_.bits_per_sample) {
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16:
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16) {
result_config_cie.bits_per_sample = codec_user_config_.bits_per_sample;
codec_capability_.bits_per_sample = codec_user_config_.bits_per_sample;
codec_config_.bits_per_sample = codec_user_config_.bits_per_sample;
}
break;
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24:
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_24) {
result_config_cie.bits_per_sample = codec_user_config_.bits_per_sample;
codec_capability_.bits_per_sample = codec_user_config_.bits_per_sample;
codec_config_.bits_per_sample = codec_user_config_.bits_per_sample;
}
break;
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32:
if (bits_per_sample & BTAV_A2DP_CODEC_BITS_PER_SAMPLE_32) {
result_config_cie.bits_per_sample = codec_user_config_.bits_per_sample;
codec_capability_.bits_per_sample = codec_user_config_.bits_per_sample;
codec_config_.bits_per_sample = codec_user_config_.bits_per_sample;
}
break;
case BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE:
result_config_cie.bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE;
codec_capability_.bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE;
codec_config_.bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE;
break;
}
// Select the bits per sample if there is no user preference
do {
// Compute the selectable capability
codec_selectable_capability_.bits_per_sample =
p_a2dp_aac_caps->bits_per_sample;
if (codec_config_.bits_per_sample != BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE)
break;
// Compute the common capability
codec_capability_.bits_per_sample = bits_per_sample;
// No user preference - the the codec audio config
if (select_audio_bits_per_sample(&codec_audio_config_,
p_a2dp_aac_caps->bits_per_sample,
&result_config_cie, &codec_config_)) {
break;
}
// No user preference - try the default config
if (select_best_bits_per_sample(a2dp_aac_default_config.bits_per_sample,
&result_config_cie, &codec_config_)) {
break;
}
// No user preference - use the best match
if (select_best_bits_per_sample(p_a2dp_aac_caps->bits_per_sample,
&result_config_cie, &codec_config_)) {
break;
}
} while (false);
if (codec_config_.bits_per_sample == BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE) {
log::error(
"cannot match bits per sample: default = 0x{:x} user preference = "
"0x{:x}",
a2dp_aac_default_config.bits_per_sample,
codec_user_config_.bits_per_sample);
goto fail;
}
//
// Select the channel mode
//
channelMode = p_a2dp_aac_caps->channelMode & peer_info_cie.channelMode;
codec_config_.channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_NONE;
switch (codec_user_config_.channel_mode) {
case BTAV_A2DP_CODEC_CHANNEL_MODE_MONO:
if (channelMode & A2DP_AAC_CHANNEL_MODE_MONO) {
result_config_cie.channelMode = A2DP_AAC_CHANNEL_MODE_MONO;
codec_capability_.channel_mode = codec_user_config_.channel_mode;
codec_config_.channel_mode = codec_user_config_.channel_mode;
}
break;
case BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO:
if (channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
result_config_cie.channelMode = A2DP_AAC_CHANNEL_MODE_STEREO;
codec_capability_.channel_mode = codec_user_config_.channel_mode;
codec_config_.channel_mode = codec_user_config_.channel_mode;
}
break;
case BTAV_A2DP_CODEC_CHANNEL_MODE_NONE:
codec_capability_.channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_NONE;
codec_config_.channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_NONE;
break;
}
// Select the channel mode if there is no user preference
do {
// Compute the selectable capability
if (channelMode & A2DP_AAC_CHANNEL_MODE_MONO) {
codec_selectable_capability_.channel_mode |=
BTAV_A2DP_CODEC_CHANNEL_MODE_MONO;
}
if (channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
codec_selectable_capability_.channel_mode |=
BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO;
}
if (codec_config_.channel_mode != BTAV_A2DP_CODEC_CHANNEL_MODE_NONE) break;
// Compute the common capability
if (channelMode & A2DP_AAC_CHANNEL_MODE_MONO)
codec_capability_.channel_mode |= BTAV_A2DP_CODEC_CHANNEL_MODE_MONO;
if (channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
codec_capability_.channel_mode |= BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO;
}
// No user preference - try the codec audio config
if (select_audio_channel_mode(&codec_audio_config_, channelMode,
&result_config_cie, &codec_config_)) {
break;
}
// No user preference - try the default config
if (select_best_channel_mode(
a2dp_aac_default_config.channelMode & peer_info_cie.channelMode,
&result_config_cie, &codec_config_)) {
break;
}
// No user preference - use the best match
if (select_best_channel_mode(channelMode, &result_config_cie,
&codec_config_)) {
break;
}
} while (false);
if (codec_config_.channel_mode == BTAV_A2DP_CODEC_CHANNEL_MODE_NONE) {
log::error(
"cannot match channel mode: source caps = 0x{:x} peer info = 0x{:x}",
p_a2dp_aac_caps->channelMode, peer_info_cie.channelMode);
goto fail;
}
//
// Copy the codec-specific fields if they are not zero
//
if (codec_user_config_.codec_specific_1 != 0) {
if (result_config_cie.variableBitRateSupport !=
A2DP_AAC_VARIABLE_BIT_RATE_DISABLED) {
auto user_bitrate_mode = codec_user_config_.codec_specific_1;
switch (static_cast<AacEncoderBitrateMode>(user_bitrate_mode)) {
case AacEncoderBitrateMode::AACENC_BR_MODE_VBR_C:
// VBR is supported, and is disabled by the user preference
result_config_cie.variableBitRateSupport =
A2DP_AAC_VARIABLE_BIT_RATE_DISABLED;
[[fallthrough]];
case AacEncoderBitrateMode::AACENC_BR_MODE_VBR_1:
[[fallthrough]];
case AacEncoderBitrateMode::AACENC_BR_MODE_VBR_2:
[[fallthrough]];
case AacEncoderBitrateMode::AACENC_BR_MODE_VBR_3:
[[fallthrough]];
case AacEncoderBitrateMode::AACENC_BR_MODE_VBR_4:
[[fallthrough]];
case AacEncoderBitrateMode::AACENC_BR_MODE_VBR_5:
codec_config_.codec_specific_1 = codec_user_config_.codec_specific_1;
break;
default:
codec_config_.codec_specific_1 =
static_cast<int64_t>(AacEncoderBitrateMode::AACENC_BR_MODE_VBR_5);
}
} else {
// It is no needed to check the user preference when Variable Bitrate
// unsupported by one of source or sink
codec_config_.codec_specific_1 =
static_cast<int64_t>(AacEncoderBitrateMode::AACENC_BR_MODE_CBR);
}
}
if (codec_user_config_.codec_specific_2 != 0)
codec_config_.codec_specific_2 = codec_user_config_.codec_specific_2;
if (codec_user_config_.codec_specific_3 != 0)
codec_config_.codec_specific_3 = codec_user_config_.codec_specific_3;
if (codec_user_config_.codec_specific_4 != 0)
codec_config_.codec_specific_4 = codec_user_config_.codec_specific_4;
if (A2DP_BuildInfoAac(AVDT_MEDIA_TYPE_AUDIO, &result_config_cie,
p_result_codec_config) != A2DP_SUCCESS) {
goto fail;
}
// Create a local copy of the peer codec capability/config, and the
// result codec config.
if (is_capability) {
status = A2DP_BuildInfoAac(AVDT_MEDIA_TYPE_AUDIO, &peer_info_cie,
ota_codec_peer_capability_);
} else {
status = A2DP_BuildInfoAac(AVDT_MEDIA_TYPE_AUDIO, &peer_info_cie,
ota_codec_peer_config_);
}
CHECK(status == A2DP_SUCCESS);
status = A2DP_BuildInfoAac(AVDT_MEDIA_TYPE_AUDIO, &result_config_cie,
ota_codec_config_);
CHECK(status == A2DP_SUCCESS);
return true;
fail:
// Restore the internal state
codec_config_ = saved_codec_config;
codec_capability_ = saved_codec_capability;
codec_selectable_capability_ = saved_codec_selectable_capability;
codec_user_config_ = saved_codec_user_config;
codec_audio_config_ = saved_codec_audio_config;
memcpy(ota_codec_config_, saved_ota_codec_config, sizeof(ota_codec_config_));
memcpy(ota_codec_peer_capability_, saved_ota_codec_peer_capability,
sizeof(ota_codec_peer_capability_));
memcpy(ota_codec_peer_config_, saved_ota_codec_peer_config,
sizeof(ota_codec_peer_config_));
return false;
}
bool A2dpCodecConfigAacBase::setPeerCodecCapabilities(
const uint8_t* p_peer_codec_capabilities) {
std::lock_guard<std::recursive_mutex> lock(codec_mutex_);
tA2DP_AAC_CIE peer_info_cie;
uint8_t channelMode;
uint16_t sampleRate;
uint8_t variableBitRateSupport;
const tA2DP_AAC_CIE* p_a2dp_aac_caps =
(is_source_) ? &a2dp_aac_source_caps : &a2dp_aac_sink_caps;
// Save the internal state
btav_a2dp_codec_config_t saved_codec_selectable_capability =
codec_selectable_capability_;
uint8_t saved_ota_codec_peer_capability[AVDT_CODEC_SIZE];
memcpy(saved_ota_codec_peer_capability, ota_codec_peer_capability_,
sizeof(ota_codec_peer_capability_));
tA2DP_STATUS status =
A2DP_ParseInfoAac(&peer_info_cie, p_peer_codec_capabilities, true);
if (status != A2DP_SUCCESS) {
log::error("can't parse peer's capabilities: error = {}", status);
goto fail;
}
// Compute the selectable capability - sample rate
sampleRate = p_a2dp_aac_caps->sampleRate & peer_info_cie.sampleRate;
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_44100) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_44100;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_48000) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_48000;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_88200) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_88200;
}
if (sampleRate & A2DP_AAC_SAMPLING_FREQ_96000) {
codec_selectable_capability_.sample_rate |=
BTAV_A2DP_CODEC_SAMPLE_RATE_96000;
}
// Compute the selectable capability - bits per sample
codec_selectable_capability_.bits_per_sample =
p_a2dp_aac_caps->bits_per_sample;
// Compute the selectable capability - channel mode
channelMode = p_a2dp_aac_caps->channelMode & peer_info_cie.channelMode;
if (channelMode & A2DP_AAC_CHANNEL_MODE_MONO) {
codec_selectable_capability_.channel_mode |=
BTAV_A2DP_CODEC_CHANNEL_MODE_MONO;
}
if (channelMode & A2DP_AAC_CHANNEL_MODE_STEREO) {
codec_selectable_capability_.channel_mode |=
BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO;
}
// Compute the selectable capability - variable bitrate mode
variableBitRateSupport = p_a2dp_aac_caps->variableBitRateSupport &
peer_info_cie.variableBitRateSupport;
if (variableBitRateSupport != A2DP_AAC_VARIABLE_BIT_RATE_DISABLED) {
codec_selectable_capability_.codec_specific_1 =
static_cast<int64_t>(AacEncoderBitrateMode::AACENC_BR_MODE_VBR_5);
} else {
codec_selectable_capability_.codec_specific_1 =
static_cast<int64_t>(AacEncoderBitrateMode::AACENC_BR_MODE_CBR);
}
status = A2DP_BuildInfoAac(AVDT_MEDIA_TYPE_AUDIO, &peer_info_cie,
ota_codec_peer_capability_);
CHECK(status == A2DP_SUCCESS);
return true;
fail:
// Restore the internal state
codec_selectable_capability_ = saved_codec_selectable_capability;
memcpy(ota_codec_peer_capability_, saved_ota_codec_peer_capability,
sizeof(ota_codec_peer_capability_));
return false;
}
A2dpCodecConfigAacSink::A2dpCodecConfigAacSink(
btav_a2dp_codec_priority_t codec_priority)
: A2dpCodecConfigAacBase(BTAV_A2DP_CODEC_INDEX_SINK_AAC,
A2DP_CodecIndexStrAacSink(), codec_priority,
false) {}
A2dpCodecConfigAacSink::~A2dpCodecConfigAacSink() {}
bool A2dpCodecConfigAacSink::init() {
if (!isValid()) return false;
// Load the decoder
if (!A2DP_LoadDecoderAac()) {
log::error("cannot load the decoder");
return false;
}
return true;
}
bool A2dpCodecConfigAacSink::useRtpHeaderMarkerBit() const {
// TODO: This method applies only to Source codecs
return false;
}