public inbox for linux-kernel@vger.kernel.org
 help / color / mirror / Atom feed
* [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier
@ 2025-04-07 15:16 Stefan Binding
  2025-04-07 15:16 ` [PATCH v1 1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config Stefan Binding
                   ` (5 more replies)
  0 siblings, 6 replies; 18+ messages in thread
From: Stefan Binding @ 2025-04-07 15:16 UTC (permalink / raw)
  To: Mark Brown; +Cc: linux-sound, linux-kernel, patches, Stefan Binding

CS35L63 is a Mono Class-D PC Smart Amplifier, with Speaker Protection
and Audio Enhancement Algorithms.

CS35L63 uses a similar control interface to CS35L56 so support for
it can be added into the CS35L56 driver.
CS35L63 only has SoundWire and I2C control interfaces.

Stefan Binding (5):
  ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap
    config
  ASoC: cs35l56: Add struct to index firmware registers
  ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware
    register list
  ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire
  ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63

 include/sound/cs35l56.h           |  28 ++++
 sound/pci/hda/cs35l56_hda.c       |  22 ++-
 sound/pci/hda/cs35l56_hda_i2c.c   |   3 +
 sound/pci/hda/cs35l56_hda_spi.c   |   3 +
 sound/soc/codecs/cs35l56-i2c.c    |  23 ++-
 sound/soc/codecs/cs35l56-sdw.c    |  91 +++++++++++-
 sound/soc/codecs/cs35l56-shared.c | 229 ++++++++++++++++++++++++++++--
 sound/soc/codecs/cs35l56-spi.c    |   3 +
 sound/soc/codecs/cs35l56.c        |  47 +++++-
 sound/soc/codecs/cs35l56.h        |   1 +
 10 files changed, 413 insertions(+), 37 deletions(-)

-- 
2.43.0


^ permalink raw reply	[flat|nested] 18+ messages in thread

* [PATCH v1 1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config
  2025-04-07 15:16 [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Stefan Binding
@ 2025-04-07 15:16 ` Stefan Binding
  2025-04-28 17:32   ` Richard Fitzgerald
  2025-04-07 15:16 ` [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers Stefan Binding
                   ` (4 subsequent siblings)
  5 siblings, 1 reply; 18+ messages in thread
From: Stefan Binding @ 2025-04-07 15:16 UTC (permalink / raw)
  To: Mark Brown; +Cc: linux-sound, linux-kernel, patches, Stefan Binding

This is to prepare for further products using slightly different
regmap configs.

Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
---
 sound/soc/codecs/cs35l56-i2c.c | 16 +++++++++++++---
 sound/soc/codecs/cs35l56-sdw.c | 16 +++++++++++++---
 2 files changed, 26 insertions(+), 6 deletions(-)

diff --git a/sound/soc/codecs/cs35l56-i2c.c b/sound/soc/codecs/cs35l56-i2c.c
index 8a518df1e16e..5962914e2180 100644
--- a/sound/soc/codecs/cs35l56-i2c.c
+++ b/sound/soc/codecs/cs35l56-i2c.c
@@ -17,9 +17,10 @@
 
 static int cs35l56_i2c_probe(struct i2c_client *client)
 {
+	unsigned int id = (u32)(uintptr_t)i2c_get_match_data(client);
 	struct cs35l56_private *cs35l56;
 	struct device *dev = &client->dev;
-	const struct regmap_config *regmap_config = &cs35l56_regmap_i2c;
+	const struct regmap_config *regmap_config;
 	int ret;
 
 	cs35l56 = devm_kzalloc(dev, sizeof(struct cs35l56_private), GFP_KERNEL);
@@ -30,6 +31,15 @@ static int cs35l56_i2c_probe(struct i2c_client *client)
 	cs35l56->base.can_hibernate = true;
 
 	i2c_set_clientdata(client, cs35l56);
+
+	switch (id) {
+	case 0x3556:
+		regmap_config = &cs35l56_regmap_i2c;
+		break;
+	default:
+		return -ENODEV;
+	}
+
 	cs35l56->base.regmap = devm_regmap_init_i2c(client, regmap_config);
 	if (IS_ERR(cs35l56->base.regmap)) {
 		ret = PTR_ERR(cs35l56->base.regmap);
@@ -57,14 +67,14 @@ static void cs35l56_i2c_remove(struct i2c_client *client)
 }
 
 static const struct i2c_device_id cs35l56_id_i2c[] = {
-	{ "cs35l56" },
+	{ "cs35l56", 0x3556 },
 	{}
 };
 MODULE_DEVICE_TABLE(i2c, cs35l56_id_i2c);
 
 #ifdef CONFIG_ACPI
 static const struct acpi_device_id cs35l56_asoc_acpi_match[] = {
-	{ "CSC355C", 0 },
+	{ "CSC355C", 0x3556 },
 	{},
 };
 MODULE_DEVICE_TABLE(acpi, cs35l56_asoc_acpi_match);
diff --git a/sound/soc/codecs/cs35l56-sdw.c b/sound/soc/codecs/cs35l56-sdw.c
index 3f91cb3f9ae7..d178357e1196 100644
--- a/sound/soc/codecs/cs35l56-sdw.c
+++ b/sound/soc/codecs/cs35l56-sdw.c
@@ -509,6 +509,7 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
 {
 	struct device *dev = &peripheral->dev;
 	struct cs35l56_private *cs35l56;
+	const struct regmap_config *regmap_config;
 	int ret;
 
 	cs35l56 = devm_kzalloc(dev, sizeof(*cs35l56), GFP_KERNEL);
@@ -521,8 +522,17 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
 
 	dev_set_drvdata(dev, cs35l56);
 
+	switch ((unsigned int)id->driver_data) {
+	case 0x3556:
+	case 0x3557:
+		regmap_config = &cs35l56_regmap_sdw;
+		break;
+	default:
+		return -ENODEV;
+	}
+
 	cs35l56->base.regmap = devm_regmap_init(dev, &cs35l56_regmap_bus_sdw,
-					   peripheral, &cs35l56_regmap_sdw);
+					   peripheral, regmap_config);
 	if (IS_ERR(cs35l56->base.regmap)) {
 		ret = PTR_ERR(cs35l56->base.regmap);
 		return dev_err_probe(dev, ret, "Failed to allocate register map\n");
@@ -562,8 +572,8 @@ static const struct dev_pm_ops cs35l56_sdw_pm = {
 };
 
 static const struct sdw_device_id cs35l56_sdw_id[] = {
-	SDW_SLAVE_ENTRY(0x01FA, 0x3556, 0),
-	SDW_SLAVE_ENTRY(0x01FA, 0x3557, 0),
+	SDW_SLAVE_ENTRY(0x01FA, 0x3556, 0x3556),
+	SDW_SLAVE_ENTRY(0x01FA, 0x3557, 0x3557),
 	{},
 };
 MODULE_DEVICE_TABLE(sdw, cs35l56_sdw_id);
-- 
2.43.0


^ permalink raw reply related	[flat|nested] 18+ messages in thread

* [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-07 15:16 [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Stefan Binding
  2025-04-07 15:16 ` [PATCH v1 1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config Stefan Binding
@ 2025-04-07 15:16 ` Stefan Binding
  2025-04-07 19:16   ` Mark Brown
  2025-04-28 17:34   ` Richard Fitzgerald
  2025-04-07 15:16 ` [PATCH v1 3/5] ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware register list Stefan Binding
                   ` (3 subsequent siblings)
  5 siblings, 2 replies; 18+ messages in thread
From: Stefan Binding @ 2025-04-07 15:16 UTC (permalink / raw)
  To: Mark Brown; +Cc: linux-sound, linux-kernel, patches, Stefan Binding

Firmware based registers may be different addresses across different
device ids and revision ids. Create a structure to store and access
these addresses.

Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
---
 include/sound/cs35l56.h           | 11 ++++++++
 sound/pci/hda/cs35l56_hda.c       |  5 ++--
 sound/pci/hda/cs35l56_hda_i2c.c   |  3 +++
 sound/pci/hda/cs35l56_hda_spi.c   |  3 +++
 sound/soc/codecs/cs35l56-i2c.c    |  1 +
 sound/soc/codecs/cs35l56-sdw.c    |  1 +
 sound/soc/codecs/cs35l56-shared.c | 42 ++++++++++++++++++++++++-------
 sound/soc/codecs/cs35l56-spi.c    |  3 +++
 sound/soc/codecs/cs35l56.c        |  5 ++--
 9 files changed, 61 insertions(+), 13 deletions(-)

diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
index 5d653a3491d0..d712cb79652b 100644
--- a/include/sound/cs35l56.h
+++ b/include/sound/cs35l56.h
@@ -267,6 +267,14 @@ struct cs35l56_spi_payload {
 } __packed;
 static_assert(sizeof(struct cs35l56_spi_payload) == 10);
 
+struct cs35l56_fw_reg {
+	unsigned int fw_ver;
+	unsigned int halo_state;
+	unsigned int pm_cur_stat;
+	unsigned int prot_sts;
+	unsigned int transducer_actual_ps;
+};
+
 struct cs35l56_base {
 	struct device *dev;
 	struct regmap *regmap;
@@ -283,6 +291,7 @@ struct cs35l56_base {
 	struct cirrus_amp_cal_data cal_data;
 	struct gpio_desc *reset_gpio;
 	struct cs35l56_spi_payload *spi_payload_buf;
+	const struct cs35l56_fw_reg *fw_reg;
 };
 
 static inline bool cs35l56_is_otp_register(unsigned int reg)
@@ -311,6 +320,8 @@ extern const struct regmap_config cs35l56_regmap_i2c;
 extern const struct regmap_config cs35l56_regmap_spi;
 extern const struct regmap_config cs35l56_regmap_sdw;
 
+extern const struct cs35l56_fw_reg cs35l56_fw_reg;
+
 extern const struct cirrus_amp_cal_controls cs35l56_calibration_controls;
 
 extern const char * const cs35l56_tx_input_texts[CS35L56_NUM_INPUT_SRC];
diff --git a/sound/pci/hda/cs35l56_hda.c b/sound/pci/hda/cs35l56_hda.c
index 4ef7878e8fd4..f6257fe45b9f 100644
--- a/sound/pci/hda/cs35l56_hda.c
+++ b/sound/pci/hda/cs35l56_hda.c
@@ -68,7 +68,7 @@ static void cs35l56_hda_play(struct cs35l56_hda *cs35l56)
 	if (ret == 0) {
 		/* Wait for firmware to enter PS0 power state */
 		ret = regmap_read_poll_timeout(cs35l56->base.regmap,
-					       CS35L56_TRANSDUCER_ACTUAL_PS,
+					       cs35l56->base.fw_reg->transducer_actual_ps,
 					       val, (val == CS35L56_PS0),
 					       CS35L56_PS0_POLL_US,
 					       CS35L56_PS0_TIMEOUT_US);
@@ -667,7 +667,8 @@ static void cs35l56_hda_fw_load(struct cs35l56_hda *cs35l56)
 
 	regcache_sync(cs35l56->base.regmap);
 
-	regmap_clear_bits(cs35l56->base.regmap, CS35L56_PROTECTION_STATUS,
+	regmap_clear_bits(cs35l56->base.regmap,
+			  cs35l56->base.fw_reg->prot_sts,
 			  CS35L56_FIRMWARE_MISSING);
 	cs35l56->base.fw_patched = true;
 
diff --git a/sound/pci/hda/cs35l56_hda_i2c.c b/sound/pci/hda/cs35l56_hda_i2c.c
index c7b836613149..d10209e4eddd 100644
--- a/sound/pci/hda/cs35l56_hda_i2c.c
+++ b/sound/pci/hda/cs35l56_hda_i2c.c
@@ -26,6 +26,9 @@ static int cs35l56_hda_i2c_probe(struct i2c_client *clt)
 #ifdef CS35L56_WAKE_HOLD_TIME_US
 	cs35l56->base.can_hibernate = true;
 #endif
+
+	cs35l56->base.fw_reg = &cs35l56_fw_reg;
+
 	cs35l56->base.regmap = devm_regmap_init_i2c(clt, &cs35l56_regmap_i2c);
 	if (IS_ERR(cs35l56->base.regmap)) {
 		ret = PTR_ERR(cs35l56->base.regmap);
diff --git a/sound/pci/hda/cs35l56_hda_spi.c b/sound/pci/hda/cs35l56_hda_spi.c
index 903578466905..f57533d3d728 100644
--- a/sound/pci/hda/cs35l56_hda_spi.c
+++ b/sound/pci/hda/cs35l56_hda_spi.c
@@ -29,6 +29,9 @@ static int cs35l56_hda_spi_probe(struct spi_device *spi)
 #ifdef CS35L56_WAKE_HOLD_TIME_US
 	cs35l56->base.can_hibernate = true;
 #endif
+
+	cs35l56->base.fw_reg = &cs35l56_fw_reg;
+
 	cs35l56->base.regmap = devm_regmap_init_spi(spi, &cs35l56_regmap_spi);
 	if (IS_ERR(cs35l56->base.regmap)) {
 		ret = PTR_ERR(cs35l56->base.regmap);
diff --git a/sound/soc/codecs/cs35l56-i2c.c b/sound/soc/codecs/cs35l56-i2c.c
index 5962914e2180..38c391d11c78 100644
--- a/sound/soc/codecs/cs35l56-i2c.c
+++ b/sound/soc/codecs/cs35l56-i2c.c
@@ -35,6 +35,7 @@ static int cs35l56_i2c_probe(struct i2c_client *client)
 	switch (id) {
 	case 0x3556:
 		regmap_config = &cs35l56_regmap_i2c;
+		cs35l56->base.fw_reg = &cs35l56_fw_reg;
 		break;
 	default:
 		return -ENODEV;
diff --git a/sound/soc/codecs/cs35l56-sdw.c b/sound/soc/codecs/cs35l56-sdw.c
index d178357e1196..2e0422b41385 100644
--- a/sound/soc/codecs/cs35l56-sdw.c
+++ b/sound/soc/codecs/cs35l56-sdw.c
@@ -526,6 +526,7 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
 	case 0x3556:
 	case 0x3557:
 		regmap_config = &cs35l56_regmap_sdw;
+		cs35l56->base.fw_reg = &cs35l56_fw_reg;
 		break;
 	default:
 		return -ENODEV;
diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
index e28bfefa72f3..bc8f9379bc74 100644
--- a/sound/soc/codecs/cs35l56-shared.c
+++ b/sound/soc/codecs/cs35l56-shared.c
@@ -253,7 +253,8 @@ int cs35l56_firmware_shutdown(struct cs35l56_base *cs35l56_base)
 	if (ret)
 		return ret;
 
-	ret = regmap_read_poll_timeout(cs35l56_base->regmap,  CS35L56_DSP1_PM_CUR_STATE,
+	ret = regmap_read_poll_timeout(cs35l56_base->regmap,
+				       cs35l56_base->fw_reg->pm_cur_stat,
 				       val, (val == CS35L56_HALO_STATE_SHUTDOWN),
 				       CS35L56_HALO_STATE_POLL_US,
 				       CS35L56_HALO_STATE_TIMEOUT_US);
@@ -278,7 +279,9 @@ int cs35l56_wait_for_firmware_boot(struct cs35l56_base *cs35l56_base)
 				     CS35L56_HALO_STATE_POLL_US,
 				     CS35L56_HALO_STATE_TIMEOUT_US,
 				     false,
-				     cs35l56_base->regmap, CS35L56_DSP1_HALO_STATE, &val);
+				     cs35l56_base->regmap,
+				     cs35l56_base->fw_reg->halo_state,
+				     &val);
 
 	if (poll_ret) {
 		dev_err(cs35l56_base->dev, "Firmware boot timed out(%d): HALO_STATE=%#x\n",
@@ -395,9 +398,17 @@ void cs35l56_system_reset(struct cs35l56_base *cs35l56_base, bool is_soundwire)
 		return;
 	}
 
-	regmap_multi_reg_write_bypassed(cs35l56_base->regmap,
-					cs35l56_system_reset_seq,
-					ARRAY_SIZE(cs35l56_system_reset_seq));
+	switch (cs35l56_base->type) {
+	case 0x54:
+	case 0x56:
+	case 0x57:
+		regmap_multi_reg_write_bypassed(cs35l56_base->regmap,
+						cs35l56_system_reset_seq,
+						ARRAY_SIZE(cs35l56_system_reset_seq));
+		break;
+	default:
+		break;
+	}
 
 	/* On SoundWire the registers won't be accessible until it re-enumerates. */
 	if (is_soundwire)
@@ -514,7 +525,9 @@ int cs35l56_is_fw_reload_needed(struct cs35l56_base *cs35l56_base)
 		return ret;
 	}
 
-	ret = regmap_read(cs35l56_base->regmap, CS35L56_PROTECTION_STATUS, &val);
+	ret = regmap_read(cs35l56_base->regmap,
+			  cs35l56_base->fw_reg->prot_sts,
+			  &val);
 	if (ret)
 		dev_err(cs35l56_base->dev, "Failed to read PROTECTION_STATUS: %d\n", ret);
 	else
@@ -562,7 +575,7 @@ int cs35l56_runtime_suspend_common(struct cs35l56_base *cs35l56_base)
 
 	/* Firmware must have entered a power-save state */
 	ret = regmap_read_poll_timeout(cs35l56_base->regmap,
-				       CS35L56_TRANSDUCER_ACTUAL_PS,
+				       cs35l56_base->fw_reg->transducer_actual_ps,
 				       val, (val >= CS35L56_PS3),
 				       CS35L56_PS3_POLL_US,
 				       CS35L56_PS3_TIMEOUT_US);
@@ -752,7 +765,8 @@ int cs35l56_read_prot_status(struct cs35l56_base *cs35l56_base,
 	unsigned int prot_status;
 	int ret;
 
-	ret = regmap_read(cs35l56_base->regmap, CS35L56_PROTECTION_STATUS, &prot_status);
+	ret = regmap_read(cs35l56_base->regmap,
+			  cs35l56_base->fw_reg->prot_sts, &prot_status);
 	if (ret) {
 		dev_err(cs35l56_base->dev, "Get PROTECTION_STATUS failed: %d\n", ret);
 		return ret;
@@ -760,7 +774,8 @@ int cs35l56_read_prot_status(struct cs35l56_base *cs35l56_base,
 
 	*fw_missing = !!(prot_status & CS35L56_FIRMWARE_MISSING);
 
-	ret = regmap_read(cs35l56_base->regmap, CS35L56_DSP1_FW_VER, fw_version);
+	ret = regmap_read(cs35l56_base->regmap,
+			  cs35l56_base->fw_reg->fw_ver, fw_version);
 	if (ret) {
 		dev_err(cs35l56_base->dev, "Get FW VER failed: %d\n", ret);
 		return ret;
@@ -1045,6 +1060,15 @@ const struct regmap_config cs35l56_regmap_sdw = {
 };
 EXPORT_SYMBOL_NS_GPL(cs35l56_regmap_sdw, "SND_SOC_CS35L56_SHARED");
 
+const struct cs35l56_fw_reg cs35l56_fw_reg = {
+	.fw_ver = CS35L56_DSP1_FW_VER,
+	.halo_state = CS35L56_DSP1_HALO_STATE,
+	.pm_cur_stat = CS35L56_DSP1_PM_CUR_STATE,
+	.prot_sts = CS35L56_PROTECTION_STATUS,
+	.transducer_actual_ps = CS35L56_TRANSDUCER_ACTUAL_PS,
+};
+EXPORT_SYMBOL_NS_GPL(cs35l56_fw_reg, "SND_SOC_CS35L56_SHARED");
+
 MODULE_DESCRIPTION("ASoC CS35L56 Shared");
 MODULE_AUTHOR("Richard Fitzgerald <rf@opensource.cirrus.com>");
 MODULE_AUTHOR("Simon Trimmer <simont@opensource.cirrus.com>");
diff --git a/sound/soc/codecs/cs35l56-spi.c b/sound/soc/codecs/cs35l56-spi.c
index ca6c03a8766d..c2ddee22cd23 100644
--- a/sound/soc/codecs/cs35l56-spi.c
+++ b/sound/soc/codecs/cs35l56-spi.c
@@ -25,6 +25,9 @@ static int cs35l56_spi_probe(struct spi_device *spi)
 		return -ENOMEM;
 
 	spi_set_drvdata(spi, cs35l56);
+
+	cs35l56->base.fw_reg = &cs35l56_fw_reg;
+
 	cs35l56->base.regmap = devm_regmap_init_spi(spi, regmap_config);
 	if (IS_ERR(cs35l56->base.regmap)) {
 		ret = PTR_ERR(cs35l56->base.regmap);
diff --git a/sound/soc/codecs/cs35l56.c b/sound/soc/codecs/cs35l56.c
index b3158a84b87a..c1d8bfb803b9 100644
--- a/sound/soc/codecs/cs35l56.c
+++ b/sound/soc/codecs/cs35l56.c
@@ -174,7 +174,7 @@ static int cs35l56_play_event(struct snd_soc_dapm_widget *w,
 	case SND_SOC_DAPM_POST_PMU:
 		/* Wait for firmware to enter PS0 power state */
 		ret = regmap_read_poll_timeout(cs35l56->base.regmap,
-					       CS35L56_TRANSDUCER_ACTUAL_PS,
+					       cs35l56->base.fw_reg->transducer_actual_ps,
 					       val, (val == CS35L56_PS0),
 					       CS35L56_PS0_POLL_US,
 					       CS35L56_PS0_TIMEOUT_US);
@@ -760,7 +760,8 @@ static void cs35l56_patch(struct cs35l56_private *cs35l56, bool firmware_missing
 		goto err_unlock;
 	}
 
-	regmap_clear_bits(cs35l56->base.regmap, CS35L56_PROTECTION_STATUS,
+	regmap_clear_bits(cs35l56->base.regmap,
+			  cs35l56->base.fw_reg->prot_sts,
 			  CS35L56_FIRMWARE_MISSING);
 	cs35l56->base.fw_patched = true;
 
-- 
2.43.0


^ permalink raw reply related	[flat|nested] 18+ messages in thread

* [PATCH v1 3/5] ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware register list
  2025-04-07 15:16 [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Stefan Binding
  2025-04-07 15:16 ` [PATCH v1 1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config Stefan Binding
  2025-04-07 15:16 ` [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers Stefan Binding
@ 2025-04-07 15:16 ` Stefan Binding
  2025-04-28 17:36   ` Richard Fitzgerald
  2025-04-07 15:16 ` [PATCH v1 4/5] ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire Stefan Binding
                   ` (2 subsequent siblings)
  5 siblings, 1 reply; 18+ messages in thread
From: Stefan Binding @ 2025-04-07 15:16 UTC (permalink / raw)
  To: Mark Brown; +Cc: linux-sound, linux-kernel, patches, Stefan Binding

Registers to set Mute, Volume and Posture are inside firmware,
which means they should be added to the list of registers set inside
firmware, in case they vary across Device or Revision.

These three registers are also used for controls, so additional
handling is required to be able to obtain and set the register inside
ALSA controls.

Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
---
 include/sound/cs35l56.h           |  3 +++
 sound/pci/hda/cs35l56_hda.c       | 17 +++++++----------
 sound/soc/codecs/cs35l56-shared.c | 25 ++++++++++++++++++++++++-
 sound/soc/codecs/cs35l56.c        | 19 +++++++++++++++++--
 4 files changed, 51 insertions(+), 13 deletions(-)

diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
index d712cb79652b..3abe4fbd2085 100644
--- a/include/sound/cs35l56.h
+++ b/include/sound/cs35l56.h
@@ -273,6 +273,9 @@ struct cs35l56_fw_reg {
 	unsigned int pm_cur_stat;
 	unsigned int prot_sts;
 	unsigned int transducer_actual_ps;
+	unsigned int user_mute;
+	unsigned int user_volume;
+	unsigned int posture_number;
 };
 
 struct cs35l56_base {
diff --git a/sound/pci/hda/cs35l56_hda.c b/sound/pci/hda/cs35l56_hda.c
index f6257fe45b9f..61aba1956338 100644
--- a/sound/pci/hda/cs35l56_hda.c
+++ b/sound/pci/hda/cs35l56_hda.c
@@ -237,7 +237,8 @@ static int cs35l56_hda_posture_get(struct snd_kcontrol *kcontrol,
 
 	cs35l56_hda_wait_dsp_ready(cs35l56);
 
-	ret = regmap_read(cs35l56->base.regmap, CS35L56_MAIN_POSTURE_NUMBER, &pos);
+	ret = regmap_read(cs35l56->base.regmap,
+			  cs35l56->base.fw_reg->posture_number, &pos);
 	if (ret)
 		return ret;
 
@@ -260,10 +261,8 @@ static int cs35l56_hda_posture_put(struct snd_kcontrol *kcontrol,
 
 	cs35l56_hda_wait_dsp_ready(cs35l56);
 
-	ret = regmap_update_bits_check(cs35l56->base.regmap,
-				       CS35L56_MAIN_POSTURE_NUMBER,
-				       CS35L56_MAIN_POSTURE_MASK,
-				       pos, &changed);
+	ret = regmap_update_bits_check(cs35l56->base.regmap, cs35l56->base.fw_reg->posture_number,
+				       CS35L56_MAIN_POSTURE_MASK, pos, &changed);
 	if (ret)
 		return ret;
 
@@ -305,7 +304,7 @@ static int cs35l56_hda_vol_get(struct snd_kcontrol *kcontrol,
 
 	cs35l56_hda_wait_dsp_ready(cs35l56);
 
-	ret = regmap_read(cs35l56->base.regmap, CS35L56_MAIN_RENDER_USER_VOLUME, &raw_vol);
+	ret = regmap_read(cs35l56->base.regmap, cs35l56->base.fw_reg->user_volume, &raw_vol);
 
 	if (ret)
 		return ret;
@@ -339,10 +338,8 @@ static int cs35l56_hda_vol_put(struct snd_kcontrol *kcontrol,
 
 	cs35l56_hda_wait_dsp_ready(cs35l56);
 
-	ret = regmap_update_bits_check(cs35l56->base.regmap,
-				       CS35L56_MAIN_RENDER_USER_VOLUME,
-				       CS35L56_MAIN_RENDER_USER_VOLUME_MASK,
-				       raw_vol, &changed);
+	ret = regmap_update_bits_check(cs35l56->base.regmap, cs35l56->base.fw_reg->user_volume,
+				       CS35L56_MAIN_RENDER_USER_VOLUME_MASK, raw_vol, &changed);
 	if (ret)
 		return ret;
 
diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
index bc8f9379bc74..da982774bb4d 100644
--- a/sound/soc/codecs/cs35l56-shared.c
+++ b/sound/soc/codecs/cs35l56-shared.c
@@ -38,7 +38,9 @@ static const struct reg_sequence cs35l56_patch[] = {
 	{ CS35L56_SWIRE_DP3_CH3_INPUT,		0x00000029 },
 	{ CS35L56_SWIRE_DP3_CH4_INPUT,		0x00000028 },
 	{ CS35L56_IRQ1_MASK_18,			0x1f7df0ff },
+};
 
+static const struct reg_sequence cs35l56_patch_fw[] = {
 	/* These are not reset by a soft-reset, so patch to defaults. */
 	{ CS35L56_MAIN_RENDER_USER_MUTE,	0x00000000 },
 	{ CS35L56_MAIN_RENDER_USER_VOLUME,	0x00000000 },
@@ -47,8 +49,26 @@ static const struct reg_sequence cs35l56_patch[] = {
 
 int cs35l56_set_patch(struct cs35l56_base *cs35l56_base)
 {
-	return regmap_register_patch(cs35l56_base->regmap, cs35l56_patch,
+	int ret;
+
+	ret = regmap_register_patch(cs35l56_base->regmap, cs35l56_patch,
 				     ARRAY_SIZE(cs35l56_patch));
+	if (ret)
+		return ret;
+
+
+	switch (cs35l56_base->type) {
+	case 0x54:
+	case 0x56:
+	case 0x57:
+		ret = regmap_register_patch(cs35l56_base->regmap, cs35l56_patch_fw,
+					    ARRAY_SIZE(cs35l56_patch_fw));
+		break;
+	default:
+		break;
+	}
+
+	return ret;
 }
 EXPORT_SYMBOL_NS_GPL(cs35l56_set_patch, "SND_SOC_CS35L56_SHARED");
 
@@ -1066,6 +1086,9 @@ const struct cs35l56_fw_reg cs35l56_fw_reg = {
 	.pm_cur_stat = CS35L56_DSP1_PM_CUR_STATE,
 	.prot_sts = CS35L56_PROTECTION_STATUS,
 	.transducer_actual_ps = CS35L56_TRANSDUCER_ACTUAL_PS,
+	.user_mute = CS35L56_MAIN_RENDER_USER_MUTE,
+	.user_volume = CS35L56_MAIN_RENDER_USER_VOLUME,
+	.posture_number = CS35L56_MAIN_POSTURE_NUMBER,
 };
 EXPORT_SYMBOL_NS_GPL(cs35l56_fw_reg, "SND_SOC_CS35L56_SHARED");
 
diff --git a/sound/soc/codecs/cs35l56.c b/sound/soc/codecs/cs35l56.c
index c1d8bfb803b9..a4a1d09097fc 100644
--- a/sound/soc/codecs/cs35l56.c
+++ b/sound/soc/codecs/cs35l56.c
@@ -838,6 +838,7 @@ static int cs35l56_component_probe(struct snd_soc_component *component)
 	struct cs35l56_private *cs35l56 = snd_soc_component_get_drvdata(component);
 	struct dentry *debugfs_root = component->debugfs_root;
 	unsigned short vendor, device;
+	int ret;
 
 	BUILD_BUG_ON(ARRAY_SIZE(cs35l56_tx_input_texts) != ARRAY_SIZE(cs35l56_tx_input_values));
 
@@ -877,6 +878,22 @@ static int cs35l56_component_probe(struct snd_soc_component *component)
 	debugfs_create_bool("can_hibernate", 0444, debugfs_root, &cs35l56->base.can_hibernate);
 	debugfs_create_bool("fw_patched", 0444, debugfs_root, &cs35l56->base.fw_patched);
 
+
+	switch (cs35l56->base.type) {
+	case 0x54:
+	case 0x56:
+	case 0x57:
+		ret = snd_soc_add_component_controls(component, cs35l56_controls,
+						     ARRAY_SIZE(cs35l56_controls));
+		break;
+	default:
+		ret = -ENODEV;
+		break;
+	}
+
+	if (ret)
+		return dev_err_probe(cs35l56->base.dev, ret, "unable to add controls\n");
+
 	queue_work(cs35l56->dsp_wq, &cs35l56->dsp_work);
 
 	return 0;
@@ -932,8 +949,6 @@ static const struct snd_soc_component_driver soc_component_dev_cs35l56 = {
 	.num_dapm_widgets = ARRAY_SIZE(cs35l56_dapm_widgets),
 	.dapm_routes = cs35l56_audio_map,
 	.num_dapm_routes = ARRAY_SIZE(cs35l56_audio_map),
-	.controls = cs35l56_controls,
-	.num_controls = ARRAY_SIZE(cs35l56_controls),
 
 	.set_bias_level = cs35l56_set_bias_level,
 
-- 
2.43.0


^ permalink raw reply related	[flat|nested] 18+ messages in thread

* [PATCH v1 4/5] ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire
  2025-04-07 15:16 [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Stefan Binding
                   ` (2 preceding siblings ...)
  2025-04-07 15:16 ` [PATCH v1 3/5] ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware register list Stefan Binding
@ 2025-04-07 15:16 ` Stefan Binding
  2025-04-28 17:38   ` Richard Fitzgerald
  2025-04-07 15:16 ` [PATCH v1 5/5] ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63 Stefan Binding
  2025-05-07 11:12 ` [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Mark Brown
  5 siblings, 1 reply; 18+ messages in thread
From: Stefan Binding @ 2025-04-07 15:16 UTC (permalink / raw)
  To: Mark Brown; +Cc: linux-sound, linux-kernel, patches, Stefan Binding

CS35L63 uses a similar control interface to CS35L56 so support for
it can be added into the CS35L56 driver.
New regmap configs have been added to support CS35L63.
CS35L63 only has SoundWire and I2C control interfaces.

Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
---
 include/sound/cs35l56.h           |  12 +++
 sound/soc/codecs/cs35l56-i2c.c    |   6 ++
 sound/soc/codecs/cs35l56-sdw.c    |  74 ++++++++++++++++++
 sound/soc/codecs/cs35l56-shared.c | 123 +++++++++++++++++++++++++++++-
 sound/soc/codecs/cs35l56.c        |  23 ++++++
 sound/soc/codecs/cs35l56.h        |   1 +
 6 files changed, 237 insertions(+), 2 deletions(-)

diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
index 3abe4fbd2085..e16e1a94c8a1 100644
--- a/include/sound/cs35l56.h
+++ b/include/sound/cs35l56.h
@@ -104,6 +104,15 @@
 #define CS35L56_DSP1_PMEM_0				0x3800000
 #define CS35L56_DSP1_PMEM_5114				0x3804FE8
 
+#define CS35L63_DSP1_FW_VER				CS35L56_DSP1_FW_VER
+#define CS35L63_DSP1_HALO_STATE				0x280396C
+#define CS35L63_DSP1_PM_CUR_STATE			0x28042C8
+#define CS35L63_PROTECTION_STATUS			0x340009C
+#define CS35L63_TRANSDUCER_ACTUAL_PS			0x34000F4
+#define CS35L63_MAIN_RENDER_USER_MUTE			0x3400020
+#define CS35L63_MAIN_RENDER_USER_VOLUME			0x3400028
+#define CS35L63_MAIN_POSTURE_NUMBER			0x3400068
+
 /* DEVID */
 #define CS35L56_DEVID_MASK				0x00FFFFFF
 
@@ -322,8 +331,11 @@ static inline bool cs35l56_is_spi(struct cs35l56_base *cs35l56)
 extern const struct regmap_config cs35l56_regmap_i2c;
 extern const struct regmap_config cs35l56_regmap_spi;
 extern const struct regmap_config cs35l56_regmap_sdw;
+extern const struct regmap_config cs35l63_regmap_i2c;
+extern const struct regmap_config cs35l63_regmap_sdw;
 
 extern const struct cs35l56_fw_reg cs35l56_fw_reg;
+extern const struct cs35l56_fw_reg cs35l63_fw_reg;
 
 extern const struct cirrus_amp_cal_controls cs35l56_calibration_controls;
 
diff --git a/sound/soc/codecs/cs35l56-i2c.c b/sound/soc/codecs/cs35l56-i2c.c
index 38c391d11c78..073f1796ae29 100644
--- a/sound/soc/codecs/cs35l56-i2c.c
+++ b/sound/soc/codecs/cs35l56-i2c.c
@@ -37,6 +37,10 @@ static int cs35l56_i2c_probe(struct i2c_client *client)
 		regmap_config = &cs35l56_regmap_i2c;
 		cs35l56->base.fw_reg = &cs35l56_fw_reg;
 		break;
+	case 0x3563:
+		regmap_config = &cs35l63_regmap_i2c;
+		cs35l56->base.fw_reg = &cs35l63_fw_reg;
+		break;
 	default:
 		return -ENODEV;
 	}
@@ -69,6 +73,7 @@ static void cs35l56_i2c_remove(struct i2c_client *client)
 
 static const struct i2c_device_id cs35l56_id_i2c[] = {
 	{ "cs35l56", 0x3556 },
+	{ "cs35l63", 0x3563 },
 	{}
 };
 MODULE_DEVICE_TABLE(i2c, cs35l56_id_i2c);
@@ -76,6 +81,7 @@ MODULE_DEVICE_TABLE(i2c, cs35l56_id_i2c);
 #ifdef CONFIG_ACPI
 static const struct acpi_device_id cs35l56_asoc_acpi_match[] = {
 	{ "CSC355C", 0x3556 },
+	{ "CSC356C", 0x3563 },
 	{},
 };
 MODULE_DEVICE_TABLE(acpi, cs35l56_asoc_acpi_match);
diff --git a/sound/soc/codecs/cs35l56-sdw.c b/sound/soc/codecs/cs35l56-sdw.c
index 2e0422b41385..13f602f51bf3 100644
--- a/sound/soc/codecs/cs35l56-sdw.c
+++ b/sound/soc/codecs/cs35l56-sdw.c
@@ -393,6 +393,74 @@ static int cs35l56_sdw_update_status(struct sdw_slave *peripheral,
 	return 0;
 }
 
+static int cs35l63_sdw_kick_divider(struct cs35l56_private *cs35l56,
+				    struct sdw_slave *peripheral)
+{
+	unsigned int curr_scale_reg, next_scale_reg;
+	int curr_scale, next_scale, ret;
+
+	if (!cs35l56->base.init_done)
+		return 0;
+
+	if (peripheral->bus->params.curr_bank) {
+		curr_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B1;
+		next_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B0;
+	} else {
+		curr_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B0;
+		next_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B1;
+	}
+
+	/*
+	 * Current clock scale value must be different to new value.
+	 * Modify current to guarantee this. If next still has the dummy
+	 * value we wrote when it was current, the core code has not set
+	 * a new scale so restore its original good value
+	 */
+	curr_scale = sdw_read_no_pm(peripheral, curr_scale_reg);
+	if (curr_scale < 0) {
+		dev_err(cs35l56->base.dev, "Failed to read current clock scale: %d\n", curr_scale);
+		return curr_scale;
+	}
+
+	next_scale = sdw_read_no_pm(peripheral, next_scale_reg);
+	if (next_scale < 0) {
+		dev_err(cs35l56->base.dev, "Failed to read next clock scale: %d\n", next_scale);
+		return next_scale;
+	}
+
+	if (next_scale == CS35L56_SDW_INVALID_BUS_SCALE) {
+		next_scale = cs35l56->old_sdw_clock_scale;
+		ret = sdw_write_no_pm(peripheral, next_scale_reg, next_scale);
+		if (ret < 0) {
+			dev_err(cs35l56->base.dev, "Failed to modify current clock scale: %d\n",
+				ret);
+			return ret;
+		}
+	}
+
+	cs35l56->old_sdw_clock_scale = curr_scale;
+	ret = sdw_write_no_pm(peripheral, curr_scale_reg, CS35L56_SDW_INVALID_BUS_SCALE);
+	if (ret < 0) {
+		dev_err(cs35l56->base.dev, "Failed to modify current clock scale: %d\n", ret);
+		return ret;
+	}
+
+	dev_dbg(cs35l56->base.dev, "Next bus scale: %#x\n", next_scale);
+
+	return 0;
+}
+
+static int cs35l56_sdw_bus_config(struct sdw_slave *peripheral,
+				  struct sdw_bus_params *params)
+{
+	struct cs35l56_private *cs35l56 = dev_get_drvdata(&peripheral->dev);
+
+	if ((cs35l56->base.type == 0x63) && (cs35l56->base.rev < 0xa1))
+		return cs35l63_sdw_kick_divider(cs35l56, peripheral);
+
+	return 0;
+}
+
 static int __maybe_unused cs35l56_sdw_clk_stop(struct sdw_slave *peripheral,
 					       enum sdw_clk_stop_mode mode,
 					       enum sdw_clk_stop_type type)
@@ -408,6 +476,7 @@ static const struct sdw_slave_ops cs35l56_sdw_ops = {
 	.read_prop = cs35l56_sdw_read_prop,
 	.interrupt_callback = cs35l56_sdw_interrupt,
 	.update_status = cs35l56_sdw_update_status,
+	.bus_config = cs35l56_sdw_bus_config,
 #ifdef DEBUG
 	.clk_stop = cs35l56_sdw_clk_stop,
 #endif
@@ -528,6 +597,10 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
 		regmap_config = &cs35l56_regmap_sdw;
 		cs35l56->base.fw_reg = &cs35l56_fw_reg;
 		break;
+	case 0x3563:
+		regmap_config = &cs35l63_regmap_sdw;
+		cs35l56->base.fw_reg = &cs35l63_fw_reg;
+		break;
 	default:
 		return -ENODEV;
 	}
@@ -575,6 +648,7 @@ static const struct dev_pm_ops cs35l56_sdw_pm = {
 static const struct sdw_device_id cs35l56_sdw_id[] = {
 	SDW_SLAVE_ENTRY(0x01FA, 0x3556, 0x3556),
 	SDW_SLAVE_ENTRY(0x01FA, 0x3557, 0x3557),
+	SDW_SLAVE_ENTRY(0x01FA, 0x3563, 0x3563),
 	{},
 };
 MODULE_DEVICE_TABLE(sdw, cs35l56_sdw_id);
diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
index da982774bb4d..76ddb1cf6889 100644
--- a/sound/soc/codecs/cs35l56-shared.c
+++ b/sound/soc/codecs/cs35l56-shared.c
@@ -47,6 +47,13 @@ static const struct reg_sequence cs35l56_patch_fw[] = {
 	{ CS35L56_MAIN_POSTURE_NUMBER,		0x00000000 },
 };
 
+static const struct reg_sequence cs35l63_patch_fw[] = {
+	/* These are not reset by a soft-reset, so patch to defaults. */
+	{ CS35L63_MAIN_RENDER_USER_MUTE,	0x00000000 },
+	{ CS35L63_MAIN_RENDER_USER_VOLUME,	0x00000000 },
+	{ CS35L63_MAIN_POSTURE_NUMBER,		0x00000000 },
+};
+
 int cs35l56_set_patch(struct cs35l56_base *cs35l56_base)
 {
 	int ret;
@@ -64,6 +71,10 @@ int cs35l56_set_patch(struct cs35l56_base *cs35l56_base)
 		ret = regmap_register_patch(cs35l56_base->regmap, cs35l56_patch_fw,
 					    ARRAY_SIZE(cs35l56_patch_fw));
 		break;
+	case 0x63:
+		ret = regmap_register_patch(cs35l56_base->regmap, cs35l63_patch_fw,
+					    ARRAY_SIZE(cs35l63_patch_fw));
+		break;
 	default:
 		break;
 	}
@@ -102,6 +113,36 @@ static const struct reg_default cs35l56_reg_defaults[] = {
 	{ CS35L56_MAIN_POSTURE_NUMBER,		0x00000000 },
 };
 
+static const struct reg_default cs35l63_reg_defaults[] = {
+	/* no defaults for OTP_MEM - first read populates cache */
+
+	{ CS35L56_ASP1_ENABLES1,		0x00000000 },
+	{ CS35L56_ASP1_CONTROL1,		0x00000028 },
+	{ CS35L56_ASP1_CONTROL2,		0x18180200 },
+	{ CS35L56_ASP1_CONTROL3,		0x00000002 },
+	{ CS35L56_ASP1_FRAME_CONTROL1,		0x03020100 },
+	{ CS35L56_ASP1_FRAME_CONTROL5,		0x00020100 },
+	{ CS35L56_ASP1_DATA_CONTROL1,		0x00000018 },
+	{ CS35L56_ASP1_DATA_CONTROL5,		0x00000018 },
+	{ CS35L56_ASP1TX1_INPUT,		0x00000000 },
+	{ CS35L56_ASP1TX2_INPUT,		0x00000000 },
+	{ CS35L56_ASP1TX3_INPUT,		0x00000000 },
+	{ CS35L56_ASP1TX4_INPUT,		0x00000000 },
+	{ CS35L56_SWIRE_DP3_CH1_INPUT,		0x00000018 },
+	{ CS35L56_SWIRE_DP3_CH2_INPUT,		0x00000019 },
+	{ CS35L56_SWIRE_DP3_CH3_INPUT,		0x00000029 },
+	{ CS35L56_SWIRE_DP3_CH4_INPUT,		0x00000028 },
+	{ CS35L56_IRQ1_MASK_1,			0x8003ffff },
+	{ CS35L56_IRQ1_MASK_2,			0xffff7fff },
+	{ CS35L56_IRQ1_MASK_4,			0xe0ffffff },
+	{ CS35L56_IRQ1_MASK_8,			0x8c000fff },
+	{ CS35L56_IRQ1_MASK_18,			0x0760f000 },
+	{ CS35L56_IRQ1_MASK_20,			0x15c00000 },
+	{ CS35L63_MAIN_RENDER_USER_MUTE,	0x00000000 },
+	{ CS35L63_MAIN_RENDER_USER_VOLUME,	0x00000000 },
+	{ CS35L63_MAIN_POSTURE_NUMBER,		0x00000000 },
+};
+
 static bool cs35l56_is_dsp_memory(unsigned int reg)
 {
 	switch (reg) {
@@ -199,7 +240,7 @@ static bool cs35l56_precious_reg(struct device *dev, unsigned int reg)
 	}
 }
 
-static bool cs35l56_volatile_reg(struct device *dev, unsigned int reg)
+static bool cs35l56_common_volatile_reg(unsigned int reg)
 {
 	switch (reg) {
 	case CS35L56_DEVID:
@@ -237,12 +278,32 @@ static bool cs35l56_volatile_reg(struct device *dev, unsigned int reg)
 	case CS35L56_DSP1_SCRATCH3:
 	case CS35L56_DSP1_SCRATCH4:
 		return true;
+	default:
+		return cs35l56_is_dsp_memory(reg);
+	}
+}
+
+static bool cs35l56_volatile_reg(struct device *dev, unsigned int reg)
+{
+	switch (reg) {
 	case CS35L56_MAIN_RENDER_USER_MUTE:
 	case CS35L56_MAIN_RENDER_USER_VOLUME:
 	case CS35L56_MAIN_POSTURE_NUMBER:
 		return false;
 	default:
-		return cs35l56_is_dsp_memory(reg);
+		return cs35l56_common_volatile_reg(reg);
+	}
+}
+
+static bool cs35l63_volatile_reg(struct device *dev, unsigned int reg)
+{
+	switch (reg) {
+	case CS35L63_MAIN_RENDER_USER_MUTE:
+	case CS35L63_MAIN_RENDER_USER_VOLUME:
+	case CS35L63_MAIN_POSTURE_NUMBER:
+		return false;
+	default:
+		return cs35l56_common_volatile_reg(reg);
 	}
 }
 
@@ -405,6 +466,11 @@ static const struct reg_sequence cs35l56_system_reset_seq[] = {
 	REG_SEQ0(CS35L56_DSP_VIRTUAL1_MBOX_1, CS35L56_MBOX_CMD_SYSTEM_RESET),
 };
 
+static const struct reg_sequence cs35l63_system_reset_seq[] = {
+	REG_SEQ0(CS35L63_DSP1_HALO_STATE, 0),
+	REG_SEQ0(CS35L56_DSP_VIRTUAL1_MBOX_1, CS35L56_MBOX_CMD_SYSTEM_RESET),
+};
+
 void cs35l56_system_reset(struct cs35l56_base *cs35l56_base, bool is_soundwire)
 {
 	/*
@@ -426,6 +492,11 @@ void cs35l56_system_reset(struct cs35l56_base *cs35l56_base, bool is_soundwire)
 						cs35l56_system_reset_seq,
 						ARRAY_SIZE(cs35l56_system_reset_seq));
 		break;
+	case 0x63:
+		regmap_multi_reg_write_bypassed(cs35l56_base->regmap,
+						cs35l63_system_reset_seq,
+						ARRAY_SIZE(cs35l63_system_reset_seq));
+		break;
 	default:
 		break;
 	}
@@ -844,6 +915,9 @@ int cs35l56_hw_init(struct cs35l56_base *cs35l56_base)
 	case 0x35A56:
 	case 0x35A57:
 		break;
+	case 0x35A630:
+		devid = devid >> 4;
+		break;
 	default:
 		dev_err(cs35l56_base->dev, "Unknown device %x\n", devid);
 		return ret;
@@ -1080,6 +1154,39 @@ const struct regmap_config cs35l56_regmap_sdw = {
 };
 EXPORT_SYMBOL_NS_GPL(cs35l56_regmap_sdw, "SND_SOC_CS35L56_SHARED");
 
+const struct regmap_config cs35l63_regmap_i2c = {
+	.reg_bits = 32,
+	.val_bits = 32,
+	.reg_stride = 4,
+	.reg_base = 0x8000,
+	.reg_format_endian = REGMAP_ENDIAN_BIG,
+	.val_format_endian = REGMAP_ENDIAN_BIG,
+	.max_register = CS35L56_DSP1_PMEM_5114,
+	.reg_defaults = cs35l63_reg_defaults,
+	.num_reg_defaults = ARRAY_SIZE(cs35l63_reg_defaults),
+	.volatile_reg = cs35l63_volatile_reg,
+	.readable_reg = cs35l56_readable_reg,
+	.precious_reg = cs35l56_precious_reg,
+	.cache_type = REGCACHE_MAPLE,
+};
+EXPORT_SYMBOL_NS_GPL(cs35l63_regmap_i2c, "SND_SOC_CS35L56_SHARED");
+
+const struct regmap_config cs35l63_regmap_sdw = {
+	.reg_bits = 32,
+	.val_bits = 32,
+	.reg_stride = 4,
+	.reg_format_endian = REGMAP_ENDIAN_LITTLE,
+	.val_format_endian = REGMAP_ENDIAN_BIG,
+	.max_register = CS35L56_DSP1_PMEM_5114,
+	.reg_defaults = cs35l63_reg_defaults,
+	.num_reg_defaults = ARRAY_SIZE(cs35l63_reg_defaults),
+	.volatile_reg = cs35l63_volatile_reg,
+	.readable_reg = cs35l56_readable_reg,
+	.precious_reg = cs35l56_precious_reg,
+	.cache_type = REGCACHE_MAPLE,
+};
+EXPORT_SYMBOL_NS_GPL(cs35l63_regmap_sdw, "SND_SOC_CS35L56_SHARED");
+
 const struct cs35l56_fw_reg cs35l56_fw_reg = {
 	.fw_ver = CS35L56_DSP1_FW_VER,
 	.halo_state = CS35L56_DSP1_HALO_STATE,
@@ -1092,6 +1199,18 @@ const struct cs35l56_fw_reg cs35l56_fw_reg = {
 };
 EXPORT_SYMBOL_NS_GPL(cs35l56_fw_reg, "SND_SOC_CS35L56_SHARED");
 
+const struct cs35l56_fw_reg cs35l63_fw_reg = {
+	.fw_ver = CS35L63_DSP1_FW_VER,
+	.halo_state = CS35L63_DSP1_HALO_STATE,
+	.pm_cur_stat = CS35L63_DSP1_PM_CUR_STATE,
+	.prot_sts = CS35L63_PROTECTION_STATUS,
+	.transducer_actual_ps = CS35L63_TRANSDUCER_ACTUAL_PS,
+	.user_mute = CS35L63_MAIN_RENDER_USER_MUTE,
+	.user_volume = CS35L63_MAIN_RENDER_USER_VOLUME,
+	.posture_number = CS35L63_MAIN_POSTURE_NUMBER,
+};
+EXPORT_SYMBOL_NS_GPL(cs35l63_fw_reg, "SND_SOC_CS35L56_SHARED");
+
 MODULE_DESCRIPTION("ASoC CS35L56 Shared");
 MODULE_AUTHOR("Richard Fitzgerald <rf@opensource.cirrus.com>");
 MODULE_AUTHOR("Simon Trimmer <simont@opensource.cirrus.com>");
diff --git a/sound/soc/codecs/cs35l56.c b/sound/soc/codecs/cs35l56.c
index a4a1d09097fc..cdb283ed938c 100644
--- a/sound/soc/codecs/cs35l56.c
+++ b/sound/soc/codecs/cs35l56.c
@@ -84,6 +84,25 @@ static const struct snd_kcontrol_new cs35l56_controls[] = {
 		       cs35l56_dspwait_get_volsw, cs35l56_dspwait_put_volsw),
 };
 
+static const struct snd_kcontrol_new cs35l63_controls[] = {
+	SOC_SINGLE_EXT("Speaker Switch",
+		       CS35L63_MAIN_RENDER_USER_MUTE, 0, 1, 1,
+		       cs35l56_dspwait_get_volsw, cs35l56_dspwait_put_volsw),
+	SOC_SINGLE_S_EXT_TLV("Speaker Volume",
+			     CS35L63_MAIN_RENDER_USER_VOLUME,
+			     CS35L56_MAIN_RENDER_USER_VOLUME_SHIFT,
+			     CS35L56_MAIN_RENDER_USER_VOLUME_MIN,
+			     CS35L56_MAIN_RENDER_USER_VOLUME_MAX,
+			     CS35L56_MAIN_RENDER_USER_VOLUME_SIGNBIT,
+			     0,
+			     cs35l56_dspwait_get_volsw,
+			     cs35l56_dspwait_put_volsw,
+			     vol_tlv),
+	SOC_SINGLE_EXT("Posture Number", CS35L63_MAIN_POSTURE_NUMBER,
+		       0, 255, 0,
+		       cs35l56_dspwait_get_volsw, cs35l56_dspwait_put_volsw),
+};
+
 static SOC_VALUE_ENUM_SINGLE_DECL(cs35l56_asp1tx1_enum,
 				  CS35L56_ASP1TX1_INPUT,
 				  0, CS35L56_ASP_TXn_SRC_MASK,
@@ -886,6 +905,10 @@ static int cs35l56_component_probe(struct snd_soc_component *component)
 		ret = snd_soc_add_component_controls(component, cs35l56_controls,
 						     ARRAY_SIZE(cs35l56_controls));
 		break;
+	case 0x63:
+		ret = snd_soc_add_component_controls(component, cs35l63_controls,
+						     ARRAY_SIZE(cs35l63_controls));
+		break;
 	default:
 		ret = -ENODEV;
 		break;
diff --git a/sound/soc/codecs/cs35l56.h b/sound/soc/codecs/cs35l56.h
index 8a987ec01507..200f695efca3 100644
--- a/sound/soc/codecs/cs35l56.h
+++ b/sound/soc/codecs/cs35l56.h
@@ -51,6 +51,7 @@ struct cs35l56_private {
 	u8 asp_slot_count;
 	bool tdm_mode;
 	bool sysclk_set;
+	u8 old_sdw_clock_scale;
 };
 
 extern const struct dev_pm_ops cs35l56_pm_ops_i2c_spi;
-- 
2.43.0


^ permalink raw reply related	[flat|nested] 18+ messages in thread

* [PATCH v1 5/5] ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63
  2025-04-07 15:16 [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Stefan Binding
                   ` (3 preceding siblings ...)
  2025-04-07 15:16 ` [PATCH v1 4/5] ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire Stefan Binding
@ 2025-04-07 15:16 ` Stefan Binding
  2025-04-28 17:39   ` Richard Fitzgerald
  2025-05-07 11:12 ` [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Mark Brown
  5 siblings, 1 reply; 18+ messages in thread
From: Stefan Binding @ 2025-04-07 15:16 UTC (permalink / raw)
  To: Mark Brown; +Cc: linux-sound, linux-kernel, patches, Stefan Binding

On CS35L63 the DIE_STS registers are populated by the Firmware from
OTP, so the driver can read these registers directly, rather than
obtaining them from OTP.

Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
---
 include/sound/cs35l56.h           |  2 ++
 sound/soc/codecs/cs35l56-shared.c | 39 ++++++++++++++++++++++++++++---
 2 files changed, 38 insertions(+), 3 deletions(-)

diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
index e16e1a94c8a1..63f2c63f7c59 100644
--- a/include/sound/cs35l56.h
+++ b/include/sound/cs35l56.h
@@ -71,6 +71,8 @@
 #define CS35L56_DSP_VIRTUAL1_MBOX_6			0x0011034
 #define CS35L56_DSP_VIRTUAL1_MBOX_7			0x0011038
 #define CS35L56_DSP_VIRTUAL1_MBOX_8			0x001103C
+#define CS35L56_DIE_STS1				0x0017040
+#define CS35L56_DIE_STS2				0x0017044
 #define CS35L56_DSP_RESTRICT_STS1			0x00190F0
 #define CS35L56_DSP1_XMEM_PACKED_0			0x2000000
 #define CS35L56_DSP1_XMEM_PACKED_6143			0x2005FFC
diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
index 76ddb1cf6889..7f768718b69b 100644
--- a/sound/soc/codecs/cs35l56-shared.c
+++ b/sound/soc/codecs/cs35l56-shared.c
@@ -214,6 +214,8 @@ static bool cs35l56_readable_reg(struct device *dev, unsigned int reg)
 	case CS35L56_DSP_VIRTUAL1_MBOX_6:
 	case CS35L56_DSP_VIRTUAL1_MBOX_7:
 	case CS35L56_DSP_VIRTUAL1_MBOX_8:
+	case CS35L56_DIE_STS1:
+	case CS35L56_DIE_STS2:
 	case CS35L56_DSP_RESTRICT_STS1:
 	case CS35L56_DSP1_SYS_INFO_ID ... CS35L56_DSP1_SYS_INFO_END:
 	case CS35L56_DSP1_AHBM_WINDOW_DEBUG_0:
@@ -802,13 +804,29 @@ static int cs35l56_read_silicon_uid(struct cs35l56_base *cs35l56_base, u64 *uid)
 	unique_id |= (u32)pte.x | ((u32)pte.y << 8) | ((u32)pte.wafer_id << 16) |
 		     ((u32)pte.dvs << 24);
 
-	dev_dbg(cs35l56_base->dev, "UniqueID = %#llx\n", unique_id);
-
 	*uid = unique_id;
 
 	return 0;
 }
 
+static int cs35l63_read_silicon_uid(struct cs35l56_base *cs35l56_base, u64 *uid)
+{
+	u32 tmp[2];
+	int ret;
+
+	ret = regmap_bulk_read(cs35l56_base->regmap, CS35L56_DIE_STS1, tmp, ARRAY_SIZE(tmp));
+	if (ret) {
+		dev_err(cs35l56_base->dev, "Cannot obtain CS35L56_DIE_STS: %d\n", ret);
+		return ret;
+	}
+
+	*uid = tmp[1];
+	*uid <<= 32;
+	*uid |= tmp[0];
+
+	return 0;
+}
+
 /* Firmware calibration controls */
 const struct cirrus_amp_cal_controls cs35l56_calibration_controls = {
 	.alg_id =	0x9f210,
@@ -829,10 +847,25 @@ int cs35l56_get_calibration(struct cs35l56_base *cs35l56_base)
 	if (cs35l56_base->secured)
 		return 0;
 
-	ret = cs35l56_read_silicon_uid(cs35l56_base, &silicon_uid);
+	switch (cs35l56_base->type) {
+	case 0x54:
+	case 0x56:
+	case 0x57:
+		ret = cs35l56_read_silicon_uid(cs35l56_base, &silicon_uid);
+		break;
+	case 0x63:
+		ret = cs35l63_read_silicon_uid(cs35l56_base, &silicon_uid);
+		break;
+	default:
+		ret = -ENODEV;
+		break;
+	}
+
 	if (ret < 0)
 		return ret;
 
+	dev_dbg(cs35l56_base->dev, "UniqueID = %#llx\n", silicon_uid);
+
 	ret = cs_amp_get_efi_calibration_data(cs35l56_base->dev, silicon_uid,
 					      cs35l56_base->cal_index,
 					      &cs35l56_base->cal_data);
-- 
2.43.0


^ permalink raw reply related	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-07 15:16 ` [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers Stefan Binding
@ 2025-04-07 19:16   ` Mark Brown
  2025-04-08 12:58     ` Richard Fitzgerald
  2025-04-28 17:34   ` Richard Fitzgerald
  1 sibling, 1 reply; 18+ messages in thread
From: Mark Brown @ 2025-04-07 19:16 UTC (permalink / raw)
  To: Stefan Binding; +Cc: linux-sound, linux-kernel, patches

[-- Attachment #1: Type: text/plain, Size: 478 bytes --]

On Mon, Apr 07, 2025 at 04:16:41PM +0100, Stefan Binding wrote:

> Firmware based registers may be different addresses across different
> device ids and revision ids. Create a structure to store and access
> these addresses.

This is fine but note that this is the use case that the regmap_field_
APIs were created for, that also helps deal with things if anyone is
clever and resizes fields or shifts within registers.  It's purely a
question of taste if you want to use that.

[-- Attachment #2: signature.asc --]
[-- Type: application/pgp-signature, Size: 488 bytes --]

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-07 19:16   ` Mark Brown
@ 2025-04-08 12:58     ` Richard Fitzgerald
  2025-04-08 16:00       ` Mark Brown
  0 siblings, 1 reply; 18+ messages in thread
From: Richard Fitzgerald @ 2025-04-08 12:58 UTC (permalink / raw)
  To: Mark Brown, Stefan Binding; +Cc: linux-sound, linux-kernel, patches

On 07/04/2025 8:16 pm, Mark Brown wrote:
> On Mon, Apr 07, 2025 at 04:16:41PM +0100, Stefan Binding wrote:
> 
>> Firmware based registers may be different addresses across different
>> device ids and revision ids. Create a structure to store and access
>> these addresses.
> 
> This is fine but note that this is the use case that the regmap_field_
> APIs were created for, that also helps deal with things if anyone is
> clever and resizes fields or shifts within registers.  It's purely a
> question of taste if you want to use that.

The regmap_field stuff looks like a lot of unnecessary complexity and
overhead just for 6 registers with alternate addresses.

(In fact, the regmap_field stuff looks like an over-complex way of
solving a non-problem. The original commit is talking about replacing
masks and shifts "all over the code" to make the code neater. But
really, all those extra structs and pointers and allocated memory just
to replace a logical & or | ? Every struct regmap_field has a pointer
to the struct regmap !!?! So if I've got 100 fields there are 100 copies
of the struct regmap pointer that my driver already has.)

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-08 12:58     ` Richard Fitzgerald
@ 2025-04-08 16:00       ` Mark Brown
  2025-04-09 11:33         ` Richard Fitzgerald
  0 siblings, 1 reply; 18+ messages in thread
From: Mark Brown @ 2025-04-08 16:00 UTC (permalink / raw)
  To: Richard Fitzgerald; +Cc: Stefan Binding, linux-sound, linux-kernel, patches

[-- Attachment #1: Type: text/plain, Size: 1203 bytes --]

On Tue, Apr 08, 2025 at 01:58:23PM +0100, Richard Fitzgerald wrote:
> On 07/04/2025 8:16 pm, Mark Brown wrote:

> > This is fine but note that this is the use case that the regmap_field_
> > APIs were created for, that also helps deal with things if anyone is
> > clever and resizes fields or shifts within registers.  It's purely a
> > question of taste if you want to use that.

> The regmap_field stuff looks like a lot of unnecessary complexity and
> overhead just for 6 registers with alternate addresses.

Yeah, like I say it's a taste thing.

> (In fact, the regmap_field stuff looks like an over-complex way of
> solving a non-problem. The original commit is talking about replacing
> masks and shifts "all over the code" to make the code neater. But
> really, all those extra structs and pointers and allocated memory just
> to replace a logical & or | ? Every struct regmap_field has a pointer
> to the struct regmap !!?! So if I've got 100 fields there are 100 copies
> of the struct regmap pointer that my driver already has.)

It was for cases where the shifts and widths may also change, there's a
bit more than applying a mask.  Like you say it's got some overhead
hence the taste thing.

[-- Attachment #2: signature.asc --]
[-- Type: application/pgp-signature, Size: 488 bytes --]

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-08 16:00       ` Mark Brown
@ 2025-04-09 11:33         ` Richard Fitzgerald
  2025-04-09 12:31           ` Mark Brown
  0 siblings, 1 reply; 18+ messages in thread
From: Richard Fitzgerald @ 2025-04-09 11:33 UTC (permalink / raw)
  To: Mark Brown; +Cc: Stefan Binding, linux-sound, linux-kernel, patches

On 08/04/2025 5:00 pm, Mark Brown wrote:
> On Tue, Apr 08, 2025 at 01:58:23PM +0100, Richard Fitzgerald wrote:
>> On 07/04/2025 8:16 pm, Mark Brown wrote:
> 
>>> This is fine but note that this is the use case that the regmap_field_
>>> APIs were created for, that also helps deal with things if anyone is
>>> clever and resizes fields or shifts within registers.  It's purely a
>>> question of taste if you want to use that.
> 
>> The regmap_field stuff looks like a lot of unnecessary complexity and
>> overhead just for 6 registers with alternate addresses.
> 
> Yeah, like I say it's a taste thing.
>

If you would prefer us to use it we'll make the changes.
If not, then we'd like to keep the code as it is for now and in
future we'll have a look at regmap_field.

>> (In fact, the regmap_field stuff looks like an over-complex way of
>> solving a non-problem. The original commit is talking about replacing
>> masks and shifts "all over the code" to make the code neater. But
>> really, all those extra structs and pointers and allocated memory just
>> to replace a logical & or | ? Every struct regmap_field has a pointer
>> to the struct regmap !!?! So if I've got 100 fields there are 100 copies
>> of the struct regmap pointer that my driver already has.)
> 
> It was for cases where the shifts and widths may also change, there's a
> bit more than applying a mask.  Like you say it's got some overhead
> hence the taste thing.

Yes. If we had registers where the fields were moving around then there
would be better justification for using regmap_field.

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-09 11:33         ` Richard Fitzgerald
@ 2025-04-09 12:31           ` Mark Brown
  0 siblings, 0 replies; 18+ messages in thread
From: Mark Brown @ 2025-04-09 12:31 UTC (permalink / raw)
  To: Richard Fitzgerald; +Cc: Stefan Binding, linux-sound, linux-kernel, patches

[-- Attachment #1: Type: text/plain, Size: 346 bytes --]

On Wed, Apr 09, 2025 at 12:33:50PM +0100, Richard Fitzgerald wrote:
> On 08/04/2025 5:00 pm, Mark Brown wrote:

> > Yeah, like I say it's a taste thing.

> If you would prefer us to use it we'll make the changes.
> If not, then we'd like to keep the code as it is for now and in
> future we'll have a look at regmap_field.

It's fine either way.

[-- Attachment #2: signature.asc --]
[-- Type: application/pgp-signature, Size: 488 bytes --]

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config
  2025-04-07 15:16 ` [PATCH v1 1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config Stefan Binding
@ 2025-04-28 17:32   ` Richard Fitzgerald
  0 siblings, 0 replies; 18+ messages in thread
From: Richard Fitzgerald @ 2025-04-28 17:32 UTC (permalink / raw)
  To: Stefan Binding, Mark Brown; +Cc: linux-sound, linux-kernel, patches

On 7/4/25 16:16, Stefan Binding wrote:
> This is to prepare for further products using slightly different
> regmap configs.
> 
> Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
> ---
>   sound/soc/codecs/cs35l56-i2c.c | 16 +++++++++++++---
>   sound/soc/codecs/cs35l56-sdw.c | 16 +++++++++++++---
>   2 files changed, 26 insertions(+), 6 deletions(-)
> 
> diff --git a/sound/soc/codecs/cs35l56-i2c.c b/sound/soc/codecs/cs35l56-i2c.c
> index 8a518df1e16e..5962914e2180 100644
> --- a/sound/soc/codecs/cs35l56-i2c.c
> +++ b/sound/soc/codecs/cs35l56-i2c.c
> @@ -17,9 +17,10 @@
>   
>   static int cs35l56_i2c_probe(struct i2c_client *client)
>   {
> +	unsigned int id = (u32)(uintptr_t)i2c_get_match_data(client);
>   	struct cs35l56_private *cs35l56;
>   	struct device *dev = &client->dev;
> -	const struct regmap_config *regmap_config = &cs35l56_regmap_i2c;
> +	const struct regmap_config *regmap_config;
>   	int ret;
>   
>   	cs35l56 = devm_kzalloc(dev, sizeof(struct cs35l56_private), GFP_KERNEL);
> @@ -30,6 +31,15 @@ static int cs35l56_i2c_probe(struct i2c_client *client)
>   	cs35l56->base.can_hibernate = true;
>   
>   	i2c_set_clientdata(client, cs35l56);
> +
> +	switch (id) {
> +	case 0x3556:
> +		regmap_config = &cs35l56_regmap_i2c;
> +		break;
> +	default:
> +		return -ENODEV;
> +	}
> +
>   	cs35l56->base.regmap = devm_regmap_init_i2c(client, regmap_config);
>   	if (IS_ERR(cs35l56->base.regmap)) {
>   		ret = PTR_ERR(cs35l56->base.regmap);
> @@ -57,14 +67,14 @@ static void cs35l56_i2c_remove(struct i2c_client *client)
>   }
>   
>   static const struct i2c_device_id cs35l56_id_i2c[] = {
> -	{ "cs35l56" },
> +	{ "cs35l56", 0x3556 },
>   	{}
>   };
>   MODULE_DEVICE_TABLE(i2c, cs35l56_id_i2c);
>   
>   #ifdef CONFIG_ACPI
>   static const struct acpi_device_id cs35l56_asoc_acpi_match[] = {
> -	{ "CSC355C", 0 },
> +	{ "CSC355C", 0x3556 },
>   	{},
>   };
>   MODULE_DEVICE_TABLE(acpi, cs35l56_asoc_acpi_match);
> diff --git a/sound/soc/codecs/cs35l56-sdw.c b/sound/soc/codecs/cs35l56-sdw.c
> index 3f91cb3f9ae7..d178357e1196 100644
> --- a/sound/soc/codecs/cs35l56-sdw.c
> +++ b/sound/soc/codecs/cs35l56-sdw.c
> @@ -509,6 +509,7 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
>   {
>   	struct device *dev = &peripheral->dev;
>   	struct cs35l56_private *cs35l56;
> +	const struct regmap_config *regmap_config;
>   	int ret;
>   
>   	cs35l56 = devm_kzalloc(dev, sizeof(*cs35l56), GFP_KERNEL);
> @@ -521,8 +522,17 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
>   
>   	dev_set_drvdata(dev, cs35l56);
>   
> +	switch ((unsigned int)id->driver_data) {
> +	case 0x3556:
> +	case 0x3557:
> +		regmap_config = &cs35l56_regmap_sdw;
> +		break;
> +	default:
> +		return -ENODEV;
> +	}
> +
>   	cs35l56->base.regmap = devm_regmap_init(dev, &cs35l56_regmap_bus_sdw,
> -					   peripheral, &cs35l56_regmap_sdw);
> +					   peripheral, regmap_config);
>   	if (IS_ERR(cs35l56->base.regmap)) {
>   		ret = PTR_ERR(cs35l56->base.regmap);
>   		return dev_err_probe(dev, ret, "Failed to allocate register map\n");
> @@ -562,8 +572,8 @@ static const struct dev_pm_ops cs35l56_sdw_pm = {
>   };
>   
>   static const struct sdw_device_id cs35l56_sdw_id[] = {
> -	SDW_SLAVE_ENTRY(0x01FA, 0x3556, 0),
> -	SDW_SLAVE_ENTRY(0x01FA, 0x3557, 0),
> +	SDW_SLAVE_ENTRY(0x01FA, 0x3556, 0x3556),
> +	SDW_SLAVE_ENTRY(0x01FA, 0x3557, 0x3557),
>   	{},
>   };
>   MODULE_DEVICE_TABLE(sdw, cs35l56_sdw_id);

Reviewed-by: Richard Fitzgerald <rf@opensource.cirrus.com>

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-07 15:16 ` [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers Stefan Binding
  2025-04-07 19:16   ` Mark Brown
@ 2025-04-28 17:34   ` Richard Fitzgerald
  2025-05-05 23:46     ` Mark Brown
  1 sibling, 1 reply; 18+ messages in thread
From: Richard Fitzgerald @ 2025-04-28 17:34 UTC (permalink / raw)
  To: Stefan Binding, Mark Brown; +Cc: linux-sound, linux-kernel, patches

On 7/4/25 16:16, Stefan Binding wrote:
> Firmware based registers may be different addresses across different
> device ids and revision ids. Create a structure to store and access
> these addresses.
> 
> Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
> ---
>   include/sound/cs35l56.h           | 11 ++++++++
>   sound/pci/hda/cs35l56_hda.c       |  5 ++--
>   sound/pci/hda/cs35l56_hda_i2c.c   |  3 +++
>   sound/pci/hda/cs35l56_hda_spi.c   |  3 +++
>   sound/soc/codecs/cs35l56-i2c.c    |  1 +
>   sound/soc/codecs/cs35l56-sdw.c    |  1 +
>   sound/soc/codecs/cs35l56-shared.c | 42 ++++++++++++++++++++++++-------
>   sound/soc/codecs/cs35l56-spi.c    |  3 +++
>   sound/soc/codecs/cs35l56.c        |  5 ++--
>   9 files changed, 61 insertions(+), 13 deletions(-)
> 
> diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
> index 5d653a3491d0..d712cb79652b 100644
> --- a/include/sound/cs35l56.h
> +++ b/include/sound/cs35l56.h
> @@ -267,6 +267,14 @@ struct cs35l56_spi_payload {
>   } __packed;
>   static_assert(sizeof(struct cs35l56_spi_payload) == 10);
>   
> +struct cs35l56_fw_reg {
> +	unsigned int fw_ver;
> +	unsigned int halo_state;
> +	unsigned int pm_cur_stat;
> +	unsigned int prot_sts;
> +	unsigned int transducer_actual_ps;
> +};
> +
>   struct cs35l56_base {
>   	struct device *dev;
>   	struct regmap *regmap;
> @@ -283,6 +291,7 @@ struct cs35l56_base {
>   	struct cirrus_amp_cal_data cal_data;
>   	struct gpio_desc *reset_gpio;
>   	struct cs35l56_spi_payload *spi_payload_buf;
> +	const struct cs35l56_fw_reg *fw_reg;
>   };
>   
>   static inline bool cs35l56_is_otp_register(unsigned int reg)
> @@ -311,6 +320,8 @@ extern const struct regmap_config cs35l56_regmap_i2c;
>   extern const struct regmap_config cs35l56_regmap_spi;
>   extern const struct regmap_config cs35l56_regmap_sdw;
>   
> +extern const struct cs35l56_fw_reg cs35l56_fw_reg;
> +
>   extern const struct cirrus_amp_cal_controls cs35l56_calibration_controls;
>   
>   extern const char * const cs35l56_tx_input_texts[CS35L56_NUM_INPUT_SRC];
> diff --git a/sound/pci/hda/cs35l56_hda.c b/sound/pci/hda/cs35l56_hda.c
> index 4ef7878e8fd4..f6257fe45b9f 100644
> --- a/sound/pci/hda/cs35l56_hda.c
> +++ b/sound/pci/hda/cs35l56_hda.c
> @@ -68,7 +68,7 @@ static void cs35l56_hda_play(struct cs35l56_hda *cs35l56)
>   	if (ret == 0) {
>   		/* Wait for firmware to enter PS0 power state */
>   		ret = regmap_read_poll_timeout(cs35l56->base.regmap,
> -					       CS35L56_TRANSDUCER_ACTUAL_PS,
> +					       cs35l56->base.fw_reg->transducer_actual_ps,
>   					       val, (val == CS35L56_PS0),
>   					       CS35L56_PS0_POLL_US,
>   					       CS35L56_PS0_TIMEOUT_US);
> @@ -667,7 +667,8 @@ static void cs35l56_hda_fw_load(struct cs35l56_hda *cs35l56)
>   
>   	regcache_sync(cs35l56->base.regmap);
>   
> -	regmap_clear_bits(cs35l56->base.regmap, CS35L56_PROTECTION_STATUS,
> +	regmap_clear_bits(cs35l56->base.regmap,
> +			  cs35l56->base.fw_reg->prot_sts,
>   			  CS35L56_FIRMWARE_MISSING);
>   	cs35l56->base.fw_patched = true;
>   
> diff --git a/sound/pci/hda/cs35l56_hda_i2c.c b/sound/pci/hda/cs35l56_hda_i2c.c
> index c7b836613149..d10209e4eddd 100644
> --- a/sound/pci/hda/cs35l56_hda_i2c.c
> +++ b/sound/pci/hda/cs35l56_hda_i2c.c
> @@ -26,6 +26,9 @@ static int cs35l56_hda_i2c_probe(struct i2c_client *clt)
>   #ifdef CS35L56_WAKE_HOLD_TIME_US
>   	cs35l56->base.can_hibernate = true;
>   #endif
> +
> +	cs35l56->base.fw_reg = &cs35l56_fw_reg;
> +
>   	cs35l56->base.regmap = devm_regmap_init_i2c(clt, &cs35l56_regmap_i2c);
>   	if (IS_ERR(cs35l56->base.regmap)) {
>   		ret = PTR_ERR(cs35l56->base.regmap);
> diff --git a/sound/pci/hda/cs35l56_hda_spi.c b/sound/pci/hda/cs35l56_hda_spi.c
> index 903578466905..f57533d3d728 100644
> --- a/sound/pci/hda/cs35l56_hda_spi.c
> +++ b/sound/pci/hda/cs35l56_hda_spi.c
> @@ -29,6 +29,9 @@ static int cs35l56_hda_spi_probe(struct spi_device *spi)
>   #ifdef CS35L56_WAKE_HOLD_TIME_US
>   	cs35l56->base.can_hibernate = true;
>   #endif
> +
> +	cs35l56->base.fw_reg = &cs35l56_fw_reg;
> +
>   	cs35l56->base.regmap = devm_regmap_init_spi(spi, &cs35l56_regmap_spi);
>   	if (IS_ERR(cs35l56->base.regmap)) {
>   		ret = PTR_ERR(cs35l56->base.regmap);
> diff --git a/sound/soc/codecs/cs35l56-i2c.c b/sound/soc/codecs/cs35l56-i2c.c
> index 5962914e2180..38c391d11c78 100644
> --- a/sound/soc/codecs/cs35l56-i2c.c
> +++ b/sound/soc/codecs/cs35l56-i2c.c
> @@ -35,6 +35,7 @@ static int cs35l56_i2c_probe(struct i2c_client *client)
>   	switch (id) {
>   	case 0x3556:
>   		regmap_config = &cs35l56_regmap_i2c;
> +		cs35l56->base.fw_reg = &cs35l56_fw_reg;
>   		break;
>   	default:
>   		return -ENODEV;
> diff --git a/sound/soc/codecs/cs35l56-sdw.c b/sound/soc/codecs/cs35l56-sdw.c
> index d178357e1196..2e0422b41385 100644
> --- a/sound/soc/codecs/cs35l56-sdw.c
> +++ b/sound/soc/codecs/cs35l56-sdw.c
> @@ -526,6 +526,7 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
>   	case 0x3556:
>   	case 0x3557:
>   		regmap_config = &cs35l56_regmap_sdw;
> +		cs35l56->base.fw_reg = &cs35l56_fw_reg;
>   		break;
>   	default:
>   		return -ENODEV;
> diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
> index e28bfefa72f3..bc8f9379bc74 100644
> --- a/sound/soc/codecs/cs35l56-shared.c
> +++ b/sound/soc/codecs/cs35l56-shared.c
> @@ -253,7 +253,8 @@ int cs35l56_firmware_shutdown(struct cs35l56_base *cs35l56_base)
>   	if (ret)
>   		return ret;
>   
> -	ret = regmap_read_poll_timeout(cs35l56_base->regmap,  CS35L56_DSP1_PM_CUR_STATE,
> +	ret = regmap_read_poll_timeout(cs35l56_base->regmap,
> +				       cs35l56_base->fw_reg->pm_cur_stat,
>   				       val, (val == CS35L56_HALO_STATE_SHUTDOWN),
>   				       CS35L56_HALO_STATE_POLL_US,
>   				       CS35L56_HALO_STATE_TIMEOUT_US);
> @@ -278,7 +279,9 @@ int cs35l56_wait_for_firmware_boot(struct cs35l56_base *cs35l56_base)
>   				     CS35L56_HALO_STATE_POLL_US,
>   				     CS35L56_HALO_STATE_TIMEOUT_US,
>   				     false,
> -				     cs35l56_base->regmap, CS35L56_DSP1_HALO_STATE, &val);
> +				     cs35l56_base->regmap,
> +				     cs35l56_base->fw_reg->halo_state,
> +				     &val);
>   
>   	if (poll_ret) {
>   		dev_err(cs35l56_base->dev, "Firmware boot timed out(%d): HALO_STATE=%#x\n",
> @@ -395,9 +398,17 @@ void cs35l56_system_reset(struct cs35l56_base *cs35l56_base, bool is_soundwire)
>   		return;
>   	}
>   
> -	regmap_multi_reg_write_bypassed(cs35l56_base->regmap,
> -					cs35l56_system_reset_seq,
> -					ARRAY_SIZE(cs35l56_system_reset_seq));
> +	switch (cs35l56_base->type) {
> +	case 0x54:
> +	case 0x56:
> +	case 0x57:
> +		regmap_multi_reg_write_bypassed(cs35l56_base->regmap,
> +						cs35l56_system_reset_seq,
> +						ARRAY_SIZE(cs35l56_system_reset_seq));
> +		break;
> +	default:
> +		break;
> +	}
>   
>   	/* On SoundWire the registers won't be accessible until it re-enumerates. */
>   	if (is_soundwire)
> @@ -514,7 +525,9 @@ int cs35l56_is_fw_reload_needed(struct cs35l56_base *cs35l56_base)
>   		return ret;
>   	}
>   
> -	ret = regmap_read(cs35l56_base->regmap, CS35L56_PROTECTION_STATUS, &val);
> +	ret = regmap_read(cs35l56_base->regmap,
> +			  cs35l56_base->fw_reg->prot_sts,
> +			  &val);
>   	if (ret)
>   		dev_err(cs35l56_base->dev, "Failed to read PROTECTION_STATUS: %d\n", ret);
>   	else
> @@ -562,7 +575,7 @@ int cs35l56_runtime_suspend_common(struct cs35l56_base *cs35l56_base)
>   
>   	/* Firmware must have entered a power-save state */
>   	ret = regmap_read_poll_timeout(cs35l56_base->regmap,
> -				       CS35L56_TRANSDUCER_ACTUAL_PS,
> +				       cs35l56_base->fw_reg->transducer_actual_ps,
>   				       val, (val >= CS35L56_PS3),
>   				       CS35L56_PS3_POLL_US,
>   				       CS35L56_PS3_TIMEOUT_US);
> @@ -752,7 +765,8 @@ int cs35l56_read_prot_status(struct cs35l56_base *cs35l56_base,
>   	unsigned int prot_status;
>   	int ret;
>   
> -	ret = regmap_read(cs35l56_base->regmap, CS35L56_PROTECTION_STATUS, &prot_status);
> +	ret = regmap_read(cs35l56_base->regmap,
> +			  cs35l56_base->fw_reg->prot_sts, &prot_status);
>   	if (ret) {
>   		dev_err(cs35l56_base->dev, "Get PROTECTION_STATUS failed: %d\n", ret);
>   		return ret;
> @@ -760,7 +774,8 @@ int cs35l56_read_prot_status(struct cs35l56_base *cs35l56_base,
>   
>   	*fw_missing = !!(prot_status & CS35L56_FIRMWARE_MISSING);
>   
> -	ret = regmap_read(cs35l56_base->regmap, CS35L56_DSP1_FW_VER, fw_version);
> +	ret = regmap_read(cs35l56_base->regmap,
> +			  cs35l56_base->fw_reg->fw_ver, fw_version);
>   	if (ret) {
>   		dev_err(cs35l56_base->dev, "Get FW VER failed: %d\n", ret);
>   		return ret;
> @@ -1045,6 +1060,15 @@ const struct regmap_config cs35l56_regmap_sdw = {
>   };
>   EXPORT_SYMBOL_NS_GPL(cs35l56_regmap_sdw, "SND_SOC_CS35L56_SHARED");
>   
> +const struct cs35l56_fw_reg cs35l56_fw_reg = {
> +	.fw_ver = CS35L56_DSP1_FW_VER,
> +	.halo_state = CS35L56_DSP1_HALO_STATE,
> +	.pm_cur_stat = CS35L56_DSP1_PM_CUR_STATE,
> +	.prot_sts = CS35L56_PROTECTION_STATUS,
> +	.transducer_actual_ps = CS35L56_TRANSDUCER_ACTUAL_PS,
> +};
> +EXPORT_SYMBOL_NS_GPL(cs35l56_fw_reg, "SND_SOC_CS35L56_SHARED");
> +
>   MODULE_DESCRIPTION("ASoC CS35L56 Shared");
>   MODULE_AUTHOR("Richard Fitzgerald <rf@opensource.cirrus.com>");
>   MODULE_AUTHOR("Simon Trimmer <simont@opensource.cirrus.com>");
> diff --git a/sound/soc/codecs/cs35l56-spi.c b/sound/soc/codecs/cs35l56-spi.c
> index ca6c03a8766d..c2ddee22cd23 100644
> --- a/sound/soc/codecs/cs35l56-spi.c
> +++ b/sound/soc/codecs/cs35l56-spi.c
> @@ -25,6 +25,9 @@ static int cs35l56_spi_probe(struct spi_device *spi)
>   		return -ENOMEM;
>   
>   	spi_set_drvdata(spi, cs35l56);
> +
> +	cs35l56->base.fw_reg = &cs35l56_fw_reg;
> +
>   	cs35l56->base.regmap = devm_regmap_init_spi(spi, regmap_config);
>   	if (IS_ERR(cs35l56->base.regmap)) {
>   		ret = PTR_ERR(cs35l56->base.regmap);
> diff --git a/sound/soc/codecs/cs35l56.c b/sound/soc/codecs/cs35l56.c
> index b3158a84b87a..c1d8bfb803b9 100644
> --- a/sound/soc/codecs/cs35l56.c
> +++ b/sound/soc/codecs/cs35l56.c
> @@ -174,7 +174,7 @@ static int cs35l56_play_event(struct snd_soc_dapm_widget *w,
>   	case SND_SOC_DAPM_POST_PMU:
>   		/* Wait for firmware to enter PS0 power state */
>   		ret = regmap_read_poll_timeout(cs35l56->base.regmap,
> -					       CS35L56_TRANSDUCER_ACTUAL_PS,
> +					       cs35l56->base.fw_reg->transducer_actual_ps,
>   					       val, (val == CS35L56_PS0),
>   					       CS35L56_PS0_POLL_US,
>   					       CS35L56_PS0_TIMEOUT_US);
> @@ -760,7 +760,8 @@ static void cs35l56_patch(struct cs35l56_private *cs35l56, bool firmware_missing
>   		goto err_unlock;
>   	}
>   
> -	regmap_clear_bits(cs35l56->base.regmap, CS35L56_PROTECTION_STATUS,
> +	regmap_clear_bits(cs35l56->base.regmap,
> +			  cs35l56->base.fw_reg->prot_sts,
>   			  CS35L56_FIRMWARE_MISSING);
>   	cs35l56->base.fw_patched = true;
>   
Reviewed-by: Richard Fitzgerald <rf@opensource.cirrus.com>

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 3/5] ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware register list
  2025-04-07 15:16 ` [PATCH v1 3/5] ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware register list Stefan Binding
@ 2025-04-28 17:36   ` Richard Fitzgerald
  0 siblings, 0 replies; 18+ messages in thread
From: Richard Fitzgerald @ 2025-04-28 17:36 UTC (permalink / raw)
  To: Stefan Binding, Mark Brown; +Cc: linux-sound, linux-kernel, patches

On 7/4/25 16:16, Stefan Binding wrote:
> Registers to set Mute, Volume and Posture are inside firmware,
> which means they should be added to the list of registers set inside
> firmware, in case they vary across Device or Revision.
> 
> These three registers are also used for controls, so additional
> handling is required to be able to obtain and set the register inside
> ALSA controls.
> 
> Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
> ---
>   include/sound/cs35l56.h           |  3 +++
>   sound/pci/hda/cs35l56_hda.c       | 17 +++++++----------
>   sound/soc/codecs/cs35l56-shared.c | 25 ++++++++++++++++++++++++-
>   sound/soc/codecs/cs35l56.c        | 19 +++++++++++++++++--
>   4 files changed, 51 insertions(+), 13 deletions(-)
> 
> diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
> index d712cb79652b..3abe4fbd2085 100644
> --- a/include/sound/cs35l56.h
> +++ b/include/sound/cs35l56.h
> @@ -273,6 +273,9 @@ struct cs35l56_fw_reg {
>   	unsigned int pm_cur_stat;
>   	unsigned int prot_sts;
>   	unsigned int transducer_actual_ps;
> +	unsigned int user_mute;
> +	unsigned int user_volume;
> +	unsigned int posture_number;
>   };
>   
>   struct cs35l56_base {
> diff --git a/sound/pci/hda/cs35l56_hda.c b/sound/pci/hda/cs35l56_hda.c
> index f6257fe45b9f..61aba1956338 100644
> --- a/sound/pci/hda/cs35l56_hda.c
> +++ b/sound/pci/hda/cs35l56_hda.c
> @@ -237,7 +237,8 @@ static int cs35l56_hda_posture_get(struct snd_kcontrol *kcontrol,
>   
>   	cs35l56_hda_wait_dsp_ready(cs35l56);
>   
> -	ret = regmap_read(cs35l56->base.regmap, CS35L56_MAIN_POSTURE_NUMBER, &pos);
> +	ret = regmap_read(cs35l56->base.regmap,
> +			  cs35l56->base.fw_reg->posture_number, &pos);
>   	if (ret)
>   		return ret;
>   
> @@ -260,10 +261,8 @@ static int cs35l56_hda_posture_put(struct snd_kcontrol *kcontrol,
>   
>   	cs35l56_hda_wait_dsp_ready(cs35l56);
>   
> -	ret = regmap_update_bits_check(cs35l56->base.regmap,
> -				       CS35L56_MAIN_POSTURE_NUMBER,
> -				       CS35L56_MAIN_POSTURE_MASK,
> -				       pos, &changed);
> +	ret = regmap_update_bits_check(cs35l56->base.regmap, cs35l56->base.fw_reg->posture_number,
> +				       CS35L56_MAIN_POSTURE_MASK, pos, &changed);
>   	if (ret)
>   		return ret;
>   
> @@ -305,7 +304,7 @@ static int cs35l56_hda_vol_get(struct snd_kcontrol *kcontrol,
>   
>   	cs35l56_hda_wait_dsp_ready(cs35l56);
>   
> -	ret = regmap_read(cs35l56->base.regmap, CS35L56_MAIN_RENDER_USER_VOLUME, &raw_vol);
> +	ret = regmap_read(cs35l56->base.regmap, cs35l56->base.fw_reg->user_volume, &raw_vol);
>   
>   	if (ret)
>   		return ret;
> @@ -339,10 +338,8 @@ static int cs35l56_hda_vol_put(struct snd_kcontrol *kcontrol,
>   
>   	cs35l56_hda_wait_dsp_ready(cs35l56);
>   
> -	ret = regmap_update_bits_check(cs35l56->base.regmap,
> -				       CS35L56_MAIN_RENDER_USER_VOLUME,
> -				       CS35L56_MAIN_RENDER_USER_VOLUME_MASK,
> -				       raw_vol, &changed);
> +	ret = regmap_update_bits_check(cs35l56->base.regmap, cs35l56->base.fw_reg->user_volume,
> +				       CS35L56_MAIN_RENDER_USER_VOLUME_MASK, raw_vol, &changed);
>   	if (ret)
>   		return ret;
>   
> diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
> index bc8f9379bc74..da982774bb4d 100644
> --- a/sound/soc/codecs/cs35l56-shared.c
> +++ b/sound/soc/codecs/cs35l56-shared.c
> @@ -38,7 +38,9 @@ static const struct reg_sequence cs35l56_patch[] = {
>   	{ CS35L56_SWIRE_DP3_CH3_INPUT,		0x00000029 },
>   	{ CS35L56_SWIRE_DP3_CH4_INPUT,		0x00000028 },
>   	{ CS35L56_IRQ1_MASK_18,			0x1f7df0ff },
> +};
>   
> +static const struct reg_sequence cs35l56_patch_fw[] = {
>   	/* These are not reset by a soft-reset, so patch to defaults. */
>   	{ CS35L56_MAIN_RENDER_USER_MUTE,	0x00000000 },
>   	{ CS35L56_MAIN_RENDER_USER_VOLUME,	0x00000000 },
> @@ -47,8 +49,26 @@ static const struct reg_sequence cs35l56_patch[] = {
>   
>   int cs35l56_set_patch(struct cs35l56_base *cs35l56_base)
>   {
> -	return regmap_register_patch(cs35l56_base->regmap, cs35l56_patch,
> +	int ret;
> +
> +	ret = regmap_register_patch(cs35l56_base->regmap, cs35l56_patch,
>   				     ARRAY_SIZE(cs35l56_patch));
> +	if (ret)
> +		return ret;
> +
> +
> +	switch (cs35l56_base->type) {
> +	case 0x54:
> +	case 0x56:
> +	case 0x57:
> +		ret = regmap_register_patch(cs35l56_base->regmap, cs35l56_patch_fw,
> +					    ARRAY_SIZE(cs35l56_patch_fw));
> +		break;
> +	default:
> +		break;
> +	}
> +
> +	return ret;
>   }
>   EXPORT_SYMBOL_NS_GPL(cs35l56_set_patch, "SND_SOC_CS35L56_SHARED");
>   
> @@ -1066,6 +1086,9 @@ const struct cs35l56_fw_reg cs35l56_fw_reg = {
>   	.pm_cur_stat = CS35L56_DSP1_PM_CUR_STATE,
>   	.prot_sts = CS35L56_PROTECTION_STATUS,
>   	.transducer_actual_ps = CS35L56_TRANSDUCER_ACTUAL_PS,
> +	.user_mute = CS35L56_MAIN_RENDER_USER_MUTE,
> +	.user_volume = CS35L56_MAIN_RENDER_USER_VOLUME,
> +	.posture_number = CS35L56_MAIN_POSTURE_NUMBER,
>   };
>   EXPORT_SYMBOL_NS_GPL(cs35l56_fw_reg, "SND_SOC_CS35L56_SHARED");
>   
> diff --git a/sound/soc/codecs/cs35l56.c b/sound/soc/codecs/cs35l56.c
> index c1d8bfb803b9..a4a1d09097fc 100644
> --- a/sound/soc/codecs/cs35l56.c
> +++ b/sound/soc/codecs/cs35l56.c
> @@ -838,6 +838,7 @@ static int cs35l56_component_probe(struct snd_soc_component *component)
>   	struct cs35l56_private *cs35l56 = snd_soc_component_get_drvdata(component);
>   	struct dentry *debugfs_root = component->debugfs_root;
>   	unsigned short vendor, device;
> +	int ret;
>   
>   	BUILD_BUG_ON(ARRAY_SIZE(cs35l56_tx_input_texts) != ARRAY_SIZE(cs35l56_tx_input_values));
>   
> @@ -877,6 +878,22 @@ static int cs35l56_component_probe(struct snd_soc_component *component)
>   	debugfs_create_bool("can_hibernate", 0444, debugfs_root, &cs35l56->base.can_hibernate);
>   	debugfs_create_bool("fw_patched", 0444, debugfs_root, &cs35l56->base.fw_patched);
>   
> +
> +	switch (cs35l56->base.type) {
> +	case 0x54:
> +	case 0x56:
> +	case 0x57:
> +		ret = snd_soc_add_component_controls(component, cs35l56_controls,
> +						     ARRAY_SIZE(cs35l56_controls));
> +		break;
> +	default:
> +		ret = -ENODEV;
> +		break;
> +	}
> +
> +	if (ret)
> +		return dev_err_probe(cs35l56->base.dev, ret, "unable to add controls\n");
> +
>   	queue_work(cs35l56->dsp_wq, &cs35l56->dsp_work);
>   
>   	return 0;
> @@ -932,8 +949,6 @@ static const struct snd_soc_component_driver soc_component_dev_cs35l56 = {
>   	.num_dapm_widgets = ARRAY_SIZE(cs35l56_dapm_widgets),
>   	.dapm_routes = cs35l56_audio_map,
>   	.num_dapm_routes = ARRAY_SIZE(cs35l56_audio_map),
> -	.controls = cs35l56_controls,
> -	.num_controls = ARRAY_SIZE(cs35l56_controls),
>   
>   	.set_bias_level = cs35l56_set_bias_level,
>   
Reviewed-by: Richard Fitzgerald <rf@opensource.cirrus.com>

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 4/5] ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire
  2025-04-07 15:16 ` [PATCH v1 4/5] ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire Stefan Binding
@ 2025-04-28 17:38   ` Richard Fitzgerald
  0 siblings, 0 replies; 18+ messages in thread
From: Richard Fitzgerald @ 2025-04-28 17:38 UTC (permalink / raw)
  To: Stefan Binding, Mark Brown; +Cc: linux-sound, linux-kernel, patches

On 7/4/25 16:16, Stefan Binding wrote:
> CS35L63 uses a similar control interface to CS35L56 so support for
> it can be added into the CS35L56 driver.
> New regmap configs have been added to support CS35L63.
> CS35L63 only has SoundWire and I2C control interfaces.
> 
> Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
> ---
>   include/sound/cs35l56.h           |  12 +++
>   sound/soc/codecs/cs35l56-i2c.c    |   6 ++
>   sound/soc/codecs/cs35l56-sdw.c    |  74 ++++++++++++++++++
>   sound/soc/codecs/cs35l56-shared.c | 123 +++++++++++++++++++++++++++++-
>   sound/soc/codecs/cs35l56.c        |  23 ++++++
>   sound/soc/codecs/cs35l56.h        |   1 +
>   6 files changed, 237 insertions(+), 2 deletions(-)
> 
> diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
> index 3abe4fbd2085..e16e1a94c8a1 100644
> --- a/include/sound/cs35l56.h
> +++ b/include/sound/cs35l56.h
> @@ -104,6 +104,15 @@
>   #define CS35L56_DSP1_PMEM_0				0x3800000
>   #define CS35L56_DSP1_PMEM_5114				0x3804FE8
>   
> +#define CS35L63_DSP1_FW_VER				CS35L56_DSP1_FW_VER
> +#define CS35L63_DSP1_HALO_STATE				0x280396C
> +#define CS35L63_DSP1_PM_CUR_STATE			0x28042C8
> +#define CS35L63_PROTECTION_STATUS			0x340009C
> +#define CS35L63_TRANSDUCER_ACTUAL_PS			0x34000F4
> +#define CS35L63_MAIN_RENDER_USER_MUTE			0x3400020
> +#define CS35L63_MAIN_RENDER_USER_VOLUME			0x3400028
> +#define CS35L63_MAIN_POSTURE_NUMBER			0x3400068
> +
>   /* DEVID */
>   #define CS35L56_DEVID_MASK				0x00FFFFFF
>   
> @@ -322,8 +331,11 @@ static inline bool cs35l56_is_spi(struct cs35l56_base *cs35l56)
>   extern const struct regmap_config cs35l56_regmap_i2c;
>   extern const struct regmap_config cs35l56_regmap_spi;
>   extern const struct regmap_config cs35l56_regmap_sdw;
> +extern const struct regmap_config cs35l63_regmap_i2c;
> +extern const struct regmap_config cs35l63_regmap_sdw;
>   
>   extern const struct cs35l56_fw_reg cs35l56_fw_reg;
> +extern const struct cs35l56_fw_reg cs35l63_fw_reg;
>   
>   extern const struct cirrus_amp_cal_controls cs35l56_calibration_controls;
>   
> diff --git a/sound/soc/codecs/cs35l56-i2c.c b/sound/soc/codecs/cs35l56-i2c.c
> index 38c391d11c78..073f1796ae29 100644
> --- a/sound/soc/codecs/cs35l56-i2c.c
> +++ b/sound/soc/codecs/cs35l56-i2c.c
> @@ -37,6 +37,10 @@ static int cs35l56_i2c_probe(struct i2c_client *client)
>   		regmap_config = &cs35l56_regmap_i2c;
>   		cs35l56->base.fw_reg = &cs35l56_fw_reg;
>   		break;
> +	case 0x3563:
> +		regmap_config = &cs35l63_regmap_i2c;
> +		cs35l56->base.fw_reg = &cs35l63_fw_reg;
> +		break;
>   	default:
>   		return -ENODEV;
>   	}
> @@ -69,6 +73,7 @@ static void cs35l56_i2c_remove(struct i2c_client *client)
>   
>   static const struct i2c_device_id cs35l56_id_i2c[] = {
>   	{ "cs35l56", 0x3556 },
> +	{ "cs35l63", 0x3563 },
>   	{}
>   };
>   MODULE_DEVICE_TABLE(i2c, cs35l56_id_i2c);
> @@ -76,6 +81,7 @@ MODULE_DEVICE_TABLE(i2c, cs35l56_id_i2c);
>   #ifdef CONFIG_ACPI
>   static const struct acpi_device_id cs35l56_asoc_acpi_match[] = {
>   	{ "CSC355C", 0x3556 },
> +	{ "CSC356C", 0x3563 },
>   	{},
>   };
>   MODULE_DEVICE_TABLE(acpi, cs35l56_asoc_acpi_match);
> diff --git a/sound/soc/codecs/cs35l56-sdw.c b/sound/soc/codecs/cs35l56-sdw.c
> index 2e0422b41385..13f602f51bf3 100644
> --- a/sound/soc/codecs/cs35l56-sdw.c
> +++ b/sound/soc/codecs/cs35l56-sdw.c
> @@ -393,6 +393,74 @@ static int cs35l56_sdw_update_status(struct sdw_slave *peripheral,
>   	return 0;
>   }
>   
> +static int cs35l63_sdw_kick_divider(struct cs35l56_private *cs35l56,
> +				    struct sdw_slave *peripheral)
> +{
> +	unsigned int curr_scale_reg, next_scale_reg;
> +	int curr_scale, next_scale, ret;
> +
> +	if (!cs35l56->base.init_done)
> +		return 0;
> +
> +	if (peripheral->bus->params.curr_bank) {
> +		curr_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B1;
> +		next_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B0;
> +	} else {
> +		curr_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B0;
> +		next_scale_reg = SDW_SCP_BUSCLOCK_SCALE_B1;
> +	}
> +
> +	/*
> +	 * Current clock scale value must be different to new value.
> +	 * Modify current to guarantee this. If next still has the dummy
> +	 * value we wrote when it was current, the core code has not set
> +	 * a new scale so restore its original good value
> +	 */
> +	curr_scale = sdw_read_no_pm(peripheral, curr_scale_reg);
> +	if (curr_scale < 0) {
> +		dev_err(cs35l56->base.dev, "Failed to read current clock scale: %d\n", curr_scale);
> +		return curr_scale;
> +	}
> +
> +	next_scale = sdw_read_no_pm(peripheral, next_scale_reg);
> +	if (next_scale < 0) {
> +		dev_err(cs35l56->base.dev, "Failed to read next clock scale: %d\n", next_scale);
> +		return next_scale;
> +	}
> +
> +	if (next_scale == CS35L56_SDW_INVALID_BUS_SCALE) {
> +		next_scale = cs35l56->old_sdw_clock_scale;
> +		ret = sdw_write_no_pm(peripheral, next_scale_reg, next_scale);
> +		if (ret < 0) {
> +			dev_err(cs35l56->base.dev, "Failed to modify current clock scale: %d\n",
> +				ret);
> +			return ret;
> +		}
> +	}
> +
> +	cs35l56->old_sdw_clock_scale = curr_scale;
> +	ret = sdw_write_no_pm(peripheral, curr_scale_reg, CS35L56_SDW_INVALID_BUS_SCALE);
> +	if (ret < 0) {
> +		dev_err(cs35l56->base.dev, "Failed to modify current clock scale: %d\n", ret);
> +		return ret;
> +	}
> +
> +	dev_dbg(cs35l56->base.dev, "Next bus scale: %#x\n", next_scale);
> +
> +	return 0;
> +}
> +
> +static int cs35l56_sdw_bus_config(struct sdw_slave *peripheral,
> +				  struct sdw_bus_params *params)
> +{
> +	struct cs35l56_private *cs35l56 = dev_get_drvdata(&peripheral->dev);
> +
> +	if ((cs35l56->base.type == 0x63) && (cs35l56->base.rev < 0xa1))
> +		return cs35l63_sdw_kick_divider(cs35l56, peripheral);
> +
> +	return 0;
> +}
> +
>   static int __maybe_unused cs35l56_sdw_clk_stop(struct sdw_slave *peripheral,
>   					       enum sdw_clk_stop_mode mode,
>   					       enum sdw_clk_stop_type type)
> @@ -408,6 +476,7 @@ static const struct sdw_slave_ops cs35l56_sdw_ops = {
>   	.read_prop = cs35l56_sdw_read_prop,
>   	.interrupt_callback = cs35l56_sdw_interrupt,
>   	.update_status = cs35l56_sdw_update_status,
> +	.bus_config = cs35l56_sdw_bus_config,
>   #ifdef DEBUG
>   	.clk_stop = cs35l56_sdw_clk_stop,
>   #endif
> @@ -528,6 +597,10 @@ static int cs35l56_sdw_probe(struct sdw_slave *peripheral, const struct sdw_devi
>   		regmap_config = &cs35l56_regmap_sdw;
>   		cs35l56->base.fw_reg = &cs35l56_fw_reg;
>   		break;
> +	case 0x3563:
> +		regmap_config = &cs35l63_regmap_sdw;
> +		cs35l56->base.fw_reg = &cs35l63_fw_reg;
> +		break;
>   	default:
>   		return -ENODEV;
>   	}
> @@ -575,6 +648,7 @@ static const struct dev_pm_ops cs35l56_sdw_pm = {
>   static const struct sdw_device_id cs35l56_sdw_id[] = {
>   	SDW_SLAVE_ENTRY(0x01FA, 0x3556, 0x3556),
>   	SDW_SLAVE_ENTRY(0x01FA, 0x3557, 0x3557),
> +	SDW_SLAVE_ENTRY(0x01FA, 0x3563, 0x3563),
>   	{},
>   };
>   MODULE_DEVICE_TABLE(sdw, cs35l56_sdw_id);
> diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
> index da982774bb4d..76ddb1cf6889 100644
> --- a/sound/soc/codecs/cs35l56-shared.c
> +++ b/sound/soc/codecs/cs35l56-shared.c
> @@ -47,6 +47,13 @@ static const struct reg_sequence cs35l56_patch_fw[] = {
>   	{ CS35L56_MAIN_POSTURE_NUMBER,		0x00000000 },
>   };
>   
> +static const struct reg_sequence cs35l63_patch_fw[] = {
> +	/* These are not reset by a soft-reset, so patch to defaults. */
> +	{ CS35L63_MAIN_RENDER_USER_MUTE,	0x00000000 },
> +	{ CS35L63_MAIN_RENDER_USER_VOLUME,	0x00000000 },
> +	{ CS35L63_MAIN_POSTURE_NUMBER,		0x00000000 },
> +};
> +
>   int cs35l56_set_patch(struct cs35l56_base *cs35l56_base)
>   {
>   	int ret;
> @@ -64,6 +71,10 @@ int cs35l56_set_patch(struct cs35l56_base *cs35l56_base)
>   		ret = regmap_register_patch(cs35l56_base->regmap, cs35l56_patch_fw,
>   					    ARRAY_SIZE(cs35l56_patch_fw));
>   		break;
> +	case 0x63:
> +		ret = regmap_register_patch(cs35l56_base->regmap, cs35l63_patch_fw,
> +					    ARRAY_SIZE(cs35l63_patch_fw));
> +		break;
>   	default:
>   		break;
>   	}
> @@ -102,6 +113,36 @@ static const struct reg_default cs35l56_reg_defaults[] = {
>   	{ CS35L56_MAIN_POSTURE_NUMBER,		0x00000000 },
>   };
>   
> +static const struct reg_default cs35l63_reg_defaults[] = {
> +	/* no defaults for OTP_MEM - first read populates cache */
> +
> +	{ CS35L56_ASP1_ENABLES1,		0x00000000 },
> +	{ CS35L56_ASP1_CONTROL1,		0x00000028 },
> +	{ CS35L56_ASP1_CONTROL2,		0x18180200 },
> +	{ CS35L56_ASP1_CONTROL3,		0x00000002 },
> +	{ CS35L56_ASP1_FRAME_CONTROL1,		0x03020100 },
> +	{ CS35L56_ASP1_FRAME_CONTROL5,		0x00020100 },
> +	{ CS35L56_ASP1_DATA_CONTROL1,		0x00000018 },
> +	{ CS35L56_ASP1_DATA_CONTROL5,		0x00000018 },
> +	{ CS35L56_ASP1TX1_INPUT,		0x00000000 },
> +	{ CS35L56_ASP1TX2_INPUT,		0x00000000 },
> +	{ CS35L56_ASP1TX3_INPUT,		0x00000000 },
> +	{ CS35L56_ASP1TX4_INPUT,		0x00000000 },
> +	{ CS35L56_SWIRE_DP3_CH1_INPUT,		0x00000018 },
> +	{ CS35L56_SWIRE_DP3_CH2_INPUT,		0x00000019 },
> +	{ CS35L56_SWIRE_DP3_CH3_INPUT,		0x00000029 },
> +	{ CS35L56_SWIRE_DP3_CH4_INPUT,		0x00000028 },
> +	{ CS35L56_IRQ1_MASK_1,			0x8003ffff },
> +	{ CS35L56_IRQ1_MASK_2,			0xffff7fff },
> +	{ CS35L56_IRQ1_MASK_4,			0xe0ffffff },
> +	{ CS35L56_IRQ1_MASK_8,			0x8c000fff },
> +	{ CS35L56_IRQ1_MASK_18,			0x0760f000 },
> +	{ CS35L56_IRQ1_MASK_20,			0x15c00000 },
> +	{ CS35L63_MAIN_RENDER_USER_MUTE,	0x00000000 },
> +	{ CS35L63_MAIN_RENDER_USER_VOLUME,	0x00000000 },
> +	{ CS35L63_MAIN_POSTURE_NUMBER,		0x00000000 },
> +};
> +
>   static bool cs35l56_is_dsp_memory(unsigned int reg)
>   {
>   	switch (reg) {
> @@ -199,7 +240,7 @@ static bool cs35l56_precious_reg(struct device *dev, unsigned int reg)
>   	}
>   }
>   
> -static bool cs35l56_volatile_reg(struct device *dev, unsigned int reg)
> +static bool cs35l56_common_volatile_reg(unsigned int reg)
>   {
>   	switch (reg) {
>   	case CS35L56_DEVID:
> @@ -237,12 +278,32 @@ static bool cs35l56_volatile_reg(struct device *dev, unsigned int reg)
>   	case CS35L56_DSP1_SCRATCH3:
>   	case CS35L56_DSP1_SCRATCH4:
>   		return true;
> +	default:
> +		return cs35l56_is_dsp_memory(reg);
> +	}
> +}
> +
> +static bool cs35l56_volatile_reg(struct device *dev, unsigned int reg)
> +{
> +	switch (reg) {
>   	case CS35L56_MAIN_RENDER_USER_MUTE:
>   	case CS35L56_MAIN_RENDER_USER_VOLUME:
>   	case CS35L56_MAIN_POSTURE_NUMBER:
>   		return false;
>   	default:
> -		return cs35l56_is_dsp_memory(reg);
> +		return cs35l56_common_volatile_reg(reg);
> +	}
> +}
> +
> +static bool cs35l63_volatile_reg(struct device *dev, unsigned int reg)
> +{
> +	switch (reg) {
> +	case CS35L63_MAIN_RENDER_USER_MUTE:
> +	case CS35L63_MAIN_RENDER_USER_VOLUME:
> +	case CS35L63_MAIN_POSTURE_NUMBER:
> +		return false;
> +	default:
> +		return cs35l56_common_volatile_reg(reg);
>   	}
>   }
>   
> @@ -405,6 +466,11 @@ static const struct reg_sequence cs35l56_system_reset_seq[] = {
>   	REG_SEQ0(CS35L56_DSP_VIRTUAL1_MBOX_1, CS35L56_MBOX_CMD_SYSTEM_RESET),
>   };
>   
> +static const struct reg_sequence cs35l63_system_reset_seq[] = {
> +	REG_SEQ0(CS35L63_DSP1_HALO_STATE, 0),
> +	REG_SEQ0(CS35L56_DSP_VIRTUAL1_MBOX_1, CS35L56_MBOX_CMD_SYSTEM_RESET),
> +};
> +
>   void cs35l56_system_reset(struct cs35l56_base *cs35l56_base, bool is_soundwire)
>   {
>   	/*
> @@ -426,6 +492,11 @@ void cs35l56_system_reset(struct cs35l56_base *cs35l56_base, bool is_soundwire)
>   						cs35l56_system_reset_seq,
>   						ARRAY_SIZE(cs35l56_system_reset_seq));
>   		break;
> +	case 0x63:
> +		regmap_multi_reg_write_bypassed(cs35l56_base->regmap,
> +						cs35l63_system_reset_seq,
> +						ARRAY_SIZE(cs35l63_system_reset_seq));
> +		break;
>   	default:
>   		break;
>   	}
> @@ -844,6 +915,9 @@ int cs35l56_hw_init(struct cs35l56_base *cs35l56_base)
>   	case 0x35A56:
>   	case 0x35A57:
>   		break;
> +	case 0x35A630:
> +		devid = devid >> 4;
> +		break;
>   	default:
>   		dev_err(cs35l56_base->dev, "Unknown device %x\n", devid);
>   		return ret;
> @@ -1080,6 +1154,39 @@ const struct regmap_config cs35l56_regmap_sdw = {
>   };
>   EXPORT_SYMBOL_NS_GPL(cs35l56_regmap_sdw, "SND_SOC_CS35L56_SHARED");
>   
> +const struct regmap_config cs35l63_regmap_i2c = {
> +	.reg_bits = 32,
> +	.val_bits = 32,
> +	.reg_stride = 4,
> +	.reg_base = 0x8000,
> +	.reg_format_endian = REGMAP_ENDIAN_BIG,
> +	.val_format_endian = REGMAP_ENDIAN_BIG,
> +	.max_register = CS35L56_DSP1_PMEM_5114,
> +	.reg_defaults = cs35l63_reg_defaults,
> +	.num_reg_defaults = ARRAY_SIZE(cs35l63_reg_defaults),
> +	.volatile_reg = cs35l63_volatile_reg,
> +	.readable_reg = cs35l56_readable_reg,
> +	.precious_reg = cs35l56_precious_reg,
> +	.cache_type = REGCACHE_MAPLE,
> +};
> +EXPORT_SYMBOL_NS_GPL(cs35l63_regmap_i2c, "SND_SOC_CS35L56_SHARED");
> +
> +const struct regmap_config cs35l63_regmap_sdw = {
> +	.reg_bits = 32,
> +	.val_bits = 32,
> +	.reg_stride = 4,
> +	.reg_format_endian = REGMAP_ENDIAN_LITTLE,
> +	.val_format_endian = REGMAP_ENDIAN_BIG,
> +	.max_register = CS35L56_DSP1_PMEM_5114,
> +	.reg_defaults = cs35l63_reg_defaults,
> +	.num_reg_defaults = ARRAY_SIZE(cs35l63_reg_defaults),
> +	.volatile_reg = cs35l63_volatile_reg,
> +	.readable_reg = cs35l56_readable_reg,
> +	.precious_reg = cs35l56_precious_reg,
> +	.cache_type = REGCACHE_MAPLE,
> +};
> +EXPORT_SYMBOL_NS_GPL(cs35l63_regmap_sdw, "SND_SOC_CS35L56_SHARED");
> +
>   const struct cs35l56_fw_reg cs35l56_fw_reg = {
>   	.fw_ver = CS35L56_DSP1_FW_VER,
>   	.halo_state = CS35L56_DSP1_HALO_STATE,
> @@ -1092,6 +1199,18 @@ const struct cs35l56_fw_reg cs35l56_fw_reg = {
>   };
>   EXPORT_SYMBOL_NS_GPL(cs35l56_fw_reg, "SND_SOC_CS35L56_SHARED");
>   
> +const struct cs35l56_fw_reg cs35l63_fw_reg = {
> +	.fw_ver = CS35L63_DSP1_FW_VER,
> +	.halo_state = CS35L63_DSP1_HALO_STATE,
> +	.pm_cur_stat = CS35L63_DSP1_PM_CUR_STATE,
> +	.prot_sts = CS35L63_PROTECTION_STATUS,
> +	.transducer_actual_ps = CS35L63_TRANSDUCER_ACTUAL_PS,
> +	.user_mute = CS35L63_MAIN_RENDER_USER_MUTE,
> +	.user_volume = CS35L63_MAIN_RENDER_USER_VOLUME,
> +	.posture_number = CS35L63_MAIN_POSTURE_NUMBER,
> +};
> +EXPORT_SYMBOL_NS_GPL(cs35l63_fw_reg, "SND_SOC_CS35L56_SHARED");
> +
>   MODULE_DESCRIPTION("ASoC CS35L56 Shared");
>   MODULE_AUTHOR("Richard Fitzgerald <rf@opensource.cirrus.com>");
>   MODULE_AUTHOR("Simon Trimmer <simont@opensource.cirrus.com>");
> diff --git a/sound/soc/codecs/cs35l56.c b/sound/soc/codecs/cs35l56.c
> index a4a1d09097fc..cdb283ed938c 100644
> --- a/sound/soc/codecs/cs35l56.c
> +++ b/sound/soc/codecs/cs35l56.c
> @@ -84,6 +84,25 @@ static const struct snd_kcontrol_new cs35l56_controls[] = {
>   		       cs35l56_dspwait_get_volsw, cs35l56_dspwait_put_volsw),
>   };
>   
> +static const struct snd_kcontrol_new cs35l63_controls[] = {
> +	SOC_SINGLE_EXT("Speaker Switch",
> +		       CS35L63_MAIN_RENDER_USER_MUTE, 0, 1, 1,
> +		       cs35l56_dspwait_get_volsw, cs35l56_dspwait_put_volsw),
> +	SOC_SINGLE_S_EXT_TLV("Speaker Volume",
> +			     CS35L63_MAIN_RENDER_USER_VOLUME,
> +			     CS35L56_MAIN_RENDER_USER_VOLUME_SHIFT,
> +			     CS35L56_MAIN_RENDER_USER_VOLUME_MIN,
> +			     CS35L56_MAIN_RENDER_USER_VOLUME_MAX,
> +			     CS35L56_MAIN_RENDER_USER_VOLUME_SIGNBIT,
> +			     0,
> +			     cs35l56_dspwait_get_volsw,
> +			     cs35l56_dspwait_put_volsw,
> +			     vol_tlv),
> +	SOC_SINGLE_EXT("Posture Number", CS35L63_MAIN_POSTURE_NUMBER,
> +		       0, 255, 0,
> +		       cs35l56_dspwait_get_volsw, cs35l56_dspwait_put_volsw),
> +};
> +
>   static SOC_VALUE_ENUM_SINGLE_DECL(cs35l56_asp1tx1_enum,
>   				  CS35L56_ASP1TX1_INPUT,
>   				  0, CS35L56_ASP_TXn_SRC_MASK,
> @@ -886,6 +905,10 @@ static int cs35l56_component_probe(struct snd_soc_component *component)
>   		ret = snd_soc_add_component_controls(component, cs35l56_controls,
>   						     ARRAY_SIZE(cs35l56_controls));
>   		break;
> +	case 0x63:
> +		ret = snd_soc_add_component_controls(component, cs35l63_controls,
> +						     ARRAY_SIZE(cs35l63_controls));
> +		break;
>   	default:
>   		ret = -ENODEV;
>   		break;
> diff --git a/sound/soc/codecs/cs35l56.h b/sound/soc/codecs/cs35l56.h
> index 8a987ec01507..200f695efca3 100644
> --- a/sound/soc/codecs/cs35l56.h
> +++ b/sound/soc/codecs/cs35l56.h
> @@ -51,6 +51,7 @@ struct cs35l56_private {
>   	u8 asp_slot_count;
>   	bool tdm_mode;
>   	bool sysclk_set;
> +	u8 old_sdw_clock_scale;
>   };
>   
>   extern const struct dev_pm_ops cs35l56_pm_ops_i2c_spi;
Reviewed-by: Richard Fitzgerald <rf@opensource.cirrus.com>

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 5/5] ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63
  2025-04-07 15:16 ` [PATCH v1 5/5] ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63 Stefan Binding
@ 2025-04-28 17:39   ` Richard Fitzgerald
  0 siblings, 0 replies; 18+ messages in thread
From: Richard Fitzgerald @ 2025-04-28 17:39 UTC (permalink / raw)
  To: Stefan Binding, Mark Brown; +Cc: linux-sound, linux-kernel, patches

On 7/4/25 16:16, Stefan Binding wrote:
> On CS35L63 the DIE_STS registers are populated by the Firmware from
> OTP, so the driver can read these registers directly, rather than
> obtaining them from OTP.
> 
> Signed-off-by: Stefan Binding <sbinding@opensource.cirrus.com>
> ---
>   include/sound/cs35l56.h           |  2 ++
>   sound/soc/codecs/cs35l56-shared.c | 39 ++++++++++++++++++++++++++++---
>   2 files changed, 38 insertions(+), 3 deletions(-)
> 
> diff --git a/include/sound/cs35l56.h b/include/sound/cs35l56.h
> index e16e1a94c8a1..63f2c63f7c59 100644
> --- a/include/sound/cs35l56.h
> +++ b/include/sound/cs35l56.h
> @@ -71,6 +71,8 @@
>   #define CS35L56_DSP_VIRTUAL1_MBOX_6			0x0011034
>   #define CS35L56_DSP_VIRTUAL1_MBOX_7			0x0011038
>   #define CS35L56_DSP_VIRTUAL1_MBOX_8			0x001103C
> +#define CS35L56_DIE_STS1				0x0017040
> +#define CS35L56_DIE_STS2				0x0017044
>   #define CS35L56_DSP_RESTRICT_STS1			0x00190F0
>   #define CS35L56_DSP1_XMEM_PACKED_0			0x2000000
>   #define CS35L56_DSP1_XMEM_PACKED_6143			0x2005FFC
> diff --git a/sound/soc/codecs/cs35l56-shared.c b/sound/soc/codecs/cs35l56-shared.c
> index 76ddb1cf6889..7f768718b69b 100644
> --- a/sound/soc/codecs/cs35l56-shared.c
> +++ b/sound/soc/codecs/cs35l56-shared.c
> @@ -214,6 +214,8 @@ static bool cs35l56_readable_reg(struct device *dev, unsigned int reg)
>   	case CS35L56_DSP_VIRTUAL1_MBOX_6:
>   	case CS35L56_DSP_VIRTUAL1_MBOX_7:
>   	case CS35L56_DSP_VIRTUAL1_MBOX_8:
> +	case CS35L56_DIE_STS1:
> +	case CS35L56_DIE_STS2:
>   	case CS35L56_DSP_RESTRICT_STS1:
>   	case CS35L56_DSP1_SYS_INFO_ID ... CS35L56_DSP1_SYS_INFO_END:
>   	case CS35L56_DSP1_AHBM_WINDOW_DEBUG_0:
> @@ -802,13 +804,29 @@ static int cs35l56_read_silicon_uid(struct cs35l56_base *cs35l56_base, u64 *uid)
>   	unique_id |= (u32)pte.x | ((u32)pte.y << 8) | ((u32)pte.wafer_id << 16) |
>   		     ((u32)pte.dvs << 24);
>   
> -	dev_dbg(cs35l56_base->dev, "UniqueID = %#llx\n", unique_id);
> -
>   	*uid = unique_id;
>   
>   	return 0;
>   }
>   
> +static int cs35l63_read_silicon_uid(struct cs35l56_base *cs35l56_base, u64 *uid)
> +{
> +	u32 tmp[2];
> +	int ret;
> +
> +	ret = regmap_bulk_read(cs35l56_base->regmap, CS35L56_DIE_STS1, tmp, ARRAY_SIZE(tmp));
> +	if (ret) {
> +		dev_err(cs35l56_base->dev, "Cannot obtain CS35L56_DIE_STS: %d\n", ret);
> +		return ret;
> +	}
> +
> +	*uid = tmp[1];
> +	*uid <<= 32;
> +	*uid |= tmp[0];
> +
> +	return 0;
> +}
> +
>   /* Firmware calibration controls */
>   const struct cirrus_amp_cal_controls cs35l56_calibration_controls = {
>   	.alg_id =	0x9f210,
> @@ -829,10 +847,25 @@ int cs35l56_get_calibration(struct cs35l56_base *cs35l56_base)
>   	if (cs35l56_base->secured)
>   		return 0;
>   
> -	ret = cs35l56_read_silicon_uid(cs35l56_base, &silicon_uid);
> +	switch (cs35l56_base->type) {
> +	case 0x54:
> +	case 0x56:
> +	case 0x57:
> +		ret = cs35l56_read_silicon_uid(cs35l56_base, &silicon_uid);
> +		break;
> +	case 0x63:
> +		ret = cs35l63_read_silicon_uid(cs35l56_base, &silicon_uid);
> +		break;
> +	default:
> +		ret = -ENODEV;
> +		break;
> +	}
> +
>   	if (ret < 0)
>   		return ret;
>   
> +	dev_dbg(cs35l56_base->dev, "UniqueID = %#llx\n", silicon_uid);
> +
>   	ret = cs_amp_get_efi_calibration_data(cs35l56_base->dev, silicon_uid,
>   					      cs35l56_base->cal_index,
>   					      &cs35l56_base->cal_data);
Reviewed-by: Richard Fitzgerald <rf@opensource.cirrus.com>

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers
  2025-04-28 17:34   ` Richard Fitzgerald
@ 2025-05-05 23:46     ` Mark Brown
  0 siblings, 0 replies; 18+ messages in thread
From: Mark Brown @ 2025-05-05 23:46 UTC (permalink / raw)
  To: Richard Fitzgerald; +Cc: Stefan Binding, linux-sound, linux-kernel, patches

[-- Attachment #1: Type: text/plain, Size: 495 bytes --]

On Mon, Apr 28, 2025 at 06:34:31PM +0100, Richard Fitzgerald wrote:
> On 7/4/25 16:16, Stefan Binding wrote:
> > Firmware based registers may be different addresses across different
> > device ids and revision ids. Create a structure to store and access
> > these addresses.

Please delete unneeded context from mails when replying.  Doing this
makes it much easier to find your reply in the message, helping ensure
it won't be missed by people scrolling through the irrelevant quoted
material.

[-- Attachment #2: signature.asc --]
[-- Type: application/pgp-signature, Size: 488 bytes --]

^ permalink raw reply	[flat|nested] 18+ messages in thread

* Re: [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier
  2025-04-07 15:16 [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Stefan Binding
                   ` (4 preceding siblings ...)
  2025-04-07 15:16 ` [PATCH v1 5/5] ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63 Stefan Binding
@ 2025-05-07 11:12 ` Mark Brown
  5 siblings, 0 replies; 18+ messages in thread
From: Mark Brown @ 2025-05-07 11:12 UTC (permalink / raw)
  To: Stefan Binding; +Cc: linux-sound, linux-kernel, patches

On Mon, 07 Apr 2025 16:16:39 +0100, Stefan Binding wrote:
> CS35L63 is a Mono Class-D PC Smart Amplifier, with Speaker Protection
> and Audio Enhancement Algorithms.
> 
> CS35L63 uses a similar control interface to CS35L56 so support for
> it can be added into the CS35L56 driver.
> CS35L63 only has SoundWire and I2C control interfaces.
> 
> [...]

Applied to

   https://git.kernel.org/pub/scm/linux/kernel/git/broonie/sound.git for-next

Thanks!

[1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config
      commit: 9ea280f3df60df014527ba5723fd8d1fd7c3b3ca
[2/5] ASoC: cs35l56: Add struct to index firmware registers
      commit: 8c0821ccaa1e53c30f6a51d034171e20d522a9c0
[3/5] ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware register list
      commit: a3ec669efcee2cffd8ca020df777adbf79ac0b75
[4/5] ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire
      commit: 978858791cedaf1dbbbaeb1e37126611b7c28a9e
[5/5] ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63
      commit: 406fbc4d0fb34c16718551bb8f4c776710f63b55

All being well this means that it will be integrated into the linux-next
tree (usually sometime in the next 24 hours) and sent to Linus during
the next merge window (or sooner if it is a bug fix), however if
problems are discovered then the patch may be dropped or reverted.

You may get further e-mails resulting from automated or manual testing
and review of the tree, please engage with people reporting problems and
send followup patches addressing any issues that are reported if needed.

If any updates are required or you are submitting further changes they
should be sent as incremental updates against current git, existing
patches will not be replaced.

Please add any relevant lists and maintainers to the CCs when replying
to this mail.

Thanks,
Mark


^ permalink raw reply	[flat|nested] 18+ messages in thread

end of thread, other threads:[~2025-05-07 11:13 UTC | newest]

Thread overview: 18+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2025-04-07 15:16 [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Stefan Binding
2025-04-07 15:16 ` [PATCH v1 1/5] ASoC: cs35l56: Add Index based on ACPI HID or SDW ID to select regmap config Stefan Binding
2025-04-28 17:32   ` Richard Fitzgerald
2025-04-07 15:16 ` [PATCH v1 2/5] ASoC: cs35l56: Add struct to index firmware registers Stefan Binding
2025-04-07 19:16   ` Mark Brown
2025-04-08 12:58     ` Richard Fitzgerald
2025-04-08 16:00       ` Mark Brown
2025-04-09 11:33         ` Richard Fitzgerald
2025-04-09 12:31           ` Mark Brown
2025-04-28 17:34   ` Richard Fitzgerald
2025-05-05 23:46     ` Mark Brown
2025-04-07 15:16 ` [PATCH v1 3/5] ASoC: cs35l56: Add Mute, Volume and Posture registers to firmware register list Stefan Binding
2025-04-28 17:36   ` Richard Fitzgerald
2025-04-07 15:16 ` [PATCH v1 4/5] ASoC: cs35l56: Add initial support for CS35L63 for I2C and SoundWire Stefan Binding
2025-04-28 17:38   ` Richard Fitzgerald
2025-04-07 15:16 ` [PATCH v1 5/5] ASoC: cs35l56: Read Silicon ID from DIE_STS registers for CS35L63 Stefan Binding
2025-04-28 17:39   ` Richard Fitzgerald
2025-05-07 11:12 ` [PATCH v1 0/5] Add support for CS35L63 Smart Amplifier Mark Brown

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox