mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
synced 2025-08-26 21:52:20 +00:00
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: 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>
This commit is contained in:
parent
72b0b8b299
commit
1b8d0d87b9
@ -118,10 +118,14 @@ config SND_SOC_QDSP6_PRM
|
|||||||
tristate
|
tristate
|
||||||
select SND_SOC_QDSP6_PRM_LPASS_CLOCKS
|
select SND_SOC_QDSP6_PRM_LPASS_CLOCKS
|
||||||
|
|
||||||
|
config SND_SOC_QCOM_OFFLOAD_UTILS
|
||||||
|
tristate
|
||||||
|
|
||||||
config SND_SOC_QDSP6_USB
|
config SND_SOC_QDSP6_USB
|
||||||
tristate "SoC ALSA USB offloading backing for QDSP6"
|
tristate "SoC ALSA USB offloading backing for QDSP6"
|
||||||
depends on SND_SOC_USB
|
depends on SND_SOC_USB
|
||||||
select AUXILIARY_BUS
|
select AUXILIARY_BUS
|
||||||
|
select SND_SOC_QCOM_OFFLOAD_UTILS
|
||||||
|
|
||||||
help
|
help
|
||||||
Adds support for USB offloading for QDSP6 ASoC
|
Adds support for USB offloading for QDSP6 ASoC
|
||||||
|
@ -30,6 +30,7 @@ snd-soc-sc8280xp-y := sc8280xp.o
|
|||||||
snd-soc-qcom-common-y := common.o
|
snd-soc-qcom-common-y := common.o
|
||||||
snd-soc-qcom-sdw-y := sdw.o
|
snd-soc-qcom-sdw-y := sdw.o
|
||||||
snd-soc-x1e80100-y := x1e80100.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_STORM) += snd-soc-storm.o
|
||||||
obj-$(CONFIG_SND_SOC_APQ8016_SBC) += snd-soc-apq8016-sbc.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_COMMON) += snd-soc-qcom-common.o
|
||||||
obj-$(CONFIG_SND_SOC_QCOM_SDW) += snd-soc-qcom-sdw.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_X1E80100) += snd-soc-x1e80100.o
|
||||||
|
obj-$(CONFIG_SND_SOC_QCOM_OFFLOAD_UTILS) += snd-soc-qcom-offload-utils.o
|
||||||
|
|
||||||
#DSP lib
|
#DSP lib
|
||||||
obj-$(CONFIG_SND_SOC_QDSP6) += qdsp6/
|
obj-$(CONFIG_SND_SOC_QDSP6) += qdsp6/
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
|
||||||
#include <sound/asound.h>
|
#include <sound/asound.h>
|
||||||
|
#include <sound/jack.h>
|
||||||
#include <sound/pcm.h>
|
#include <sound/pcm.h>
|
||||||
#include <sound/pcm_params.h>
|
#include <sound/pcm_params.h>
|
||||||
#include <sound/q6usboffload.h>
|
#include <sound/q6usboffload.h>
|
||||||
@ -32,6 +33,7 @@ struct q6usb_port_data {
|
|||||||
struct auxiliary_device uauxdev;
|
struct auxiliary_device uauxdev;
|
||||||
struct q6afe_usb_cfg usb_cfg;
|
struct q6afe_usb_cfg usb_cfg;
|
||||||
struct snd_soc_usb *usb;
|
struct snd_soc_usb *usb;
|
||||||
|
struct snd_soc_jack *hs_jack;
|
||||||
struct q6usb_offload priv;
|
struct q6usb_offload priv;
|
||||||
|
|
||||||
/* Protects against operations between SOC USB and ASoC */
|
/* 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);
|
mutex_lock(&data->mutex);
|
||||||
if (connected) {
|
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 */
|
/* Selects the latest USB headset plugged in for offloading */
|
||||||
list_add_tail(&sdev->list, &data->devices);
|
list_add_tail(&sdev->list, &data->devices);
|
||||||
} else {
|
} else {
|
||||||
list_del(&sdev->list);
|
list_del(&sdev->list);
|
||||||
|
|
||||||
|
if (data->hs_jack)
|
||||||
|
snd_jack_report(data->hs_jack->jack, 0);
|
||||||
}
|
}
|
||||||
mutex_unlock(&data->mutex);
|
mutex_unlock(&data->mutex);
|
||||||
|
|
||||||
return 0;
|
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 void q6usb_dai_aux_release(struct device *dev) {}
|
||||||
|
|
||||||
static int q6usb_dai_add_aux_device(struct q6usb_port_data *data,
|
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 = {
|
static const struct snd_soc_component_driver q6usb_dai_component = {
|
||||||
.probe = q6usb_component_probe,
|
.probe = q6usb_component_probe,
|
||||||
|
.set_jack = q6usb_component_set_jack,
|
||||||
.remove = q6usb_component_remove,
|
.remove = q6usb_component_remove,
|
||||||
.name = "q6usb-dai-component",
|
.name = "q6usb-dai-component",
|
||||||
.dapm_widgets = q6usb_dai_widgets,
|
.dapm_widgets = q6usb_dai_widgets,
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
#include <linux/input-event-codes.h>
|
#include <linux/input-event-codes.h>
|
||||||
#include "qdsp6/q6afe.h"
|
#include "qdsp6/q6afe.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
#include "usb_offload_utils.h"
|
||||||
#include "sdw.h"
|
#include "sdw.h"
|
||||||
|
|
||||||
#define DRIVER_NAME "sm8250"
|
#define DRIVER_NAME "sm8250"
|
||||||
@ -23,14 +24,34 @@ struct sm8250_snd_data {
|
|||||||
struct snd_soc_card *card;
|
struct snd_soc_card *card;
|
||||||
struct sdw_stream_runtime *sruntime[AFE_PORT_MAX];
|
struct sdw_stream_runtime *sruntime[AFE_PORT_MAX];
|
||||||
struct snd_soc_jack jack;
|
struct snd_soc_jack jack;
|
||||||
|
struct snd_soc_jack usb_offload_jack;
|
||||||
|
bool usb_offload_jack_setup;
|
||||||
bool jack_setup;
|
bool jack_setup;
|
||||||
};
|
};
|
||||||
|
|
||||||
static int sm8250_snd_init(struct snd_soc_pcm_runtime *rtd)
|
static int sm8250_snd_init(struct snd_soc_pcm_runtime *rtd)
|
||||||
{
|
{
|
||||||
struct sm8250_snd_data *data = snd_soc_card_get_drvdata(rtd->card);
|
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,
|
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) {
|
for_each_card_prelinks(card, i, link) {
|
||||||
if (link->no_pcm == 1) {
|
if (link->no_pcm == 1) {
|
||||||
link->init = sm8250_snd_init;
|
link->init = sm8250_snd_init;
|
||||||
|
link->exit = sm8250_snd_exit;
|
||||||
link->be_hw_params_fixup = sm8250_be_hw_params_fixup;
|
link->be_hw_params_fixup = sm8250_be_hw_params_fixup;
|
||||||
link->ops = &sm8250_be_ops;
|
link->ops = &sm8250_be_ops;
|
||||||
}
|
}
|
||||||
|
56
sound/soc/qcom/usb_offload_utils.c
Normal file
56
sound/soc/qcom/usb_offload_utils.c
Normal file
@ -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");
|
30
sound/soc/qcom/usb_offload_utils.h
Normal file
30
sound/soc/qcom/usb_offload_utils.h
Normal file
@ -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__ */
|
Loading…
Reference in New Issue
Block a user