diff options
author | Takashi Iwai <tiwai@suse.de> | 2009-02-20 08:11:16 -0500 |
---|---|---|
committer | Takashi Iwai <tiwai@suse.de> | 2009-02-20 10:48:26 -0500 |
commit | 3be141494a080a9189b51fa78154c975ad8d9806 (patch) | |
tree | 17b62025f9616afe5938a1638195f9b1e9498f11 | |
parent | d2f57cd54a602c979e571aa337b206c88609876d (diff) |
ALSA: hda - Add generic pincfg initialization
Added the generic pincfg cache and save/restore functions.
Also introduced the pin-overriding via hwdep sysfs.
Signed-off-by: Takashi Iwai <tiwai@suse.de>
-rw-r--r-- | sound/pci/hda/hda_codec.c | 151 | ||||
-rw-r--r-- | sound/pci/hda/hda_codec.h | 15 | ||||
-rw-r--r-- | sound/pci/hda/hda_hwdep.c | 66 |
3 files changed, 223 insertions, 9 deletions
diff --git a/sound/pci/hda/hda_codec.c b/sound/pci/hda/hda_codec.c index 98884bc8f35f..6fa871f66a72 100644 --- a/sound/pci/hda/hda_codec.c +++ b/sound/pci/hda/hda_codec.c | |||
@@ -682,11 +682,132 @@ static int read_widget_caps(struct hda_codec *codec, hda_nid_t fg_node) | |||
682 | return 0; | 682 | return 0; |
683 | } | 683 | } |
684 | 684 | ||
685 | /* read all pin default configurations and save codec->init_pins */ | ||
686 | static int read_pin_defaults(struct hda_codec *codec) | ||
687 | { | ||
688 | int i; | ||
689 | hda_nid_t nid = codec->start_nid; | ||
690 | |||
691 | for (i = 0; i < codec->num_nodes; i++, nid++) { | ||
692 | struct hda_pincfg *pin; | ||
693 | unsigned int wcaps = get_wcaps(codec, nid); | ||
694 | unsigned int wid_type = (wcaps & AC_WCAP_TYPE) >> | ||
695 | AC_WCAP_TYPE_SHIFT; | ||
696 | if (wid_type != AC_WID_PIN) | ||
697 | continue; | ||
698 | pin = snd_array_new(&codec->init_pins); | ||
699 | if (!pin) | ||
700 | return -ENOMEM; | ||
701 | pin->nid = nid; | ||
702 | pin->cfg = snd_hda_codec_read(codec, nid, 0, | ||
703 | AC_VERB_GET_CONFIG_DEFAULT, 0); | ||
704 | } | ||
705 | return 0; | ||
706 | } | ||
707 | |||
708 | /* look up the given pin config list and return the item matching with NID */ | ||
709 | static struct hda_pincfg *look_up_pincfg(struct hda_codec *codec, | ||
710 | struct snd_array *array, | ||
711 | hda_nid_t nid) | ||
712 | { | ||
713 | int i; | ||
714 | for (i = 0; i < array->used; i++) { | ||
715 | struct hda_pincfg *pin = snd_array_elem(array, i); | ||
716 | if (pin->nid == nid) | ||
717 | return pin; | ||
718 | } | ||
719 | return NULL; | ||
720 | } | ||
721 | |||
722 | /* write a config value for the given NID */ | ||
723 | static void set_pincfg(struct hda_codec *codec, hda_nid_t nid, | ||
724 | unsigned int cfg) | ||
725 | { | ||
726 | int i; | ||
727 | for (i = 0; i < 4; i++) { | ||
728 | snd_hda_codec_write(codec, nid, 0, | ||
729 | AC_VERB_SET_CONFIG_DEFAULT_BYTES_0 + i, | ||
730 | cfg & 0xff); | ||
731 | cfg >>= 8; | ||
732 | } | ||
733 | } | ||
734 | |||
735 | /* set the current pin config value for the given NID. | ||
736 | * the value is cached, and read via snd_hda_codec_get_pincfg() | ||
737 | */ | ||
738 | int snd_hda_add_pincfg(struct hda_codec *codec, struct snd_array *list, | ||
739 | hda_nid_t nid, unsigned int cfg) | ||
740 | { | ||
741 | struct hda_pincfg *pin; | ||
742 | |||
743 | pin = look_up_pincfg(codec, list, nid); | ||
744 | if (!pin) { | ||
745 | pin = snd_array_new(list); | ||
746 | if (!pin) | ||
747 | return -ENOMEM; | ||
748 | pin->nid = nid; | ||
749 | } | ||
750 | pin->cfg = cfg; | ||
751 | set_pincfg(codec, nid, cfg); | ||
752 | return 0; | ||
753 | } | ||
754 | |||
755 | int snd_hda_codec_set_pincfg(struct hda_codec *codec, | ||
756 | hda_nid_t nid, unsigned int cfg) | ||
757 | { | ||
758 | return snd_hda_add_pincfg(codec, &codec->cur_pins, nid, cfg); | ||
759 | } | ||
760 | EXPORT_SYMBOL_HDA(snd_hda_codec_set_pincfg); | ||
761 | |||
762 | /* get the current pin config value of the given pin NID */ | ||
763 | unsigned int snd_hda_codec_get_pincfg(struct hda_codec *codec, hda_nid_t nid) | ||
764 | { | ||
765 | struct hda_pincfg *pin; | ||
766 | |||
767 | pin = look_up_pincfg(codec, &codec->cur_pins, nid); | ||
768 | if (pin) | ||
769 | return pin->cfg; | ||
770 | #ifdef CONFIG_SND_HDA_HWDEP | ||
771 | pin = look_up_pincfg(codec, &codec->override_pins, nid); | ||
772 | if (pin) | ||
773 | return pin->cfg; | ||
774 | #endif | ||
775 | pin = look_up_pincfg(codec, &codec->init_pins, nid); | ||
776 | if (pin) | ||
777 | return pin->cfg; | ||
778 | return 0; | ||
779 | } | ||
780 | EXPORT_SYMBOL_HDA(snd_hda_codec_get_pincfg); | ||
781 | |||
782 | /* restore all current pin configs */ | ||
783 | static void restore_pincfgs(struct hda_codec *codec) | ||
784 | { | ||
785 | int i; | ||
786 | for (i = 0; i < codec->init_pins.used; i++) { | ||
787 | struct hda_pincfg *pin = snd_array_elem(&codec->init_pins, i); | ||
788 | set_pincfg(codec, pin->nid, | ||
789 | snd_hda_codec_get_pincfg(codec, pin->nid)); | ||
790 | } | ||
791 | } | ||
685 | 792 | ||
686 | static void init_hda_cache(struct hda_cache_rec *cache, | 793 | static void init_hda_cache(struct hda_cache_rec *cache, |
687 | unsigned int record_size); | 794 | unsigned int record_size); |
688 | static void free_hda_cache(struct hda_cache_rec *cache); | 795 | static void free_hda_cache(struct hda_cache_rec *cache); |
689 | 796 | ||
797 | /* restore the initial pin cfgs and release all pincfg lists */ | ||
798 | static void restore_init_pincfgs(struct hda_codec *codec) | ||
799 | { | ||
800 | /* first free cur_pins and override_pins, then call restore_pincfg | ||
801 | * so that only the values in init_pins are restored | ||
802 | */ | ||
803 | snd_array_free(&codec->cur_pins); | ||
804 | #ifdef CONFIG_SND_HDA_HWDEP | ||
805 | snd_array_free(&codec->override_pins); | ||
806 | #endif | ||
807 | restore_pincfgs(codec); | ||
808 | snd_array_free(&codec->init_pins); | ||
809 | } | ||
810 | |||
690 | /* | 811 | /* |
691 | * codec destructor | 812 | * codec destructor |
692 | */ | 813 | */ |
@@ -694,6 +815,7 @@ static void snd_hda_codec_free(struct hda_codec *codec) | |||
694 | { | 815 | { |
695 | if (!codec) | 816 | if (!codec) |
696 | return; | 817 | return; |
818 | restore_init_pincfgs(codec); | ||
697 | #ifdef CONFIG_SND_HDA_POWER_SAVE | 819 | #ifdef CONFIG_SND_HDA_POWER_SAVE |
698 | cancel_delayed_work(&codec->power_work); | 820 | cancel_delayed_work(&codec->power_work); |
699 | flush_workqueue(codec->bus->workq); | 821 | flush_workqueue(codec->bus->workq); |
@@ -751,6 +873,8 @@ int /*__devinit*/ snd_hda_codec_new(struct hda_bus *bus, unsigned int codec_addr | |||
751 | init_hda_cache(&codec->amp_cache, sizeof(struct hda_amp_info)); | 873 | init_hda_cache(&codec->amp_cache, sizeof(struct hda_amp_info)); |
752 | init_hda_cache(&codec->cmd_cache, sizeof(struct hda_cache_head)); | 874 | init_hda_cache(&codec->cmd_cache, sizeof(struct hda_cache_head)); |
753 | snd_array_init(&codec->mixers, sizeof(struct snd_kcontrol *), 32); | 875 | snd_array_init(&codec->mixers, sizeof(struct snd_kcontrol *), 32); |
876 | snd_array_init(&codec->init_pins, sizeof(struct hda_pincfg), 16); | ||
877 | snd_array_init(&codec->cur_pins, sizeof(struct hda_pincfg), 16); | ||
754 | if (codec->bus->modelname) { | 878 | if (codec->bus->modelname) { |
755 | codec->modelname = kstrdup(codec->bus->modelname, GFP_KERNEL); | 879 | codec->modelname = kstrdup(codec->bus->modelname, GFP_KERNEL); |
756 | if (!codec->modelname) { | 880 | if (!codec->modelname) { |
@@ -787,15 +911,18 @@ int /*__devinit*/ snd_hda_codec_new(struct hda_bus *bus, unsigned int codec_addr | |||
787 | setup_fg_nodes(codec); | 911 | setup_fg_nodes(codec); |
788 | if (!codec->afg && !codec->mfg) { | 912 | if (!codec->afg && !codec->mfg) { |
789 | snd_printdd("hda_codec: no AFG or MFG node found\n"); | 913 | snd_printdd("hda_codec: no AFG or MFG node found\n"); |
790 | snd_hda_codec_free(codec); | 914 | err = -ENODEV; |
791 | return -ENODEV; | 915 | goto error; |
792 | } | 916 | } |
793 | 917 | ||
794 | if (read_widget_caps(codec, codec->afg ? codec->afg : codec->mfg) < 0) { | 918 | err = read_widget_caps(codec, codec->afg ? codec->afg : codec->mfg); |
919 | if (err < 0) { | ||
795 | snd_printk(KERN_ERR "hda_codec: cannot malloc\n"); | 920 | snd_printk(KERN_ERR "hda_codec: cannot malloc\n"); |
796 | snd_hda_codec_free(codec); | 921 | goto error; |
797 | return -ENOMEM; | ||
798 | } | 922 | } |
923 | err = read_pin_defaults(codec); | ||
924 | if (err < 0) | ||
925 | goto error; | ||
799 | 926 | ||
800 | if (!codec->subsystem_id) { | 927 | if (!codec->subsystem_id) { |
801 | hda_nid_t nid = codec->afg ? codec->afg : codec->mfg; | 928 | hda_nid_t nid = codec->afg ? codec->afg : codec->mfg; |
@@ -808,10 +935,8 @@ int /*__devinit*/ snd_hda_codec_new(struct hda_bus *bus, unsigned int codec_addr | |||
808 | 935 | ||
809 | if (do_init) { | 936 | if (do_init) { |
810 | err = snd_hda_codec_configure(codec); | 937 | err = snd_hda_codec_configure(codec); |
811 | if (err < 0) { | 938 | if (err < 0) |
812 | snd_hda_codec_free(codec); | 939 | goto error; |
813 | return err; | ||
814 | } | ||
815 | } | 940 | } |
816 | snd_hda_codec_proc_new(codec); | 941 | snd_hda_codec_proc_new(codec); |
817 | 942 | ||
@@ -824,6 +949,10 @@ int /*__devinit*/ snd_hda_codec_new(struct hda_bus *bus, unsigned int codec_addr | |||
824 | if (codecp) | 949 | if (codecp) |
825 | *codecp = codec; | 950 | *codecp = codec; |
826 | return 0; | 951 | return 0; |
952 | |||
953 | error: | ||
954 | snd_hda_codec_free(codec); | ||
955 | return err; | ||
827 | } | 956 | } |
828 | EXPORT_SYMBOL_HDA(snd_hda_codec_new); | 957 | EXPORT_SYMBOL_HDA(snd_hda_codec_new); |
829 | 958 | ||
@@ -1334,6 +1463,9 @@ void snd_hda_codec_reset(struct hda_codec *codec) | |||
1334 | free_hda_cache(&codec->cmd_cache); | 1463 | free_hda_cache(&codec->cmd_cache); |
1335 | init_hda_cache(&codec->amp_cache, sizeof(struct hda_amp_info)); | 1464 | init_hda_cache(&codec->amp_cache, sizeof(struct hda_amp_info)); |
1336 | init_hda_cache(&codec->cmd_cache, sizeof(struct hda_cache_head)); | 1465 | init_hda_cache(&codec->cmd_cache, sizeof(struct hda_cache_head)); |
1466 | /* free only cur_pins so that init_pins + override_pins are restored */ | ||
1467 | snd_array_free(&codec->cur_pins); | ||
1468 | restore_pincfgs(codec); | ||
1337 | codec->num_pcms = 0; | 1469 | codec->num_pcms = 0; |
1338 | codec->pcm_info = NULL; | 1470 | codec->pcm_info = NULL; |
1339 | codec->preset = NULL; | 1471 | codec->preset = NULL; |
@@ -2175,6 +2307,7 @@ static void hda_call_codec_resume(struct hda_codec *codec) | |||
2175 | hda_set_power_state(codec, | 2307 | hda_set_power_state(codec, |
2176 | codec->afg ? codec->afg : codec->mfg, | 2308 | codec->afg ? codec->afg : codec->mfg, |
2177 | AC_PWRST_D0); | 2309 | AC_PWRST_D0); |
2310 | restore_pincfgs(codec); /* restore all current pin configs */ | ||
2178 | hda_exec_init_verbs(codec); | 2311 | hda_exec_init_verbs(codec); |
2179 | if (codec->patch_ops.resume) | 2312 | if (codec->patch_ops.resume) |
2180 | codec->patch_ops.resume(codec); | 2313 | codec->patch_ops.resume(codec); |
diff --git a/sound/pci/hda/hda_codec.h b/sound/pci/hda/hda_codec.h index 09a332ada0c6..6d01a8058f0a 100644 --- a/sound/pci/hda/hda_codec.h +++ b/sound/pci/hda/hda_codec.h | |||
@@ -778,11 +778,14 @@ struct hda_codec { | |||
778 | unsigned short spdif_ctls; /* SPDIF control bits */ | 778 | unsigned short spdif_ctls; /* SPDIF control bits */ |
779 | unsigned int spdif_in_enable; /* SPDIF input enable? */ | 779 | unsigned int spdif_in_enable; /* SPDIF input enable? */ |
780 | hda_nid_t *slave_dig_outs; /* optional digital out slave widgets */ | 780 | hda_nid_t *slave_dig_outs; /* optional digital out slave widgets */ |
781 | struct snd_array init_pins; /* initial (BIOS) pin configurations */ | ||
782 | struct snd_array cur_pins; /* current pin configurations */ | ||
781 | 783 | ||
782 | #ifdef CONFIG_SND_HDA_HWDEP | 784 | #ifdef CONFIG_SND_HDA_HWDEP |
783 | struct snd_hwdep *hwdep; /* assigned hwdep device */ | 785 | struct snd_hwdep *hwdep; /* assigned hwdep device */ |
784 | struct snd_array init_verbs; /* additional init verbs */ | 786 | struct snd_array init_verbs; /* additional init verbs */ |
785 | struct snd_array hints; /* additional hints */ | 787 | struct snd_array hints; /* additional hints */ |
788 | struct snd_array override_pins; /* default pin configs to override */ | ||
786 | #endif | 789 | #endif |
787 | 790 | ||
788 | /* misc flags */ | 791 | /* misc flags */ |
@@ -855,6 +858,18 @@ void snd_hda_codec_resume_cache(struct hda_codec *codec); | |||
855 | #define snd_hda_sequence_write_cache snd_hda_sequence_write | 858 | #define snd_hda_sequence_write_cache snd_hda_sequence_write |
856 | #endif | 859 | #endif |
857 | 860 | ||
861 | /* the struct for codec->pin_configs */ | ||
862 | struct hda_pincfg { | ||
863 | hda_nid_t nid; | ||
864 | unsigned int cfg; | ||
865 | }; | ||
866 | |||
867 | unsigned int snd_hda_codec_get_pincfg(struct hda_codec *codec, hda_nid_t nid); | ||
868 | int snd_hda_codec_set_pincfg(struct hda_codec *codec, hda_nid_t nid, | ||
869 | unsigned int cfg); | ||
870 | int snd_hda_add_pincfg(struct hda_codec *codec, struct snd_array *list, | ||
871 | hda_nid_t nid, unsigned int cfg); /* for hwdep */ | ||
872 | |||
858 | /* | 873 | /* |
859 | * Mixer | 874 | * Mixer |
860 | */ | 875 | */ |
diff --git a/sound/pci/hda/hda_hwdep.c b/sound/pci/hda/hda_hwdep.c index 4ae51dcb81af..71039a6dec28 100644 --- a/sound/pci/hda/hda_hwdep.c +++ b/sound/pci/hda/hda_hwdep.c | |||
@@ -109,6 +109,7 @@ static void clear_hwdep_elements(struct hda_codec *codec) | |||
109 | for (i = 0; i < codec->hints.used; i++, head++) | 109 | for (i = 0; i < codec->hints.used; i++, head++) |
110 | kfree(*head); | 110 | kfree(*head); |
111 | snd_array_free(&codec->hints); | 111 | snd_array_free(&codec->hints); |
112 | snd_array_free(&codec->override_pins); | ||
112 | } | 113 | } |
113 | 114 | ||
114 | static void hwdep_free(struct snd_hwdep *hwdep) | 115 | static void hwdep_free(struct snd_hwdep *hwdep) |
@@ -141,6 +142,7 @@ int /*__devinit*/ snd_hda_create_hwdep(struct hda_codec *codec) | |||
141 | 142 | ||
142 | snd_array_init(&codec->init_verbs, sizeof(struct hda_verb), 32); | 143 | snd_array_init(&codec->init_verbs, sizeof(struct hda_verb), 32); |
143 | snd_array_init(&codec->hints, sizeof(char *), 32); | 144 | snd_array_init(&codec->hints, sizeof(char *), 32); |
145 | snd_array_init(&codec->override_pins, sizeof(struct hda_pincfg), 16); | ||
144 | 146 | ||
145 | return 0; | 147 | return 0; |
146 | } | 148 | } |
@@ -316,6 +318,67 @@ static ssize_t hints_store(struct device *dev, | |||
316 | return count; | 318 | return count; |
317 | } | 319 | } |
318 | 320 | ||
321 | static ssize_t pin_configs_show(struct hda_codec *codec, | ||
322 | struct snd_array *list, | ||
323 | char *buf) | ||
324 | { | ||
325 | int i, len = 0; | ||
326 | for (i = 0; i < list->used; i++) { | ||
327 | struct hda_pincfg *pin = snd_array_elem(list, i); | ||
328 | len += sprintf(buf + len, "0x%02x 0x%08x\n", | ||
329 | pin->nid, pin->cfg); | ||
330 | } | ||
331 | return len; | ||
332 | } | ||
333 | |||
334 | static ssize_t init_pin_configs_show(struct device *dev, | ||
335 | struct device_attribute *attr, | ||
336 | char *buf) | ||
337 | { | ||
338 | struct snd_hwdep *hwdep = dev_get_drvdata(dev); | ||
339 | struct hda_codec *codec = hwdep->private_data; | ||
340 | return pin_configs_show(codec, &codec->init_pins, buf); | ||
341 | } | ||
342 | |||
343 | static ssize_t override_pin_configs_show(struct device *dev, | ||
344 | struct device_attribute *attr, | ||
345 | char *buf) | ||
346 | { | ||
347 | struct snd_hwdep *hwdep = dev_get_drvdata(dev); | ||
348 | struct hda_codec *codec = hwdep->private_data; | ||
349 | return pin_configs_show(codec, &codec->override_pins, buf); | ||
350 | } | ||
351 | |||
352 | static ssize_t cur_pin_configs_show(struct device *dev, | ||
353 | struct device_attribute *attr, | ||
354 | char *buf) | ||
355 | { | ||
356 | struct snd_hwdep *hwdep = dev_get_drvdata(dev); | ||
357 | struct hda_codec *codec = hwdep->private_data; | ||
358 | return pin_configs_show(codec, &codec->cur_pins, buf); | ||
359 | } | ||
360 | |||
361 | #define MAX_PIN_CONFIGS 32 | ||
362 | |||
363 | static ssize_t override_pin_configs_store(struct device *dev, | ||
364 | struct device_attribute *attr, | ||
365 | const char *buf, size_t count) | ||
366 | { | ||
367 | struct snd_hwdep *hwdep = dev_get_drvdata(dev); | ||
368 | struct hda_codec *codec = hwdep->private_data; | ||
369 | int nid, cfg; | ||
370 | int err; | ||
371 | |||
372 | if (sscanf(buf, "%i %i", &nid, &cfg) != 2) | ||
373 | return -EINVAL; | ||
374 | if (!nid) | ||
375 | return -EINVAL; | ||
376 | err = snd_hda_add_pincfg(codec, &codec->override_pins, nid, cfg); | ||
377 | if (err < 0) | ||
378 | return err; | ||
379 | return count; | ||
380 | } | ||
381 | |||
319 | #define CODEC_ATTR_RW(type) \ | 382 | #define CODEC_ATTR_RW(type) \ |
320 | __ATTR(type, 0644, type##_show, type##_store) | 383 | __ATTR(type, 0644, type##_show, type##_store) |
321 | #define CODEC_ATTR_RO(type) \ | 384 | #define CODEC_ATTR_RO(type) \ |
@@ -333,6 +396,9 @@ static struct device_attribute codec_attrs[] = { | |||
333 | CODEC_ATTR_RW(modelname), | 396 | CODEC_ATTR_RW(modelname), |
334 | CODEC_ATTR_WO(init_verbs), | 397 | CODEC_ATTR_WO(init_verbs), |
335 | CODEC_ATTR_WO(hints), | 398 | CODEC_ATTR_WO(hints), |
399 | CODEC_ATTR_RO(init_pin_configs), | ||
400 | CODEC_ATTR_RW(override_pin_configs), | ||
401 | CODEC_ATTR_RO(cur_pin_configs), | ||
336 | CODEC_ATTR_WO(reconfig), | 402 | CODEC_ATTR_WO(reconfig), |
337 | CODEC_ATTR_WO(clear), | 403 | CODEC_ATTR_WO(clear), |
338 | }; | 404 | }; |