hal: Add support for Backend port specific channel map

Currently, channel map mixer control is generic for all
playback backends. Add support for backend device specific
mixer control to avoid overwrite of channel map configuration.
Also, always set proper channel map/allocation for playback
for HDMI and display port.

Change-Id: I41e3db6e07be869d82bdd3712744a5b78a707228
This commit is contained in:
Rohit kumar 2019-02-04 16:26:33 +05:30
parent 054513d029
commit 4dbd2c04eb
3 changed files with 54 additions and 23 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2013-2018, The Linux Foundation. All rights reserved. * Copyright (c) 2013-2019, The Linux Foundation. All rights reserved.
* Not a Contribution. * Not a Contribution.
* *
* Copyright (C) 2013 The Android Open Source Project * Copyright (C) 2013 The Android Open Source Project
@ -6068,7 +6068,7 @@ static int platform_set_codec_backend_cfg(struct audio_device* adev,
my_data->current_backend_cfg[backend_idx].channels = channels; my_data->current_backend_cfg[backend_idx].channels = channels;
if (backend_idx == HDMI_RX_BACKEND) if (backend_idx == HDMI_RX_BACKEND)
platform_set_edid_channels_configuration(adev->platform, channels, HDMI_RX_BACKEND); platform_set_edid_channels_configuration(adev->platform, channels, HDMI_RX_BACKEND, snd_device);
ALOGD("%s:becf: afe: %s set to %s", __func__, ALOGD("%s:becf: afe: %s set to %s", __func__,
my_data->current_backend_cfg[backend_idx].channels_mixer_ctl, channel_cnt_str); my_data->current_backend_cfg[backend_idx].channels_mixer_ctl, channel_cnt_str);
@ -6446,6 +6446,8 @@ static bool platform_check_codec_backend_cfg(struct audio_device* adev,
if (channels != my_data->current_backend_cfg[backend_idx].channels) if (channels != my_data->current_backend_cfg[backend_idx].channels)
channels_updated = true; channels_updated = true;
platform_set_edid_channels_configuration(adev->platform, channels, backend_idx, snd_device);
} }
//check if mulitchannel clip needs to be down sampled to 48k //check if mulitchannel clip needs to be down sampled to 48k
@ -6563,7 +6565,7 @@ bool platform_check_and_set_codec_backend_cfg(struct audio_device* adev,
if ((my_data->spkr_ch_map != NULL) && if ((my_data->spkr_ch_map != NULL) &&
(platform_get_backend_index(snd_device) == DEFAULT_CODEC_BACKEND)) (platform_get_backend_index(snd_device) == DEFAULT_CODEC_BACKEND))
platform_set_channel_map(my_data, my_data->spkr_ch_map->num_ch, platform_set_channel_map(my_data, my_data->spkr_ch_map->num_ch,
my_data->spkr_ch_map->chmap, -1); my_data->spkr_ch_map->chmap, -1, -1);
if (platform_split_snd_device(adev->platform, snd_device, if (platform_split_snd_device(adev->platform, snd_device,
&num_devices, new_snd_devices) < 0) &num_devices, new_snd_devices) < 0)
@ -7113,7 +7115,7 @@ int platform_set_stream_channel_map(void *platform, audio_channel_mask_t channel
return -1; return -1;
} }
} }
ret = platform_set_channel_map(platform, channels, channel_map, snd_id); ret = platform_set_channel_map(platform, channels, channel_map, snd_id, -1);
return ret; return ret;
} }
@ -7253,7 +7255,7 @@ int platform_set_channel_allocation(void *platform, int channel_alloc)
return ret; return ret;
} }
int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int snd_id) int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int snd_id, int be_idx __unused)
{ {
struct mixer_ctl *ctl; struct mixer_ctl *ctl;
char mixer_ctl_name[44] = {0}; // max length of name is 44 as defined char mixer_ctl_name[44] = {0}; // max length of name is 44 as defined
@ -7415,7 +7417,7 @@ bool platform_is_edid_supported_sample_rate(void *platform, int sample_rate)
return false; return false;
} }
int platform_set_edid_channels_configuration(void *platform, int channels, int backend_idx __unused) { int platform_set_edid_channels_configuration(void *platform, int channels, int backend_idx __unused, snd_device_t snd_device __unused) {
struct platform_data *my_data = (struct platform_data *)platform; struct platform_data *my_data = (struct platform_data *)platform;
struct audio_device *adev = my_data->adev; struct audio_device *adev = my_data->adev;
@ -7448,9 +7450,9 @@ int platform_set_edid_channels_configuration(void *platform, int channels, int b
*/ */
if (adev_device_cfg_ptr->use_client_dev_cfg) { if (adev_device_cfg_ptr->use_client_dev_cfg) {
platform_set_channel_map(platform, adev_device_cfg_ptr->dev_cfg_params.channels, platform_set_channel_map(platform, adev_device_cfg_ptr->dev_cfg_params.channels,
(char *)adev_device_cfg_ptr->dev_cfg_params.channel_map, -1); (char *)adev_device_cfg_ptr->dev_cfg_params.channel_map, -1, -1);
} else { } else {
platform_set_channel_map(platform, channel_count, info->channel_map, -1); platform_set_channel_map(platform, channel_count, info->channel_map, -1, -1);
} }
if (adev_device_cfg_ptr->use_client_dev_cfg) { if (adev_device_cfg_ptr->use_client_dev_cfg) {
@ -7469,7 +7471,7 @@ int platform_set_edid_channels_configuration(void *platform, int channels, int b
default_channelMap[0] = PCM_CHANNEL_FL; default_channelMap[0] = PCM_CHANNEL_FL;
default_channelMap[1] = PCM_CHANNEL_FR; default_channelMap[1] = PCM_CHANNEL_FR;
} }
platform_set_channel_map(platform,2,default_channelMap,-1); platform_set_channel_map(platform, 2, default_channelMap, -1, -1);
platform_set_channel_allocation(platform,0); platform_set_channel_allocation(platform,0);
} }
} }

