]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
ASoC: SDCA: Factor out jack handling into new c file
authorCharles Keepax <ckeepax@opensource.cirrus.com>
Mon, 15 Dec 2025 15:36:47 +0000 (15:36 +0000)
committerMark Brown <broonie@kernel.org>
Mon, 22 Dec 2025 09:00:56 +0000 (09:00 +0000)
The jack code is perhaps a bit large for being in the interrupt
code directly. Improve the encapsulation by factoring out the
jack handling code into a new c file, as is already done for HID
and FDL. Whilst doing so also add a jack_state structure to hold
the jack state for improved expandability in the future.

Signed-off-by: Charles Keepax <ckeepax@opensource.cirrus.com>
Link: https://patch.msgid.link/20251215153650.3913117-2-ckeepax@opensource.cirrus.com
Reviewed-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Mark Brown <broonie@kernel.org>
include/sound/sdca_jack.h [new file with mode: 0644]
sound/soc/sdca/Makefile
sound/soc/sdca/sdca_interrupts.c
sound/soc/sdca/sdca_jack.c [new file with mode: 0644]

diff --git a/include/sound/sdca_jack.h b/include/sound/sdca_jack.h
new file mode 100644 (file)
index 0000000..9fad5f2
--- /dev/null
@@ -0,0 +1,27 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+/*
+ * The MIPI SDCA specification is available for public downloads at
+ * https://www.mipi.org/mipi-sdca-v1-0-download
+ *
+ * Copyright (C) 2025 Cirrus Logic, Inc. and
+ *                    Cirrus Logic International Semiconductor Ltd.
+ */
+
+#ifndef __SDCA_JACK_H__
+#define __SDCA_JACK_H__
+
+struct sdca_interrupt;
+struct snd_kcontrol;
+
+/**
+ * struct jack_state - Jack state structure to keep data between interrupts
+ * @kctl: Pointer to the ALSA control attached to this jack
+ */
+struct jack_state {
+       struct snd_kcontrol *kctl;
+};
+
+int sdca_jack_alloc_state(struct sdca_interrupt *interrupt);
+int sdca_jack_process(struct sdca_interrupt *interrupt);
+
+#endif // __SDCA_JACK_H__
index f6b73275d96493a25817800fbd46f121c873a5c6..b3b0f5d94c8de81751333611018ee052e4ed17da 100644 (file)
@@ -3,7 +3,7 @@
 snd-soc-sdca-y := sdca_functions.o sdca_device.o sdca_function_device.o \
                  sdca_regmap.o sdca_asoc.o sdca_ump.o
 snd-soc-sdca-$(CONFIG_SND_SOC_SDCA_HID) += sdca_hid.o
-snd-soc-sdca-$(CONFIG_SND_SOC_SDCA_IRQ) += sdca_interrupts.o
+snd-soc-sdca-$(CONFIG_SND_SOC_SDCA_IRQ) += sdca_interrupts.o sdca_jack.o
 snd-soc-sdca-$(CONFIG_SND_SOC_SDCA_FDL) += sdca_fdl.o
 
 snd-soc-sdca-class-y := sdca_class.o
index 8f6a2adfb6fbea9489f308961215e672ac6ab163..ff3a7e405fdcb5a5abc78973c890bd2098cb4d31 100644 (file)
@@ -22,6 +22,7 @@
 #include <sound/sdca_function.h>
 #include <sound/sdca_hid.h>
 #include <sound/sdca_interrupts.h>
+#include <sound/sdca_jack.h>
 #include <sound/sdca_ump.h>
 #include <sound/soc-component.h>
 #include <sound/soc.h>
@@ -155,14 +156,7 @@ static irqreturn_t detected_mode_handler(int irq, void *data)
 {
        struct sdca_interrupt *interrupt = data;
        struct device *dev = interrupt->dev;
-       struct snd_soc_component *component = interrupt->component;
-       struct snd_soc_card *card = component->card;
-       struct rw_semaphore *rwsem = &card->snd_card->controls_rwsem;
-       struct snd_kcontrol *kctl = interrupt->priv;
-       struct snd_ctl_elem_value *ucontrol __free(kfree) = NULL;
-       struct soc_enum *soc_enum;
        irqreturn_t irqret = IRQ_NONE;
-       unsigned int reg, val;
        int ret;
 
        ret = pm_runtime_get_sync(dev);
@@ -171,76 +165,9 @@ static irqreturn_t detected_mode_handler(int irq, void *data)
                goto error;
        }
 
