Commit 9ba17b4d authored by Takashi Iwai's avatar Takashi Iwai

ALSA: hda - Implement uncached version of parameter reads

Sometimes we need the uncached reads, e.g. for refreshing the tree.
This patch provides the helper function for that and uses it for
refreshing widgets, reading subtrees and the whole proc reads.
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent 01ed3c06
...@@ -107,6 +107,8 @@ int snd_hdac_read(struct hdac_device *codec, hda_nid_t nid, ...@@ -107,6 +107,8 @@ int snd_hdac_read(struct hdac_device *codec, hda_nid_t nid,
unsigned int verb, unsigned int parm, unsigned int *res); unsigned int verb, unsigned int parm, unsigned int *res);
int _snd_hdac_read_parm(struct hdac_device *codec, hda_nid_t nid, int parm, int _snd_hdac_read_parm(struct hdac_device *codec, hda_nid_t nid, int parm,
unsigned int *res); unsigned int *res);
int snd_hdac_read_parm_uncached(struct hdac_device *codec, hda_nid_t nid,
int parm);
int snd_hdac_get_connections(struct hdac_device *codec, hda_nid_t nid, int snd_hdac_get_connections(struct hdac_device *codec, hda_nid_t nid,
hda_nid_t *conn_list, int max_conns); hda_nid_t *conn_list, int max_conns);
int snd_hdac_get_sub_nodes(struct hdac_device *codec, hda_nid_t nid, int snd_hdac_get_sub_nodes(struct hdac_device *codec, hda_nid_t nid,
......
...@@ -249,6 +249,29 @@ int _snd_hdac_read_parm(struct hdac_device *codec, hda_nid_t nid, int parm, ...@@ -249,6 +249,29 @@ int _snd_hdac_read_parm(struct hdac_device *codec, hda_nid_t nid, int parm,
} }
EXPORT_SYMBOL_GPL(_snd_hdac_read_parm); EXPORT_SYMBOL_GPL(_snd_hdac_read_parm);
/**
* snd_hdac_read_parm_uncached - read a codec parameter without caching
* @codec: the codec object
* @nid: NID to read a parameter
* @parm: parameter to read
*
* Returns -1 for error. If you need to distinguish the error more
* strictly, use snd_hdac_read() directly.
*/
int snd_hdac_read_parm_uncached(struct hdac_device *codec, hda_nid_t nid,
int parm)
{
int val;
if (codec->regmap)
regcache_cache_bypass(codec->regmap, true);
val = snd_hdac_read_parm(codec, nid, parm);
if (codec->regmap)
regcache_cache_bypass(codec->regmap, false);
return val;
}
EXPORT_SYMBOL_GPL(snd_hdac_read_parm_uncached);
/** /**
* snd_hdac_get_sub_nodes - get start NID and number of subtree nodes * snd_hdac_get_sub_nodes - get start NID and number of subtree nodes
* @codec: the codec object * @codec: the codec object
...@@ -256,13 +279,14 @@ EXPORT_SYMBOL_GPL(_snd_hdac_read_parm); ...@@ -256,13 +279,14 @@ EXPORT_SYMBOL_GPL(_snd_hdac_read_parm);
* @start_id: the pointer to store the starting NID * @start_id: the pointer to store the starting NID
* *
* Returns the number of subtree nodes or zero if not found. * Returns the number of subtree nodes or zero if not found.
* This function reads parameters always without caching.
*/ */
int snd_hdac_get_sub_nodes(struct hdac_device *codec, hda_nid_t nid, int snd_hdac_get_sub_nodes(struct hdac_device *codec, hda_nid_t nid,
hda_nid_t *start_id) hda_nid_t *start_id)
{ {
unsigned int parm; unsigned int parm;
parm = snd_hdac_read_parm(codec, nid, AC_PAR_NODE_COUNT); parm = snd_hdac_read_parm_uncached(codec, nid, AC_PAR_NODE_COUNT);
if (parm == -1) { if (parm == -1) {
*start_id = 0; *start_id = 0;
return 0; return 0;
......
...@@ -586,8 +586,8 @@ static int read_widget_caps(struct hda_codec *codec, hda_nid_t fg_node) ...@@ -586,8 +586,8 @@ static int read_widget_caps(struct hda_codec *codec, hda_nid_t fg_node)
return -ENOMEM; return -ENOMEM;
nid = codec->core.start_nid; nid = codec->core.start_nid;
for (i = 0; i < codec->core.num_nodes; i++, nid++) for (i = 0; i < codec->core.num_nodes; i++, nid++)
codec->wcaps[i] = snd_hda_param_read(codec, nid, codec->wcaps[i] = snd_hdac_read_parm_uncached(&codec->core,
AC_PAR_AUDIO_WIDGET_CAP); nid, AC_PAR_AUDIO_WIDGET_CAP);
return 0; return 0;
} }
......
...@@ -32,6 +32,10 @@ static int dump_coef = -1; ...@@ -32,6 +32,10 @@ static int dump_coef = -1;
module_param(dump_coef, int, 0644); module_param(dump_coef, int, 0644);
MODULE_PARM_DESC(dump_coef, "Dump processing coefficients in codec proc file (-1=auto, 0=disable, 1=enable)"); MODULE_PARM_DESC(dump_coef, "Dump processing coefficients in codec proc file (-1=auto, 0=disable, 1=enable)");
/* always use noncached version */
#define param_read(codec, nid, parm) \
snd_hdac_read_parm_uncached(&(codec)->core, nid, parm)
static char *bits_names(unsigned int bits, char *names[], int size) static char *bits_names(unsigned int bits, char *names[], int size)
{ {
int i, n; int i, n;
...@@ -119,8 +123,7 @@ static void print_amp_caps(struct snd_info_buffer *buffer, ...@@ -119,8 +123,7 @@ static void print_amp_caps(struct snd_info_buffer *buffer,
struct hda_codec *codec, hda_nid_t nid, int dir) struct hda_codec *codec, hda_nid_t nid, int dir)
{ {
unsigned int caps; unsigned int caps;
caps = snd_hda_param_read(codec, nid, caps = param_read(codec, nid, dir == HDA_OUTPUT ?
dir == HDA_OUTPUT ?
AC_PAR_AMP_OUT_CAP : AC_PAR_AMP_IN_CAP); AC_PAR_AMP_OUT_CAP : AC_PAR_AMP_IN_CAP);
if (caps == -1 || caps == 0) { if (caps == -1 || caps == 0) {
snd_iprintf(buffer, "N/A\n"); snd_iprintf(buffer, "N/A\n");
...@@ -225,8 +228,8 @@ static void print_pcm_formats(struct snd_info_buffer *buffer, ...@@ -225,8 +228,8 @@ static void print_pcm_formats(struct snd_info_buffer *buffer,
static void print_pcm_caps(struct snd_info_buffer *buffer, static void print_pcm_caps(struct snd_info_buffer *buffer,
struct hda_codec *codec, hda_nid_t nid) struct hda_codec *codec, hda_nid_t nid)
{ {
unsigned int pcm = snd_hda_param_read(codec, nid, AC_PAR_PCM); unsigned int pcm = param_read(codec, nid, AC_PAR_PCM);
unsigned int stream = snd_hda_param_read(codec, nid, AC_PAR_STREAM); unsigned int stream = param_read(codec, nid, AC_PAR_STREAM);
if (pcm == -1 || stream == -1) { if (pcm == -1 || stream == -1) {
snd_iprintf(buffer, "N/A\n"); snd_iprintf(buffer, "N/A\n");
return; return;
...@@ -273,7 +276,7 @@ static void print_pin_caps(struct snd_info_buffer *buffer, ...@@ -273,7 +276,7 @@ static void print_pin_caps(struct snd_info_buffer *buffer,
static char *jack_conns[4] = { "Jack", "N/A", "Fixed", "Both" }; static char *jack_conns[4] = { "Jack", "N/A", "Fixed", "Both" };
unsigned int caps, val; unsigned int caps, val;
caps = snd_hda_param_read(codec, nid, AC_PAR_PIN_CAP); caps = param_read(codec, nid, AC_PAR_PIN_CAP);
snd_iprintf(buffer, " Pincap 0x%08x:", caps); snd_iprintf(buffer, " Pincap 0x%08x:", caps);
if (caps & AC_PINCAP_IN) if (caps & AC_PINCAP_IN)
snd_iprintf(buffer, " IN"); snd_iprintf(buffer, " IN");
...@@ -401,8 +404,7 @@ static void print_pin_ctls(struct snd_info_buffer *buffer, ...@@ -401,8 +404,7 @@ static void print_pin_ctls(struct snd_info_buffer *buffer,
static void print_vol_knob(struct snd_info_buffer *buffer, static void print_vol_knob(struct snd_info_buffer *buffer,
struct hda_codec *codec, hda_nid_t nid) struct hda_codec *codec, hda_nid_t nid)
{ {
unsigned int cap = snd_hda_param_read(codec, nid, unsigned int cap = param_read(codec, nid, AC_PAR_VOL_KNB_CAP);
AC_PAR_VOL_KNB_CAP);
snd_iprintf(buffer, " Volume-Knob: delta=%d, steps=%d, ", snd_iprintf(buffer, " Volume-Knob: delta=%d, steps=%d, ",
(cap >> 7) & 1, cap & 0x7f); (cap >> 7) & 1, cap & 0x7f);
cap = snd_hda_codec_read(codec, nid, 0, cap = snd_hda_codec_read(codec, nid, 0,
...@@ -487,7 +489,7 @@ static void print_power_state(struct snd_info_buffer *buffer, ...@@ -487,7 +489,7 @@ static void print_power_state(struct snd_info_buffer *buffer,
[ilog2(AC_PWRST_EPSS)] = "EPSS", [ilog2(AC_PWRST_EPSS)] = "EPSS",
}; };
int sup = snd_hda_param_read(codec, nid, AC_PAR_POWER_STATE); int sup = param_read(codec, nid, AC_PAR_POWER_STATE);
int pwr = snd_hda_codec_read(codec, nid, 0, int pwr = snd_hda_codec_read(codec, nid, 0,
AC_VERB_GET_POWER_STATE, 0); AC_VERB_GET_POWER_STATE, 0);
if (sup != -1) if (sup != -1)
...@@ -531,8 +533,7 @@ static void print_proc_caps(struct snd_info_buffer *buffer, ...@@ -531,8 +533,7 @@ static void print_proc_caps(struct snd_info_buffer *buffer,
struct hda_codec *codec, hda_nid_t nid) struct hda_codec *codec, hda_nid_t nid)
{ {
unsigned int i, ncoeff, oldindex; unsigned int i, ncoeff, oldindex;
unsigned int proc_caps = snd_hda_param_read(codec, nid, unsigned int proc_caps = param_read(codec, nid, AC_PAR_PROC_CAP);
AC_PAR_PROC_CAP);
ncoeff = (proc_caps & AC_PCAP_NUM_COEF) >> AC_PCAP_NUM_COEF_SHIFT; ncoeff = (proc_caps & AC_PCAP_NUM_COEF) >> AC_PCAP_NUM_COEF_SHIFT;
snd_iprintf(buffer, " Processing caps: benign=%d, ncoeff=%d\n", snd_iprintf(buffer, " Processing caps: benign=%d, ncoeff=%d\n",
proc_caps & AC_PCAP_BENIGN, ncoeff); proc_caps & AC_PCAP_BENIGN, ncoeff);
...@@ -597,7 +598,7 @@ static void print_gpio(struct snd_info_buffer *buffer, ...@@ -597,7 +598,7 @@ static void print_gpio(struct snd_info_buffer *buffer,
struct hda_codec *codec, hda_nid_t nid) struct hda_codec *codec, hda_nid_t nid)
{ {
unsigned int gpio = unsigned int gpio =
snd_hda_param_read(codec, codec->core.afg, AC_PAR_GPIO_CAP); param_read(codec, codec->core.afg, AC_PAR_GPIO_CAP);
unsigned int enable, direction, wake, unsol, sticky, data; unsigned int enable, direction, wake, unsol, sticky, data;
int i, max; int i, max;
snd_iprintf(buffer, "GPIO: io=%d, o=%d, i=%d, " snd_iprintf(buffer, "GPIO: io=%d, o=%d, i=%d, "
...@@ -727,8 +728,7 @@ static void print_codec_info(struct snd_info_entry *entry, ...@@ -727,8 +728,7 @@ static void print_codec_info(struct snd_info_entry *entry,
for (i = 0; i < nodes; i++, nid++) { for (i = 0; i < nodes; i++, nid++) {
unsigned int wid_caps = unsigned int wid_caps =
snd_hda_param_read(codec, nid, param_read(codec, nid, AC_PAR_AUDIO_WIDGET_CAP);
AC_PAR_AUDIO_WIDGET_CAP);
unsigned int wid_type = get_wcaps_type(wid_caps); unsigned int wid_type = get_wcaps_type(wid_caps);
hda_nid_t *conn = NULL; hda_nid_t *conn = NULL;
int conn_len = 0; int conn_len = 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment