mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-08 23:04:35 +08:00
5204a05d70
This patch adds DP-MST jack support which will be used on NVIDIA platforms. Today, DP-MST audio is supported only if the codec has acomp support. This patch makes it possible to add DP-MST support for non-acomp codecs. For the codecs supporting DP-MST audio, each pin can contain several device entries. Each device entry is a virtual pin, described by pin_nid and dev_id in struct hdmi_spec_per_pin. For monitor hotplug event handling, non-acomp codecs enable and register jack-detection for every hdmi_spec_per_pin. This patch updates every relevant function in hda_jack.h and its implementation in hda_jack.c, to consider dev_id along with pin_nid. Changes to the HD Audio specification to support DP-MST audio are described in the Intel Document Change Notification (DCN) number HDA040-A. From HDA040-A, "For the case of multi stream capable Digital Display Pin Widget, [the Get Pin Sense verb] can be used to read a specific Device Entry state as reported in Get Device List Entry verb." This patch updates the read_pin_sense() function to take the dev_id as an argument and pass it as a parameter to the Get Pin Sense verb. Bits 15 through 20 from the Unsolicited Response for intrinsic events contain the index of the Device Entry that generated the event. This patch updates the Unsolicited Response event handlers to extract the device entry index from the response and pass it to snd_hda_jack_tbl_get_from_tag(). This patch updates snd_hda_jack_tbl_new() to take a dev_id argument and store it in the jack structure, and to make sure not to generate a different tag when called more than once for the same nid. Signed-off-by: Nikhil Mahale <nmahale@nvidia.com> Link: https://lore.kernel.org/r/20191119084710.29267-3-nmahale@nvidia.com Signed-off-by: Takashi Iwai <tiwai@suse.de>
187 lines
5.4 KiB
C
187 lines
5.4 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
/*
|
|
* Jack-detection handling for HD-audio
|
|
*
|
|
* Copyright (c) 2011 Takashi Iwai <tiwai@suse.de>
|
|
*/
|
|
|
|
#ifndef __SOUND_HDA_JACK_H
|
|
#define __SOUND_HDA_JACK_H
|
|
|
|
#include <linux/err.h>
|
|
#include <sound/jack.h>
|
|
|
|
struct auto_pin_cfg;
|
|
struct hda_jack_tbl;
|
|
struct hda_jack_callback;
|
|
|
|
typedef void (*hda_jack_callback_fn) (struct hda_codec *, struct hda_jack_callback *);
|
|
|
|
struct hda_jack_callback {
|
|
hda_nid_t nid;
|
|
int dev_id;
|
|
hda_jack_callback_fn func;
|
|
unsigned int private_data; /* arbitrary data */
|
|
unsigned int unsol_res; /* unsolicited event bits */
|
|
struct hda_jack_tbl *jack; /* associated jack entry */
|
|
struct hda_jack_callback *next;
|
|
};
|
|
|
|
struct hda_jack_tbl {
|
|
hda_nid_t nid;
|
|
int dev_id;
|
|
unsigned char tag; /* unsol event tag */
|
|
struct hda_jack_callback *callback;
|
|
/* jack-detection stuff */
|
|
unsigned int pin_sense; /* cached pin-sense value */
|
|
unsigned int jack_detect:1; /* capable of jack-detection? */
|
|
unsigned int jack_dirty:1; /* needs to update? */
|
|
unsigned int phantom_jack:1; /* a fixed, always present port? */
|
|
unsigned int block_report:1; /* in a transitional state - do not report to userspace */
|
|
hda_nid_t gating_jack; /* valid when gating jack plugged */
|
|
hda_nid_t gated_jack; /* gated is dependent on this jack */
|
|
int type;
|
|
int button_state;
|
|
struct snd_jack *jack;
|
|
};
|
|
|
|
struct hda_jack_keymap {
|
|
enum snd_jack_types type;
|
|
int key;
|
|
};
|
|
|
|
struct hda_jack_tbl *
|
|
snd_hda_jack_tbl_get_mst(struct hda_codec *codec, hda_nid_t nid, int dev_id);
|
|
|
|
/**
|
|
* snd_hda_jack_tbl_get - query the jack-table entry for the given NID
|
|
* @codec: the HDA codec
|
|
* @nid: pin NID to refer to
|
|
*/
|
|
static inline struct hda_jack_tbl *
|
|
snd_hda_jack_tbl_get(struct hda_codec *codec, hda_nid_t nid)
|
|
{
|
|
return snd_hda_jack_tbl_get_mst(codec, nid, 0);
|
|
}
|
|
|
|
struct hda_jack_tbl *
|
|
snd_hda_jack_tbl_get_from_tag(struct hda_codec *codec,
|
|
unsigned char tag, int dev_id);
|
|
|
|
void snd_hda_jack_tbl_clear(struct hda_codec *codec);
|
|
|
|
void snd_hda_jack_set_dirty_all(struct hda_codec *codec);
|
|
|
|
int snd_hda_jack_detect_enable(struct hda_codec *codec, hda_nid_t nid,
|
|
int dev_id);
|
|
|
|
struct hda_jack_callback *
|
|
snd_hda_jack_detect_enable_callback_mst(struct hda_codec *codec, hda_nid_t nid,
|
|
int dev_id, hda_jack_callback_fn cb);
|
|
|
|
/**
|
|
* snd_hda_jack_detect_enable - enable the jack-detection
|
|
* @codec: the HDA codec
|
|
* @nid: pin NID to enable
|
|
* @func: callback function to register
|
|
*
|
|
* In the case of error, the return value will be a pointer embedded with
|
|
* errno. Check and handle the return value appropriately with standard
|
|
* macros such as @IS_ERR() and @PTR_ERR().
|
|
*/
|
|
static inline struct hda_jack_callback *
|
|
snd_hda_jack_detect_enable_callback(struct hda_codec *codec, hda_nid_t nid,
|
|
hda_jack_callback_fn cb)
|
|
{
|
|
return snd_hda_jack_detect_enable_callback_mst(codec, nid, 0, cb);
|
|
}
|
|
|
|
int snd_hda_jack_set_gating_jack(struct hda_codec *codec, hda_nid_t gated_nid,
|
|
hda_nid_t gating_nid);
|
|
|
|
u32 snd_hda_jack_pin_sense(struct hda_codec *codec, hda_nid_t nid, int dev_id);
|
|
|
|
/* the jack state returned from snd_hda_jack_detect_state() */
|
|
enum {
|
|
HDA_JACK_NOT_PRESENT, HDA_JACK_PRESENT, HDA_JACK_PHANTOM,
|
|
};
|
|
|
|
int snd_hda_jack_detect_state_mst(struct hda_codec *codec, hda_nid_t nid,
|
|
int dev_id);
|
|
|
|
/**
|
|
* snd_hda_jack_detect_state - query pin Presence Detect status
|
|
* @codec: the CODEC to sense
|
|
* @nid: the pin NID to sense
|
|
*
|
|
* Query and return the pin's Presence Detect status, as either
|
|
* HDA_JACK_NOT_PRESENT, HDA_JACK_PRESENT or HDA_JACK_PHANTOM.
|
|
*/
|
|
static inline int
|
|
snd_hda_jack_detect_state(struct hda_codec *codec, hda_nid_t nid)
|
|
{
|
|
return snd_hda_jack_detect_state_mst(codec, nid, 0);
|
|
}
|
|
|
|
/**
|
|
* snd_hda_jack_detect_mst - Detect the jack
|
|
* @codec: the HDA codec
|
|
* @nid: pin NID to check jack detection
|
|
* @dev_id: pin device entry id
|
|
*/
|
|
static inline bool
|
|
snd_hda_jack_detect_mst(struct hda_codec *codec, hda_nid_t nid, int dev_id)
|
|
{
|
|
return snd_hda_jack_detect_state_mst(codec, nid, dev_id) !=
|
|
HDA_JACK_NOT_PRESENT;
|
|
}
|
|
|
|
/**
|
|
* snd_hda_jack_detect - Detect the jack
|
|
* @codec: the HDA codec
|
|
* @nid: pin NID to check jack detection
|
|
*/
|
|
static inline bool
|
|
snd_hda_jack_detect(struct hda_codec *codec, hda_nid_t nid)
|
|
{
|
|
return snd_hda_jack_detect_mst(codec, nid, 0);
|
|
}
|
|
|
|
bool is_jack_detectable(struct hda_codec *codec, hda_nid_t nid);
|
|
|
|
int snd_hda_jack_add_kctl_mst(struct hda_codec *codec, hda_nid_t nid,
|
|
int dev_id, const char *name, bool phantom_jack,
|
|
int type, const struct hda_jack_keymap *keymap);
|
|
|
|
/**
|
|
* snd_hda_jack_add_kctl - Add a kctl for the given pin
|
|
* @codec: the HDA codec
|
|
* @nid: pin NID to assign
|
|
* @name: string name for the jack
|
|
* @phantom_jack: flag to deal as a phantom jack
|
|
* @type: jack type bits to be reported, 0 for guessing from pincfg
|
|
* @keymap: optional jack / key mapping
|
|
*
|
|
* This assigns a jack-detection kctl to the given pin. The kcontrol
|
|
* will have the given name and index.
|
|
*/
|
|
static inline int
|
|
snd_hda_jack_add_kctl(struct hda_codec *codec, hda_nid_t nid,
|
|
const char *name, bool phantom_jack,
|
|
int type, const struct hda_jack_keymap *keymap)
|
|
{
|
|
return snd_hda_jack_add_kctl_mst(codec, nid, 0,
|
|
name, phantom_jack, type, keymap);
|
|
}
|
|
|
|
int snd_hda_jack_add_kctls(struct hda_codec *codec,
|
|
const struct auto_pin_cfg *cfg);
|
|
|
|
void snd_hda_jack_report_sync(struct hda_codec *codec);
|
|
|
|
void snd_hda_jack_unsol_event(struct hda_codec *codec, unsigned int res);
|
|
|
|
void snd_hda_jack_poll_all(struct hda_codec *codec);
|
|
|
|
#endif /* __SOUND_HDA_JACK_H */
|