[alsa-devel] [PATCH 2/2] ASoC: hdmi-codec: add channel mapping control

Arnaud Pouliquen arnaud.pouliquen at st.com
Mon Dec 12 10:38:45 CET 2016



On 12/11/2016 07:09 AM, Takashi Sakamoto wrote:
> On Dec 9 2016 01:37, Arnaud Pouliquen wrote:
>> Add user interface to provide channel mapping.
>> In a first step this control is read only.
>>
>> As TLV type, the control provides all configurations available for
>> HDMI sink(ELD), and provides current channel mapping selected by codec
>> based on ELD and number of channels specified by user on open.
>> When control is called before the number of the channel is specified
>> (i.e. hw_params is set), it returns all channels set to UNKNOWN.
>>
>> Notice that SNDRV_CTL_TLVT_CHMAP_FIXED is used for all mappings,
>> as no information is available from HDMI driver to allow channel swapping.
>>
>> Signed-off-by: Arnaud Pouliquen <arnaud.pouliquen at st.com>
>> ---
>>  sound/soc/codecs/hdmi-codec.c | 346 +++++++++++++++++++++++++++++++++++++++++-
>>  1 file changed, 345 insertions(+), 1 deletion(-)
>>
>> diff --git a/sound/soc/codecs/hdmi-codec.c b/sound/soc/codecs/hdmi-codec.c
>> index f27d115..0cb83a3 100644
>> --- a/sound/soc/codecs/hdmi-codec.c
>> +++ b/sound/soc/codecs/hdmi-codec.c
>> @@ -18,12 +18,137 @@
>>  #include <sound/pcm.h>
>>  #include <sound/pcm_params.h>
>>  #include <sound/soc.h>
>> +#include <sound/tlv.h>
>>  #include <sound/pcm_drm_eld.h>
>>  #include <sound/hdmi-codec.h>
>>  #include <sound/pcm_iec958.h>
>>
>>  #include <drm/drm_crtc.h> /* This is only to get MAX_ELD_BYTES */
>>
>> +#define HDMI_MAX_SPEAKERS  8
>> +
>> +/*
>> + * CEA speaker placement for HDMI 1.4:
>> + *
>> + *  FL  FLC   FC   FRC   FR   FRW
>> + *
>> + *                                  LFE
>> + *
>> + *  RL  RLC   RC   RRC   RR
>> + *
>> + *  Speaker placement has to be extended to support HDMI 2.0
>> + */
>> +enum hdmi_codec_cea_spk_placement {
>> +	FL  = (1 <<  0),	/* Front Left           */
>> +	FC  = (1 <<  1),	/* Front Center         */
>> +	FR  = (1 <<  2),	/* Front Right          */
>> +	FLC = (1 <<  3),	/* Front Left Center    */
>> +	FRC = (1 <<  4),	/* Front Right Center   */
>> +	RL  = (1 <<  5),	/* Rear Left            */
>> +	RC  = (1 <<  6),	/* Rear Center          */
>> +	RR  = (1 <<  7),	/* Rear Right           */
>> +	RLC = (1 <<  8),	/* Rear Left Center     */
>> +	RRC = (1 <<  9),	/* Rear Right Center    */
>> +	LFE = (1 << 10),	/* Low Frequency Effect */
>> +};
> 
> BIT() macro in "linux/bitops.h" is available.
will be corrected in a v2
> 
>> +
>> +/*
>> + * ELD Speaker allocation bits in the CEA Speaker Allocation data block
>> + */
>> +static int hdmi_codec_eld_spk_alloc_bits[] = {
>> +	[0] = FL | FR,
>> +	[1] = LFE,
>> +	[2] = FC,
>> +	[3] = RL | RR,
>> +	[4] = RC,
>> +	[5] = FLC | FRC,
>> +	[6] = RLC | RRC,
>> +};
> 
> Please put this kind of invariant table into .rodata section with 
> 'const' modifier.
will be corrected in a v2

> 
>> +
>> +struct hdmi_codec_channel_map_table {
>> +	unsigned char map;	/* ALSA API channel map position */
>> +	int spk_mask;		/* speaker position bit mask */
>> +};
>> +
>> +static struct hdmi_codec_channel_map_table hdmi_codec_map_table[] = {
>> +	{ SNDRV_CHMAP_FL,	FL },
>> +	{ SNDRV_CHMAP_FR,	FR },
>> +	{ SNDRV_CHMAP_RL,	RL },
>> +	{ SNDRV_CHMAP_RR,	RR },
>> +	{ SNDRV_CHMAP_LFE,	LFE },
>> +	{ SNDRV_CHMAP_FC,	FC },
>> +	{ SNDRV_CHMAP_RLC,	RLC },
>> +	{ SNDRV_CHMAP_RRC,	RRC },
>> +	{ SNDRV_CHMAP_RC,	RC },
>> +	{ SNDRV_CHMAP_FLC,	FLC },
>> +	{ SNDRV_CHMAP_FRC,	FRC },
>> +	{} /* terminator */
>> +};
> 
> In this case, the table can be put into snd_hdac_spk_to_chmap().
will be corrected in a v2
> 
>> +
>> +/*
>> + * cea Speaker allocation structure
>> + */
>> +struct hdmi_codec_cea_spk_alloc {
>> +	int ca_index;
>> +	int speakers[HDMI_MAX_SPEAKERS];
>> +
>> +	/* Derived values, computed during init */
>> +	int channels;
>> +	int spk_mask;
>> +	int spk_na_mask;
>> +};
>> +
>> +/*
>> + * This is an ordered list!
>> + *
>> + * The preceding ones have better chances to be selected by
>> + * hdmi_channel_allocation().
> 
> The function is not defined and implemented. It takes developers to be 
> puzzled.
will be corrected in a v2

> 
>> + */
>> +static struct hdmi_codec_cea_spk_alloc hdmi_codec_channel_alloc[] = {
>> +/*			  channel:   7     6    5    4    3     2    1    0  */
>> +{ .ca_index = 0x00,  .speakers = {   0,    0,   0,   0,   0,    0,  FR,  FL } },
>> +				 /* 2.1 */
>> +{ .ca_index = 0x01,  .speakers = {   0,    0,   0,   0,   0,  LFE,  FR,  FL } },
>> +				 /* Dolby Surround */
>> +{ .ca_index = 0x02,  .speakers = {   0,    0,   0,   0,  FC,    0,  FR,  FL } },
>> +				 /* surround51 */
>> +{ .ca_index = 0x0b,  .speakers = {   0,    0,  RR,  RL,  FC,  LFE,  FR,  FL } },
>> +				 /* surround40 */
>> +{ .ca_index = 0x08,  .speakers = {   0,    0,  RR,  RL,   0,    0,  FR,  FL } },
>> +				 /* surround41 */
>> +{ .ca_index = 0x09,  .speakers = {   0,    0,  RR,  RL,   0,  LFE,  FR,  FL } },
>> +				 /* surround50 */
>> +{ .ca_index = 0x0a,  .speakers = {   0,    0,  RR,  RL,  FC,    0,  FR,  FL } },
>> +				 /* 6.1 */
>> +{ .ca_index = 0x0f,  .speakers = {   0,   RC,  RR,  RL,  FC,  LFE,  FR,  FL } },
>> +				 /* surround71 */
>> +{ .ca_index = 0x13,  .speakers = { RRC,  RLC,  RR,  RL,  FC,  LFE,  FR,  FL } },
>> +
>> +{ .ca_index = 0x03,  .speakers = {   0,    0,   0,   0,  FC,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x04,  .speakers = {   0,    0,   0,  RC,   0,    0,  FR,  FL } },
>> +{ .ca_index = 0x05,  .speakers = {   0,    0,   0,  RC,   0,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x06,  .speakers = {   0,    0,   0,  RC,  FC,    0,  FR,  FL } },
>> +{ .ca_index = 0x07,  .speakers = {   0,    0,   0,  RC,  FC,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x0c,  .speakers = {   0,   RC,  RR,  RL,   0,    0,  FR,  FL } },
>> +{ .ca_index = 0x0d,  .speakers = {   0,   RC,  RR,  RL,   0,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x0e,  .speakers = {   0,   RC,  RR,  RL,  FC,    0,  FR,  FL } },
>> +{ .ca_index = 0x10,  .speakers = { RRC,  RLC,  RR,  RL,   0,    0,  FR,  FL } },
>> +{ .ca_index = 0x11,  .speakers = { RRC,  RLC,  RR,  RL,   0,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x12,  .speakers = { RRC,  RLC,  RR,  RL,  FC,    0,  FR,  FL } },
>> +{ .ca_index = 0x14,  .speakers = { FRC,  FLC,   0,   0,   0,    0,  FR,  FL } },
>> +{ .ca_index = 0x15,  .speakers = { FRC,  FLC,   0,   0,   0,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x16,  .speakers = { FRC,  FLC,   0,   0,  FC,    0,  FR,  FL } },
>> +{ .ca_index = 0x17,  .speakers = { FRC,  FLC,   0,   0,  FC,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x18,  .speakers = { FRC,  FLC,   0,  RC,   0,    0,  FR,  FL } },
>> +{ .ca_index = 0x19,  .speakers = { FRC,  FLC,   0,  RC,   0,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x1a,  .speakers = { FRC,  FLC,   0,  RC,  FC,    0,  FR,  FL } },
>> +{ .ca_index = 0x1b,  .speakers = { FRC,  FLC,   0,  RC,  FC,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x1c,  .speakers = { FRC,  FLC,  RR,  RL,   0,    0,  FR,  FL } },
>> +{ .ca_index = 0x1d,  .speakers = { FRC,  FLC,  RR,  RL,   0,  LFE,  FR,  FL } },
>> +{ .ca_index = 0x1e,  .speakers = { FRC,  FLC,  RR,  RL,  FC,    0,  FR,  FL } },
>> +{ .ca_index = 0x1f,  .speakers = { FRC,  FLC,  RR,  RL,  FC,  LFE,  FR,  FL } },
>> +};
> 
> Ditto.
Sorry not understand this comment vs the previous one, could you please
clarify?
> 
>> +
>>  struct hdmi_codec_priv {
>>  	struct hdmi_codec_pdata hcd;
>>  	struct snd_soc_dai_driver *daidrv;
>> @@ -32,6 +157,7 @@ struct hdmi_codec_priv {
>>  	struct snd_pcm_substream *current_stream;
>>  	struct snd_pcm_hw_constraint_list ratec;
>>  	uint8_t eld[MAX_ELD_BYTES];
>> +	unsigned int chmap[HDMI_MAX_SPEAKERS];
>>  };
>>
>>  static const struct snd_soc_dapm_widget hdmi_widgets[] = {
>> @@ -70,6 +196,201 @@ static int hdmi_eld_ctl_get(struct snd_kcontrol *kcontrol,
>>  	return 0;
>>  }
>>
>> +static int hdmi_codec_chmap_ctl_info(struct snd_kcontrol *kcontrol,
>> +				     struct snd_ctl_elem_info *uinfo)
>> +{
>> +	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
>> +	uinfo->count = HDMI_MAX_SPEAKERS;
>> +	uinfo->value.integer.min = 0;
>> +	uinfo->value.integer.max = SNDRV_CHMAP_LAST;
>> +
>> +	return 0;
>> +}
>> +
>> +static int hdmi_codec_spk_mask_from_alloc(int spk_alloc)
>> +{
>> +	int i;
>> +	int spk_mask = hdmi_codec_eld_spk_alloc_bits[0];
>> +
>> +	for (i = 0; i < ARRAY_SIZE(hdmi_codec_eld_spk_alloc_bits); i++) {
>> +		if (spk_alloc & (1 << i))
>> +			spk_mask |= hdmi_codec_eld_spk_alloc_bits[i];
>> +	}
>> +
>> +	return spk_mask;
>> +}
>> +
>> +static int hdmi_codec_chmap_ctl_get(struct snd_kcontrol *kcontrol,
>> +				    struct snd_ctl_elem_value *ucontrol)
>> +{
>> +	struct snd_soc_component *component = snd_kcontrol_chip(kcontrol);
>> +	struct hdmi_codec_priv *hcp = snd_soc_component_get_drvdata(component);
>> +	int i;
>> +
>> +	memset(ucontrol->value.integer.value, 0,
>> +	       sizeof(ucontrol->value.integer.value));
>> +
>> +	mutex_lock(&hcp->current_stream_lock);
>> +	if (hcp->current_stream)
>> +		for (i = 0; i < HDMI_MAX_SPEAKERS; i++)
>> +			ucontrol->value.integer.value[i] = hcp->chmap[i];
>> +
>> +	mutex_unlock(&hcp->current_stream_lock);
>> +
>> +	return 0;
>> +}
>> +
>> +/* From speaker bit mask to ALSA API channel position */
>> +static int snd_hdac_spk_to_chmap(int spk)
>> +{
>> +	struct hdmi_codec_channel_map_table *t = hdmi_codec_map_table;
>> +
>> +	for (; t->map; t++) {
>> +		if (t->spk_mask == spk)
>> +			return t->map;
>> +	}
>> +
>> +	return 0;
>> +}
> 
> Why hdac? Are there some relationship between HDA controller and table 
> you added?
No relation ship just a stupid guy who does a copy/past from hda without
renaming...
> 
>> +/**
>> + * hdmi_codec_cea_init_channel_alloc:
>> + * Compute derived values in hdmi_codec_channel_alloc[].
>> + * spk_na_mask is used to store unused channels in mid of the channel
>> + * allocations. These particular channels are then considered as active channels
>> + * For instance:
>> + *    CA_ID 0x02: CA =  (FL, FR, 0, FC) => spk_na_mask = 0x04, channels = 4
>> + *    CA_ID 0x04: CA =  (FL, FR, 0, 0, RC) => spk_na_mask = 0x03C, channels = 5
>> + */
>> +static void hdmi_codec_cea_init_channel_alloc(void)
>> +{
>> +	int i, j, k, last;
>> +	struct hdmi_codec_cea_spk_alloc *p;
>> +
>> +	for (i = 0; i < ARRAY_SIZE(hdmi_codec_channel_alloc); i++) {
>> +		p = hdmi_codec_channel_alloc + i;
>> +		p->spk_mask = 0;
>> +		p->spk_na_mask = 0;
>> +		last = HDMI_MAX_SPEAKERS;
>> +		for (j = 0, k = 7; j < HDMI_MAX_SPEAKERS; j++, k--) {
>> +			if (p->speakers[j]) {
>> +				p->spk_mask |= p->speakers[j];
>> +				if (last == HDMI_MAX_SPEAKERS)
>> +					last = j;
>> +			} else if (last != HDMI_MAX_SPEAKERS) {
>> +				p->spk_na_mask |= 1 << k;
>> +			}
>> +		}
>> +		p->channels = 8 - last;
>> +	}
>> +}
>> +
>> +static int hdmi_codec_get_ch_alloc_table_idx(struct hdmi_codec_priv *hcp,
>> +					     unsigned char channels)
>> +{
>> +	int i, spk_alloc, spk_mask;
>> +	struct hdmi_codec_cea_spk_alloc *cap = hdmi_codec_channel_alloc;
>> +
>> +	spk_alloc = drm_eld_get_spk_alloc(hcp->eld);
>> +	spk_mask = hdmi_codec_spk_mask_from_alloc(spk_alloc);
>> +
>> +	for (i = 0; i < ARRAY_SIZE(hdmi_codec_channel_alloc); i++, cap++) {
>> +		if (cap->channels != channels)
>> +			continue;
>> +		if (!(cap->spk_mask == (spk_mask & cap->spk_mask)))
>> +			continue;
>> +		return i;
>> +	}
>> +
>> +	return -EINVAL;
>> +}
>> +
>> +static void hdmi_cea_alloc_to_tlv_chmap(struct hdmi_codec_cea_spk_alloc *cap,
>> +					unsigned int *chmap)
>> +{
>> +	int count = 0;
>> +	int c, spk;
>> +
>> +	/* Detect unused channels in cea caps, tag them as N/A channel in TLV */
>> +	for (c = 0; c < HDMI_MAX_SPEAKERS; c++) {
>> +		spk = cap->speakers[7 - c];
>> +		if (cap->spk_na_mask & BIT(c))
>> +			chmap[count++] = SNDRV_CHMAP_NA;
>> +		else
>> +			chmap[count++] = snd_hdac_spk_to_chmap(spk);
>> +	}
>> +}
>> +
>> +static int hdmi_codec_chmap_ctl_tlv(struct snd_kcontrol *kcontrol, int op_flag,
>> +				    unsigned int size, unsigned int __user *tlv)
>> +{
>> +	struct snd_soc_component *component = snd_kcontrol_chip(kcontrol);
>> +	struct hdmi_codec_priv *hcp = snd_soc_component_get_drvdata(component);
>> +	unsigned int __user *dst;
>> +	int chs, count = 0;
>> +	int num_ca = ARRAY_SIZE(hdmi_codec_channel_alloc);
>> +	unsigned long max_chs;
>> +	int spk_alloc, spk_mask;
>> +
>> +	if (size < 8)
>> +		return -ENOMEM;
>> +
>> +	if (put_user(SNDRV_CTL_TLVT_CONTAINER, tlv))
>> +		return -EFAULT;
>> +	size -= 8;
>> +	dst = tlv + 2;
>> +
>> +	spk_alloc = drm_eld_get_spk_alloc(hcp->eld);
>> +	spk_mask = hdmi_codec_spk_mask_from_alloc(spk_alloc);
>> +
>> +	max_chs = hweight_long(spk_mask);
>> +
>> +	for (chs = 2; chs <= max_chs; chs++) {
>> +		int i;
>> +		struct hdmi_codec_cea_spk_alloc *cap;
>> +
>> +		cap = hdmi_codec_channel_alloc;
>> +		for (i = 0; i < num_ca; i++, cap++) {
>> +			int chs_bytes = chs * 4;
>> +			unsigned int tlv_chmap[HDMI_MAX_SPEAKERS];
>> +
>> +			if (cap->channels != chs)
>> +				continue;
>> +
>> +			if (!(cap->spk_mask == (spk_mask & cap->spk_mask)))
>> +				continue;
>> +
Seems missing check here, related question below, in answer to your comment
			if (size < 8)
				return -ENOMEM;
>> +			/*
>> +			 * Channel mapping is fixed as hdmi codec capability
>> +			 * is not know.
>> +			 */
>> +			if (put_user(SNDRV_CTL_TLVT_CHMAP_FIXED, dst) ||
>> +			    put_user(chs_bytes, dst + 1))
>> +				return -EFAULT;
>> +
>> +			dst += 2;
>> +			size -= 8;
>> +			count += 8;
>> +
>> +			if (size < chs_bytes)
>> +				return -ENOMEM;
>> +
>> +			size -= chs_bytes;
>> +			count += chs_bytes;
>> +			hdmi_cea_alloc_to_tlv_chmap(cap, tlv_chmap);
>> +
>> +			if (copy_to_user(dst, tlv_chmap, chs_bytes))
>> +				return -EFAULT;
>> +			dst += chs;
>> +		}
>> +	}
>> +
>> +	if (put_user(count, tlv + 1))
>> +		return -EFAULT;
>> +
>> +	return 0;
>> +}
>> +
> 
> This function has a bug to cause buffer-over-run in user space because 
> applications can request with a small buffer.
"size" is used for this, the bug is that a check is missing (the one i
added in comment above), or i missed something else?
> 
>>  static const struct snd_kcontrol_new hdmi_controls[] = {
>>  	{
>>  		.access = SNDRV_CTL_ELEM_ACCESS_READ |
>> @@ -79,6 +400,17 @@ static const struct snd_kcontrol_new hdmi_controls[] = {
>>  		.info = hdmi_eld_ctl_info,
>>  		.get = hdmi_eld_ctl_get,
>>  	},
>> +	{
>> +		.access = SNDRV_CTL_ELEM_ACCESS_READ |
>> +			  SNDRV_CTL_ELEM_ACCESS_TLV_READ |
>> +			  SNDRV_CTL_ELEM_ACCESS_TLV_CALLBACK |
>> +			  SNDRV_CTL_ELEM_ACCESS_VOLATILE,
>> +		.iface = SNDRV_CTL_ELEM_IFACE_PCM,
>> +		.name = "Playback Channel Map",
>> +		.info = hdmi_codec_chmap_ctl_info,
>> +		.get = hdmi_codec_chmap_ctl_get,
>> +		.tlv.c = hdmi_codec_chmap_ctl_tlv,
>> +	},
>>  };
> 
> If you can keep the same interface for applications as 
> 'snd_pcm_add_chmap_ctls()' have, it's better to integrate the function 
> to have different tables/callbacks depending on drivers.
> 
I had a look before define it. Unfortunately i cannot use
snd_pcm_add_chmap_ctls. snd_pcm_add_chmap_ctls requests "snd_pcm"
structure as input param. ASoC codec not aware of it.
i could add an helper for ASoC, but hdmi-codec should be used for HDMI
drivers and i'm not sure that there is another need in ASoC.

Thanks and Regards

Arnaud


More information about the Alsa-devel mailing list