ASoC: qcom: qdsp6: Add headphone jack for offload connection status
authorWesley Cheng <quic_wcheng@quicinc.com>
Wed, 9 Apr 2025 19:47:58 +0000 (12:47 -0700)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 11 Apr 2025 11:02:32 +0000 (13:02 +0200)
The headphone jack framework has a well defined infrastructure for
notifying userspace entities through input devices.  Expose a jack device
that carries information about if an offload capable device is connected.
Applications can further identify specific offloading information through
other SND kcontrols.

Signed-off-by: Wesley Cheng <quic_wcheng@quicinc.com>
Acked-by: Mark Brown <broonie@kernel.org>
Link: https://lore.kernel.org/r/20250409194804.3773260-26-quic_wcheng@quicinc.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
sound/soc/qcom/Kconfig
sound/soc/qcom/Makefile
sound/soc/qcom/qdsp6/q6usb.c
sound/soc/qcom/sm8250.c
sound/soc/qcom/usb_offload_utils.c [new file with mode: 0644]
sound/soc/qcom/usb_offload_utils.h [new file with mode: 0644]

index 1b4f3faadbc7de944cea4198e9ef32fe7ef00e67..e86b4a03dd61d2d3ad6a4d9602f69effbaefaa83 100644 (file)
@@ -118,10 +118,14 @@ config SND_SOC_QDSP6_PRM
        tristate
        select SND_SOC_QDSP6_PRM_LPASS_CLOCKS
 
+config SND_SOC_QCOM_OFFLOAD_UTILS
+       tristate
+
 config SND_SOC_QDSP6_USB
     tristate "SoC ALSA USB offloading backing for QDSP6"
     depends on SND_SOC_USB
     select AUXILIARY_BUS
+    select SND_SOC_QCOM_OFFLOAD_UTILS
 
     help
       Adds support for USB offloading for QDSP6 ASoC
index 16db7b53ddac7e4941f52f78cb753678e896afa9..985ce2ae286ba1860f338b37aaff0825e659f486 100644 (file)
@@ -30,6 +30,7 @@ snd-soc-sc8280xp-y := sc8280xp.o
 snd-soc-qcom-common-y := common.o
 snd-soc-qcom-sdw-y := sdw.o
 snd-soc-x1e80100-y := x1e80100.o
+snd-soc-qcom-offload-utils-objs := usb_offload_utils.o
 
 obj-$(CONFIG_SND_SOC_STORM) += snd-soc-storm.o
 obj-$(CONFIG_SND_SOC_APQ8016_SBC) += snd-soc-apq8016-sbc.o
@@ -42,6 +43,7 @@ obj-$(CONFIG_SND_SOC_SM8250) += snd-soc-sm8250.o
 obj-$(CONFIG_SND_SOC_QCOM_COMMON) += snd-soc-qcom-common.o
 obj-$(CONFIG_SND_SOC_QCOM_SDW) += snd-soc-qcom-sdw.o
 obj-$(CONFIG_SND_SOC_X1E80100) += snd-soc-x1e80100.o
+obj-$(CONFIG_SND_SOC_QCOM_OFFLOAD_UTILS) += snd-soc-qcom-offload-utils.o
 
 #DSP lib
 obj-$(CONFIG_SND_SOC_QDSP6) += qdsp6/
index 99848a320784e3feb91944de95c5e7d355b4ac04..6634e132787e0bfa5a13a18c79c89a7c7fd1d752 100644 (file)
@@ -15,6 +15,7 @@
 #include <linux/slab.h>
 
 #include <sound/asound.h>
+#include <sound/jack.h>
 #include <sound/pcm.h>
 #include <sound/pcm_params.h>
 #include <sound/q6usboffload.h>