-       if (!kctl) {
-               const char *name __free(kfree) = kasprintf(GFP_KERNEL, "%s %s",
-                                                          interrupt->entity->label,
-                                                          SDCA_CTL_SELECTED_MODE_NAME);
-
-               if (!name)
-                       goto error;
-
-               kctl = snd_soc_component_get_kcontrol(component, name);
-               if (!kctl) {
-                       dev_dbg(dev, "control not found: %s\n", name);
-                       goto error;
-               }
-
-               interrupt->priv = kctl;
-       }
-
-       soc_enum = (struct soc_enum *)kctl->private_value;
-
-       reg = SDW_SDCA_CTL(interrupt->function->desc->adr, interrupt->entity->id,
-                          interrupt->control->sel, 0);
-
-       ret = regmap_read(interrupt->function_regmap, reg, &val);
-       if (ret < 0) {
-               dev_err(dev, "failed to read detected mode: %d\n", ret);
-               goto error;
-       }
-
-       switch (val) {
-       case SDCA_DETECTED_MODE_DETECTION_IN_PROGRESS:
-       case SDCA_DETECTED_MODE_JACK_UNKNOWN:
-               reg = SDW_SDCA_CTL(interrupt->function->desc->adr,
-                                  interrupt->entity->id,
-                                  SDCA_CTL_GE_SELECTED_MODE, 0);
-
-               /*
-                * Selected mode is not normally marked as volatile register
-                * (RW), but here force a read from the hardware. If the
-                * detected mode is unknown we need to see what the device
-                * selected as a "safe" option.
-                */
-               regcache_drop_region(interrupt->function_regmap, reg, reg);
-
-               ret = regmap_read(interrupt->function_regmap, reg, &val);
-               if (ret) {
-                       dev_err(dev, "failed to re-check selected mode: %d\n", ret);
-                       goto error;
-               }
-               break;
-       default:
-               break;
-       }
-
-       dev_dbg(dev, "%s: %#x\n", interrupt->name, val);
-
-       ucontrol = kzalloc(sizeof(*ucontrol), GFP_KERNEL);
-       if (!ucontrol)
-               goto error;
-
-       ucontrol->value.enumerated.item[0] = snd_soc_enum_val_to_item(soc_enum, val);
-
-       down_write(rwsem);
-       ret = kctl->put(kctl, ucontrol);
-       up_write(rwsem);
-       if (ret < 0) {
-               dev_err(dev, "failed to update selected mode: %d\n", ret);
+       ret = sdca_jack_process(interrupt);
+       if (ret)
                goto error;
-       }
-
-       snd_ctl_notify(card->snd_card, SNDRV_CTL_EVENT_MASK_VALUE, &kctl->id);
 
        irqret = IRQ_HANDLED;
 error:
