ASoC: SDCA: Add a helper to get the SoundWire port number

Add a helper function to extract the SoundWire hardware port number
from the SDCA DataPort Selector Control. Typically this would be
called from hw_params() and used to call sdw_stream_add_slave().

Signed-off-by: Charles Keepax <ckeepax@opensource.cirrus.com>
Reviewed-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.dev>
Link: https://patch.msgid.link/20250707124155.2596744-7-ckeepax@opensource.cirrus.com
Signed-off-by: Mark Brown <broonie@kernel.org>
This commit is contained in:
Charles Keepax 2025-07-07 13:41:54 +01:00 committed by Mark Brown
parent 7b0d60dbb4
commit 264d3d776f
No known key found for this signature in database
GPG key ID: 24D68B725D5487D0
3 changed files with 86 additions and 0 deletions

View file

@ -48,5 +48,8 @@ int sdca_asoc_set_constraints(struct device *dev, struct regmap *regmap,
struct snd_soc_dai *dai);
void sdca_asoc_free_constraints(struct snd_pcm_substream *substream,
struct snd_soc_dai *dai);
int sdca_asoc_get_port(struct device *dev, struct regmap *regmap,
struct sdca_function_data *function,
struct snd_soc_dai *dai);
#endif // __SDCA_ASOC_H__

View file

@ -185,6 +185,14 @@ enum sdca_usage_range {
SDCA_USAGE_NCOLS = 7,
};
/**
* enum sdca_dataport_selector_range - Column definitions for DataPort_Selector
*/
enum sdca_dataport_selector_range {
SDCA_DATAPORT_SELECTOR_NCOLS = 16,
SDCA_DATAPORT_SELECTOR_NROWS = 4,
};
/**
* enum sdca_mu_controls - SDCA Controls for Mixer Unit
*

View file

@ -19,6 +19,7 @@
#include <linux/regmap.h>
#include <linux/soundwire/sdw_registers.h>
#include <linux/string_helpers.h>
#include <linux/types.h>
#include <sound/control.h>
#include <sound/pcm.h>
#include <sound/sdca.h>
@ -1368,3 +1369,77 @@ void sdca_asoc_free_constraints(struct snd_pcm_substream *substream,
kfree(constraint);
}
EXPORT_SYMBOL_NS(sdca_asoc_free_constraints, "SND_SOC_SDCA");
/**
* sdca_asoc_get_port - return SoundWire port for a DAI
* @dev: Pointer to the device, used for error messages.
* @regmap: Pointer to the Function register map.
* @function: Pointer to the Function information.
* @dai: Pointer to the ASoC DAI.
*
* Typically called from hw_params().
*
* Return: Returns a positive port number on success, and a negative error
* code on failure.
*/
int sdca_asoc_get_port(struct device *dev, struct regmap *regmap,
struct sdca_function_data *function,
struct snd_soc_dai *dai)
{
struct sdca_entity *entity = &function->entities[dai->id];
struct sdca_control_range *range;
unsigned int reg, val;
int sel = -EINVAL;
int i, ret;
switch (entity->type) {
case SDCA_ENTITY_TYPE_IT:
sel = SDCA_CTL_IT_DATAPORT_SELECTOR;
break;
case SDCA_ENTITY_TYPE_OT:
sel = SDCA_CTL_OT_DATAPORT_SELECTOR;
break;
default:
break;
}
if (sel < 0 || !entity->iot.is_dataport) {
dev_err(dev, "%s: port number only available for dataports\n",
entity->label);
return -EINVAL;
}
range = sdca_selector_find_range(dev, entity, sel, SDCA_DATAPORT_SELECTOR_NCOLS,
SDCA_DATAPORT_SELECTOR_NROWS);
if (!range)
return -EINVAL;
reg = SDW_SDCA_CTL(function->desc->adr, entity->id, sel, 0);
ret = regmap_read(regmap, reg, &val);
if (ret) {
dev_err(dev, "%s: failed to read dataport selector: %d\n",
entity->label, ret);
return ret;
}
for (i = 0; i < range->rows; i++) {
static const u8 port_mask = 0xF;
sel = sdca_range(range, val & port_mask, i);
/*
* FIXME: Currently only a single dataport is supported, so
* return the first one found, technically up to 4 dataports
* could be linked, but this is not yet supported.
*/
if (sel != 0xFF)
return sel;
val >>= hweight8(port_mask);
}
dev_err(dev, "%s: no dataport found\n", entity->label);
return -ENODEV;
}
EXPORT_SYMBOL_NS(sdca_asoc_get_port, "SND_SOC_SDCA");