@@ -32,6 +33,7 @@ struct q6usb_port_data {
        struct auxiliary_device uauxdev;
        struct q6afe_usb_cfg usb_cfg;
        struct snd_soc_usb *usb;
+       struct snd_soc_jack *hs_jack;
        struct q6usb_offload priv;
 
        /* Protects against operations between SOC USB and ASoC */
@@ -144,16 +146,54 @@ static int q6usb_alsa_connection_cb(struct snd_soc_usb *usb,
 
        mutex_lock(&data->mutex);
        if (connected) {
+               if (data->hs_jack)
+                       snd_jack_report(data->hs_jack->jack, SND_JACK_USB);
+
                /* Selects the latest USB headset plugged in for offloading */
                list_add_tail(&sdev->list, &data->devices);
        } else {
                list_del(&sdev->list);
+
+               if (data->hs_jack)
+                       snd_jack_report(data->hs_jack->jack, 0);
        }
        mutex_unlock(&data->mutex);
 
        return 0;
 }
 
+static void q6usb_component_disable_jack(struct q6usb_port_data *data)
+{
+       /* Offload jack has already been disabled */
+       if (!data->hs_jack)
+               return;
+
+       snd_jack_report(data->hs_jack->jack, 0);
+       data->hs_jack = NULL;
+}
+
+static void q6usb_component_enable_jack(struct q6usb_port_data *data,
+                                       struct snd_soc_jack *jack)
+{
+       snd_jack_report(jack->jack, !list_empty(&data->devices) ? SND_JACK_USB : 0);
+       data->hs_jack = jack;
+}
+
+static int q6usb_component_set_jack(struct snd_soc_component *component,
+                                   struct snd_soc_jack *jack, void *priv)
+{
+       struct q6usb_port_data *data = dev_get_drvdata(component->dev);
+
+       mutex_lock(&data->mutex);
+       if (jack)
+               q6usb_component_enable_jack(data, jack);
+       else
+               q6usb_component_disable_jack(data);
+       mutex_unlock(&data->mutex);
+
+       return 0;
+}
+
 static void q6usb_dai_aux_release(struct device *dev) {}
 
 static int q6usb_dai_add_aux_device(struct q6usb_port_data *data,
@@ -211,6 +251,7 @@ static void q6usb_component_remove(struct snd_soc_component *component)
 
 static const struct snd_soc_component_driver q6usb_dai_component = {
        .probe = q6usb_component_probe,
+       .set_jack = q6usb_component_set_jack,
        .remove = q6usb_component_remove,
        .name = "q6usb-dai-component",
        .dapm_widgets = q6usb_dai_widgets,
index 9039107972e2b34c5c4bfbb6e7da35878afaf38d..b70b2a5031dfbf69024666f8a1049c263efcde0a 100644 (file)
@@ -13,6 +13,7 @@
 #include <linux/input-event-codes.h>
 #include "qdsp6/q6afe.h"
 #include "common.h"
+#include "usb_offload_utils.h"
 #include "sdw.h"
 
 #define DRIVER_NAME            "sm8250"
@@ -23,14 +24,34 @@ struct sm8250_snd_data {
        struct snd_soc_card *card;
        struct sdw_stream_runtime *sruntime[AFE_PORT_MAX];
        struct snd_soc_jack jack;
+       struct snd_soc_jack usb_offload_jack;
+       bool usb_offload_jack_setup;
        bool jack_setup;
 };
 
 static int sm8250_snd_init(struct snd_soc_pcm_runtime *rtd)
 {
        struct sm8250_snd_data *data = snd_soc_card_get_drvdata(rtd->card);
+       struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
+       int ret;
+
+       if (cpu_dai->id == USB_RX)
+               ret = qcom_snd_usb_offload_jack_setup(rtd, &data->usb_offload_jack,
+                                                     &data->usb_offload_jack_setup);
+       else
+               ret = qcom_snd_wcd_jack_setup(rtd, &data->jack, &data->jack_setup);
+       return ret;
+}
+
+static void sm8250_snd_exit(struct snd_soc_pcm_runtime *rtd)
+{
+       struct sm8250_snd_data *data = snd_soc_card_get_drvdata(rtd->card);
+       struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
+
+       if (cpu_dai->id == USB_RX)
+               qcom_snd_usb_offload_jack_remove(rtd,
+                                                &data->usb_offload_jack_setup);
 
-       return qcom_snd_wcd_jack_setup(rtd, &data->jack, &data->jack_setup);
 }
 
 static int sm8250_be_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
@@ -148,6 +169,7 @@ static void sm8250_add_be_ops(struct snd_soc_card *card)
        for_each_card_prelinks(card, i, link) {
                if (link->no_pcm == 1) {
                        link->init = sm8250_snd_init;
+                       link->exit = sm8250_snd_exit;
                        link->be_hw_params_fixup = sm8250_be_hw_params_fixup;
                        link->ops = &sm8250_be_ops;
                }
diff --git a/sound/soc/qcom/usb_offload_utils.c b/sound/soc/qcom/usb_offload_utils.c
new file mode 100644 (file)
index 0000000..0a24b27
--- /dev/null
@@ -0,0 +1,56 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (c) 2022-2025 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+#include <dt-bindings/sound/qcom,q6afe.h>
+#include <linux/module.h>
+#include <sound/jack.h>
+#include <sound/soc-usb.h>
+
+#include "usb_offload_utils.h"
+
+int qcom_snd_usb_offload_jack_setup(struct snd_soc_pcm_runtime *rtd,
+                                   struct snd_soc_jack *jack, bool *jack_setup)
+{
+       struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
+       struct snd_soc_dai *codec_dai = snd_soc_rtd_to_codec(rtd, 0);
+       int ret = 0;
+
+       if (cpu_dai->id != USB_RX)
+               return -EINVAL;
+
+       if (!*jack_setup) {
+               ret = snd_soc_usb_setup_offload_jack(codec_dai->component, jack);
+               if (ret)
+                       return ret;
+       }
+
+       *jack_setup = true;
+
+       return 0;
+}
+EXPORT_SYMBOL_GPL(qcom_snd_usb_offload_jack_setup);
+
+int qcom_snd_usb_offload_jack_remove(struct snd_soc_pcm_runtime *rtd,
+                                    bool *jack_setup)
+{
+       struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
+       struct snd_soc_dai *codec_dai = snd_soc_rtd_to_codec(rtd, 0);
+       int ret = 0;
+
+       if (cpu_dai->id != USB_RX)
+               return -EINVAL;
+
+       if (*jack_setup) {
+               ret = snd_soc_component_set_jack(codec_dai->component, NULL, NULL);
+               if (ret)
+                       return ret;
+       }
+
+       *jack_setup = false;
+
+       return 0;
+}
+EXPORT_SYMBOL_GPL(qcom_snd_usb_offload_jack_remove);
+MODULE_DESCRIPTION("ASoC Q6 USB offload controls");
+MODULE_LICENSE("GPL");
diff --git a/sound/soc/qcom/usb_offload_utils.h b/sound/soc/qcom/usb_offload_utils.h
new file mode 100644 (file)
index 0000000..c3f411f
--- /dev/null
@@ -0,0 +1,30 @@
+/* SPDX-License-Identifier: GPL-2.0
+ *
+ * Copyright (c) 2022-2025 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
+#ifndef __QCOM_SND_USB_OFFLOAD_UTILS_H__
+#define __QCOM_SND_USB_OFFLOAD_UTILS_H__
+
+#include <sound/soc.h>
+
+#if IS_ENABLED(CONFIG_SND_SOC_QCOM_OFFLOAD_UTILS)
+int qcom_snd_usb_offload_jack_setup(struct snd_soc_pcm_runtime *rtd,
+                                   struct snd_soc_jack *jack, bool *jack_setup);
+
+int qcom_snd_usb_offload_jack_remove(struct snd_soc_pcm_runtime *rtd,
+                                    bool *jack_setup);
+#else
+static inline int qcom_snd_usb_offload_jack_setup(struct snd_soc_pcm_runtime *rtd,
+                                                 struct snd_soc_jack *jack,
+                                                 bool *jack_setup)
+{
+       return -ENODEV;
+}
+
+static inline int qcom_snd_usb_offload_jack_remove(struct snd_soc_pcm_runtime *rtd,
+                                                  bool *jack_setup)
+{
+       return -ENODEV;
+}
+#endif /* IS_ENABLED(CONFIG_SND_SOC_QCOM_OFFLOAD_UTILS) */
+#endif /* __QCOM_SND_USB_OFFLOAD_UTILS_H__ */