View File

@ -7036,7 +7036,7 @@ static int platform_set_codec_backend_cfg(struct audio_device* adev,
if ((backend_idx == HDMI_RX_BACKEND) || if ((backend_idx == HDMI_RX_BACKEND) ||
(backend_idx == DISP_PORT_RX_BACKEND)) (backend_idx == DISP_PORT_RX_BACKEND))
platform_set_edid_channels_configuration(adev->platform, channels, backend_idx); platform_set_edid_channels_configuration(adev->platform, channels, backend_idx, snd_device);
ALOGD("%s:becf: afe: %s set to %s ", __func__, ALOGD("%s:becf: afe: %s set to %s ", __func__,
my_data->current_backend_cfg[backend_idx].channels_mixer_ctl, my_data->current_backend_cfg[backend_idx].channels_mixer_ctl,
@ -7460,6 +7460,8 @@ static bool platform_check_codec_backend_cfg(struct audio_device* adev,
if (channels != my_data->current_backend_cfg[backend_idx].channels) if (channels != my_data->current_backend_cfg[backend_idx].channels)
channels_updated = true; channels_updated = true;
platform_set_edid_channels_configuration(adev->platform, channels, backend_idx, snd_device);
} }
ALOGI("%s:becf: afe: Codec selected backend: %d updated bit width: %d and sample rate: %d", ALOGI("%s:becf: afe: Codec selected backend: %d updated bit width: %d and sample rate: %d",
@ -7509,11 +7511,13 @@ bool platform_check_and_set_codec_backend_cfg(struct audio_device* adev,
int backend_idx = DEFAULT_CODEC_BACKEND; int backend_idx = DEFAULT_CODEC_BACKEND;
int new_snd_devices[SND_DEVICE_OUT_END] = {0}; int new_snd_devices[SND_DEVICE_OUT_END] = {0};
int i, num_devices = 1; int i, num_devices = 1;
int device_be_idx = -1;
bool ret = false; bool ret = false;
struct platform_data *my_data = (struct platform_data *)adev->platform; struct platform_data *my_data = (struct platform_data *)adev->platform;
struct audio_backend_cfg backend_cfg; struct audio_backend_cfg backend_cfg;
backend_idx = platform_get_backend_index(snd_device); backend_idx = platform_get_backend_index(snd_device);
device_be_idx = platform_get_snd_device_backend_index(snd_device);
if (usecase->type == TRANSCODE_LOOPBACK_RX) { if (usecase->type == TRANSCODE_LOOPBACK_RX) {
backend_cfg.bit_width = usecase->stream.inout->out_config.bit_width; backend_cfg.bit_width = usecase->stream.inout->out_config.bit_width;
@ -7553,7 +7557,7 @@ bool platform_check_and_set_codec_backend_cfg(struct audio_device* adev,
if ((my_data->spkr_ch_map != NULL) && if ((my_data->spkr_ch_map != NULL) &&
(platform_get_backend_index(snd_device) == DEFAULT_CODEC_BACKEND)) (platform_get_backend_index(snd_device) == DEFAULT_CODEC_BACKEND))
platform_set_channel_map(my_data, my_data->spkr_ch_map->num_ch, platform_set_channel_map(my_data, my_data->spkr_ch_map->num_ch,
my_data->spkr_ch_map->chmap, -1); my_data->spkr_ch_map->chmap, -1, device_be_idx);
if (platform_split_snd_device(my_data, snd_device, &num_devices, if (platform_split_snd_device(my_data, snd_device, &num_devices,
new_snd_devices) < 0) new_snd_devices) < 0)
@ -8144,7 +8148,7 @@ int platform_set_stream_channel_map(void *platform, audio_channel_mask_t channel
return -1; return -1;
} }
} }
ret = platform_set_channel_map(platform, channels, channel_map, snd_id); ret = platform_set_channel_map(platform, channels, channel_map, snd_id, -1);
return ret; return ret;
} }
@ -8258,13 +8262,16 @@ int platform_set_channel_allocation(void *platform, int channel_alloc)
return ret; return ret;
} }
int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int snd_id) int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int snd_id, int be_idx)
{ {
struct mixer_ctl *ctl; struct mixer_ctl *ctl, *be_ctl = NULL;
char mixer_ctl_name[44] = {0}; // max length of name is 44 as defined char mixer_ctl_name[44] = {0}; // max length of name is 44 as defined
char be_mixer_ctl_name[] = "Backend Device Channel Map";
int ret; int ret;
int i=0, n=0; int i=0, n=0;
int be_id_count = 0;
long set_values[AUDIO_MAX_DSP_CHANNELS]; long set_values[AUDIO_MAX_DSP_CHANNELS];
long be_set_values[AUDIO_MAX_DSP_CHANNELS + 1] = {0};
struct platform_data *my_data = (struct platform_data *)platform; struct platform_data *my_data = (struct platform_data *)platform;
struct audio_device *adev = my_data->adev; struct audio_device *adev = my_data->adev;
ALOGV("%s channel_count:%d",__func__, ch_count); ALOGV("%s channel_count:%d",__func__, ch_count);
@ -8289,7 +8296,20 @@ int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int snd
if (snd_id >= 0) { if (snd_id >= 0) {
snprintf(mixer_ctl_name, sizeof(mixer_ctl_name), "Playback Channel Map%d", snd_id); snprintf(mixer_ctl_name, sizeof(mixer_ctl_name), "Playback Channel Map%d", snd_id);
} else { } else {
strlcpy(mixer_ctl_name, "Playback Device Channel Map", sizeof(mixer_ctl_name)); if (be_idx >= 0) {
be_ctl = mixer_get_ctl_by_name(adev->mixer, be_mixer_ctl_name);
if (!be_ctl) {
ALOGD("%s: Could not get ctl for mixer cmd - %s, using default control",
__func__, be_mixer_ctl_name);
strlcpy(mixer_ctl_name, "Playback Device Channel Map", sizeof(mixer_ctl_name));
be_idx = -1;
} else {
strlcpy(mixer_ctl_name, "Backend Device Channel Map", sizeof(mixer_ctl_name));
be_id_count = 1;
}
} else {
strlcpy(mixer_ctl_name, "Playback Device Channel Map", sizeof(mixer_ctl_name));
}
} }
ALOGD("%s mixer_ctl_name:%s", __func__, mixer_ctl_name); ALOGD("%s mixer_ctl_name:%s", __func__, mixer_ctl_name);
@ -8313,7 +8333,7 @@ int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int snd
return -EINVAL; return -EINVAL;
} }
if (n > AUDIO_MAX_DSP_CHANNELS) { if (n > (AUDIO_MAX_DSP_CHANNELS + be_id_count)) {
ALOGE("%s mixerctl elem size %d > AUDIO_MAX_DSP_CHANNELS %d",__func__, n, AUDIO_MAX_DSP_CHANNELS); ALOGE("%s mixerctl elem size %d > AUDIO_MAX_DSP_CHANNELS %d",__func__, n, AUDIO_MAX_DSP_CHANNELS);
return -EINVAL; return -EINVAL;
} }
@ -8330,7 +8350,13 @@ int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int snd
set_values[0], set_values[1], set_values[2], set_values[3], set_values[4], set_values[0], set_values[1], set_values[2], set_values[3], set_values[4],
set_values[5], set_values[6], set_values[7], ch_count); set_values[5], set_values[6], set_values[7], ch_count);
ret = mixer_ctl_set_array(ctl, set_values, n); if (be_idx >= 0) {
be_set_values[0] = be_idx;
memcpy(&be_set_values[1], set_values, sizeof(long) * ch_count);
ret = mixer_ctl_set_array(ctl, be_set_values, ARRAY_SIZE(be_set_values));
} else {
ret = mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
}
if (ret < 0) { if (ret < 0) {
ALOGE("%s: Could not set ctl, error:%d ch_count:%d", ALOGE("%s: Could not set ctl, error:%d ch_count:%d",
@ -8536,7 +8562,7 @@ bool platform_spkr_use_default_sample_rate(void *platform) {
return my_data->use_sprk_default_sample_rate; return my_data->use_sprk_default_sample_rate;
} }
int platform_set_edid_channels_configuration(void *platform, int channels, int backend_idx) { int platform_set_edid_channels_configuration(void *platform, int channels, int backend_idx, snd_device_t snd_device) {
struct platform_data *my_data = (struct platform_data *)platform; struct platform_data *my_data = (struct platform_data *)platform;
struct audio_device *adev = my_data->adev; struct audio_device *adev = my_data->adev;
@ -8545,6 +8571,7 @@ int platform_set_edid_channels_configuration(void *platform, int channels, int b
int i, ret; int i, ret;
char default_channelMap[MAX_CHANNELS_SUPPORTED] = {0}; char default_channelMap[MAX_CHANNELS_SUPPORTED] = {0};
struct audio_device_config_param *adev_device_cfg_ptr = adev->device_cfg_params; struct audio_device_config_param *adev_device_cfg_ptr = adev->device_cfg_params;
int be_idx = -1;
if ((backend_idx != HDMI_RX_BACKEND) && if ((backend_idx != HDMI_RX_BACKEND) &&
(backend_idx != DISP_PORT_RX_BACKEND)) { (backend_idx != DISP_PORT_RX_BACKEND)) {
@ -8552,6 +8579,7 @@ int platform_set_edid_channels_configuration(void *platform, int channels, int b
return -EINVAL; return -EINVAL;
} }
be_idx = platform_get_snd_device_backend_index(snd_device);
ret = platform_get_edid_info(platform); ret = platform_get_edid_info(platform);
info = (edid_audio_info *)my_data->edid_info; info = (edid_audio_info *)my_data->edid_info;
adev_device_cfg_ptr += backend_idx; adev_device_cfg_ptr += backend_idx;
@ -8574,9 +8602,9 @@ int platform_set_edid_channels_configuration(void *platform, int channels, int b
*/ */
if (adev_device_cfg_ptr->use_client_dev_cfg) { if (adev_device_cfg_ptr->use_client_dev_cfg) {
platform_set_channel_map(platform, adev_device_cfg_ptr->dev_cfg_params.channels, platform_set_channel_map(platform, adev_device_cfg_ptr->dev_cfg_params.channels,
(char *)adev_device_cfg_ptr->dev_cfg_params.channel_map, -1); (char *)adev_device_cfg_ptr->dev_cfg_params.channel_map, -1, be_idx);
} else { } else {
platform_set_channel_map(platform, channel_count, info->channel_map, -1); platform_set_channel_map(platform, channel_count, info->channel_map, -1, be_idx);
} }
if (adev_device_cfg_ptr->use_client_dev_cfg) { if (adev_device_cfg_ptr->use_client_dev_cfg) {
@ -8595,7 +8623,7 @@ int platform_set_edid_channels_configuration(void *platform, int channels, int b
default_channelMap[0] = PCM_CHANNEL_FL; default_channelMap[0] = PCM_CHANNEL_FL;
default_channelMap[1] = PCM_CHANNEL_FR; default_channelMap[1] = PCM_CHANNEL_FR;
} }
platform_set_channel_map(platform,2,default_channelMap,-1); platform_set_channel_map(platform, 2, default_channelMap, -1, be_idx);
platform_set_channel_allocation(platform,0); platform_set_channel_allocation(platform,0);
} }
} }

View File

@ -230,7 +230,7 @@ int platform_set_channel_allocation(void *platform, int channel_alloc);
int platform_get_edid_info(void *platform); int platform_get_edid_info(void *platform);
int platform_get_supported_copp_sampling_rate(uint32_t stream_sr); int platform_get_supported_copp_sampling_rate(uint32_t stream_sr);
int platform_set_channel_map(void *platform, int ch_count, char *ch_map, int platform_set_channel_map(void *platform, int ch_count, char *ch_map,
int snd_id); int snd_id, int be_idx);
int platform_set_stream_channel_map(void *platform, audio_channel_mask_t channel_mask, int platform_set_stream_channel_map(void *platform, audio_channel_mask_t channel_mask,
int snd_id, uint8_t *input_channel_map); int snd_id, uint8_t *input_channel_map);
int platform_set_stream_pan_scale_params(void *platform, int platform_set_stream_pan_scale_params(void *platform,
@ -240,7 +240,8 @@ int platform_set_stream_downmix_params(void *platform,
int snd_id, int snd_id,
snd_device_t snd_device, snd_device_t snd_device,
struct mix_matrix_params mm_params); struct mix_matrix_params mm_params);
int platform_set_edid_channels_configuration(void *platform, int channels, int backend_idx); int platform_set_edid_channels_configuration(void *platform, int channels,
int backend_idx, snd_device_t snd_device);
bool platform_spkr_use_default_sample_rate(void *platform); bool platform_spkr_use_default_sample_rate(void *platform);
unsigned char platform_map_to_edid_format(int format); unsigned char platform_map_to_edid_format(int format);
bool platform_is_edid_supported_format(void *platform, int format); bool platform_is_edid_supported_format(void *platform, int format);