Commit 1b8d0d87 authored by Wesley Cheng's avatar Wesley Cheng Committed by Greg Kroah-Hartman
Browse files

ASoC: qcom: qdsp6: Add headphone jack for offload connection status



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: default avatarWesley Cheng <quic_wcheng@quicinc.com>
Acked-by: default avatarMark Brown <broonie@kernel.org>
Link: https://lore.kernel.org/r/20250409194804.3773260-26-quic_wcheng@quicinc.com


Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 72b0b8b2
Loading
Loading
Loading
Loading
+4 −0
Original line number Diff line number Diff line
@@ -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
+2 −0
Original line number Diff line number Diff line
@@ -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/
+41 −0
Original line number Diff line number Diff line
@@ -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,11 +146,49 @@ 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;
@@ -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,
+23 −1
Original line number Diff line number Diff line
@@ -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;
		}
+56 −0
Original line number Diff line number Diff line
// 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");
Loading