@@ -536,6 +463,10 @@ int sdca_irq_populate(struct sdca_function_data *function,
                                handler = function_status_handler;
                                break;
                        case SDCA_CTL_TYPE_S(GE, DETECTED_MODE):
+                               ret = sdca_jack_alloc_state(interrupt);
+                               if (ret)
+                                       return ret;
+
                                handler = detected_mode_handler;
                                break;
                        case SDCA_CTL_TYPE_S(XU, FDL_CURRENTOWNER):
diff --git a/sound/soc/sdca/sdca_jack.c b/sound/soc/sdca/sdca_jack.c
new file mode 100644 (file)
index 0000000..83b2b9c
--- /dev/null
@@ -0,0 +1,140 @@
+// SPDX-License-Identifier: GPL-2.0
+// Copyright (C) 2025 Cirrus Logic, Inc. and
+//                    Cirrus Logic International Semiconductor Ltd.
+
+/*
+ * The MIPI SDCA specification is available for public downloads at
+ * https://www.mipi.org/mipi-sdca-v1-0-download
+ */
+
+#include <linux/cleanup.h>
+#include <linux/device.h>
+#include <linux/dev_printk.h>
+#include <linux/soundwire/sdw.h>
+#include <linux/soundwire/sdw_registers.h>
+#include <linux/sprintf.h>
+#include <linux/regmap.h>
+#include <linux/rwsem.h>
+#include <sound/asound.h>
+#include <sound/control.h>
+#include <sound/sdca.h>
+#include <sound/sdca_function.h>
+#include <sound/sdca_interrupts.h>
+#include <sound/sdca_jack.h>
+#include <sound/soc-component.h>
+#include <sound/soc.h>
+
+/**
+ * sdca_jack_process - Process an SDCA jack event
+ * @interrupt: SDCA interrupt structure
+ *
+ * Return: Zero on success or a negative error code.
+ */
+int sdca_jack_process(struct sdca_interrupt *interrupt)
+{
+       struct device *dev = interrupt->dev;
+       struct snd_soc_component *component = interrupt->component;
+       struct snd_soc_card *card = component->card;
+       struct rw_semaphore *rwsem = &card->snd_card->controls_rwsem;
+       struct jack_state *state = interrupt->priv;
+       struct snd_kcontrol *kctl = state->kctl;
+       struct snd_ctl_elem_value *ucontrol __free(kfree) = NULL;
+       struct soc_enum *soc_enum;
+       unsigned int reg, val;
+       int ret;
+
+       if (!kctl) {
+               const char *name __free(kfree) = kasprintf(GFP_KERNEL, "%s %s",
+                                                          interrupt->entity->label,
+                                                          SDCA_CTL_SELECTED_MODE_NAME);
+
+               if (!name)
+                       return -ENOMEM;
+
+               kctl = snd_soc_component_get_kcontrol(component, name);
+               if (!kctl) {
+                       dev_dbg(dev, "control not found: %s\n", name);
+                       return -ENOENT;
+               }
+
+               state->kctl = kctl;
+       }
+
+       soc_enum = (struct soc_enum *)kctl->private_value;
+
+       reg = SDW_SDCA_CTL(interrupt->function->desc->adr, interrupt->entity->id,
+                          interrupt->control->sel, 0);
+
+       ret = regmap_read(interrupt->function_regmap, reg, &val);
+       if (ret < 0) {
+               dev_err(dev, "failed to read detected mode: %d\n", ret);
+               return ret;
+       }
+
+       switch (val) {
+       case SDCA_DETECTED_MODE_DETECTION_IN_PROGRESS:
+       case SDCA_DETECTED_MODE_JACK_UNKNOWN:
+               reg = SDW_SDCA_CTL(interrupt->function->desc->adr,
+                                  interrupt->entity->id,
+                                  SDCA_CTL_GE_SELECTED_MODE, 0);
+
+               /*
+                * Selected mode is not normally marked as volatile register
+                * (RW), but here force a read from the hardware. If the
+                * detected mode is unknown we need to see what the device
+                * selected as a "safe" option.
+                */
+               regcache_drop_region(interrupt->function_regmap, reg, reg);
+
+               ret = regmap_read(interrupt->function_regmap, reg, &val);
+               if (ret) {
+                       dev_err(dev, "failed to re-check selected mode: %d\n", ret);
+                       return ret;
+               }
+               break;
+       default:
+               break;
+       }
+
+       dev_dbg(dev, "%s: %#x\n", interrupt->name, val);
+
+       ucontrol = kzalloc(sizeof(*ucontrol), GFP_KERNEL);
+       if (!ucontrol)
+               return -ENOMEM;
+
+       ucontrol->value.enumerated.item[0] = snd_soc_enum_val_to_item(soc_enum, val);
+
+       down_write(rwsem);
+       ret = kctl->put(kctl, ucontrol);
+       up_write(rwsem);
+       if (ret < 0) {
+               dev_err(dev, "failed to update selected mode: %d\n", ret);
+               return ret;
+       }
+
+       snd_ctl_notify(card->snd_card, SNDRV_CTL_EVENT_MASK_VALUE, &kctl->id);
+
+       return 0;
+}
+EXPORT_SYMBOL_NS_GPL(sdca_jack_process, "SND_SOC_SDCA");
+
+/**
+ * sdca_jack_alloc_state - allocate state for a jack interrupt
+ * @interrupt: SDCA interrupt structure.
+ *
+ * Return: Zero on success or a negative error code.
+ */
+int sdca_jack_alloc_state(struct sdca_interrupt *interrupt)
+{
+       struct device *dev = interrupt->dev;
+       struct jack_state *jack_state;
+
+       jack_state = devm_kzalloc(dev, sizeof(*jack_state), GFP_KERNEL);
+       if (!jack_state)
+               return -ENOMEM;
+
+       interrupt->priv = jack_state;
+
+       return 0;
+}
+EXPORT_SYMBOL_NS_GPL(sdca_jack_alloc_state, "SND_SOC_SDCA");