diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2013-03-15 20:35:49 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2013-03-15 20:35:49 -0400 |
| commit | 5cd8846c3b7c104135ee602ab1887f4c1de445ef (patch) | |
| tree | 409f0b7b82761c1e09a566736ea929e04590c474 | |
| parent | c7f17deb316e41a9db28d7486f4067d06d68ebf0 (diff) | |
| parent | 6d3073e124e1a6138b929479301d3a7ecde00f27 (diff) | |
Merge tag 'sound-3.9' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound
Pull sound fixes from Takashi Iwai:
"A collection of small fixes, as expected for the middle rc:
- A couple of fixes for potential NULL dereferences and out-of-range
array accesses revealed by static code parsers
- A fix for the wrong error handling detected by trinity
- A regression fix for missing audio on some MacBooks
- CA0132 DSP loader fixes
- Fix for EAPD control of IDT codecs on machines w/o speaker
- Fix a regression in the HD-audio widget list parser code
- Workaround for the NuForce UDH-100 USB audio"
* tag 'sound-3.9' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound:
ALSA: hda - Fix missing EAPD/GPIO setup for Cirrus codecs
sound: sequencer: cap array index in seq_chn_common_event()
ALSA: hda/ca0132 - Remove extra setting of dsp_state.
ALSA: hda/ca0132 - Check download state of DSP.
ALSA: hda/ca0132 - Check if dspload_image succeeded.
ALSA: hda - Disable IDT eapd_switch if there are no internal speakers
ALSA: hda - Fix snd_hda_get_num_raw_conns() to return a correct value
ALSA: usb-audio: add a workaround for the NuForce UDH-100
ALSA: asihpi - fix potential NULL pointer dereference
ALSA: seq: Fix missing error handling in snd_seq_timer_open()
| -rw-r--r-- | sound/core/seq/seq_timer.c | 8 | ||||
| -rw-r--r-- | sound/oss/sequencer.c | 6 | ||||
| -rw-r--r-- | sound/pci/asihpi/asihpi.c | 3 | ||||
| -rw-r--r-- | sound/pci/hda/hda_codec.c | 24 | ||||
| -rw-r--r-- | sound/pci/hda/patch_ca0132.c | 28 | ||||
| -rw-r--r-- | sound/pci/hda/patch_cirrus.c | 4 | ||||
| -rw-r--r-- | sound/pci/hda/patch_sigmatel.c | 29 | ||||
| -rw-r--r-- | sound/usb/card.c | 15 |
8 files changed, 89 insertions, 28 deletions
diff --git a/sound/core/seq/seq_timer.c b/sound/core/seq/seq_timer.c index 160b1bd0cd62..24d44b2f61ac 100644 --- a/sound/core/seq/seq_timer.c +++ b/sound/core/seq/seq_timer.c | |||
| @@ -290,10 +290,10 @@ int snd_seq_timer_open(struct snd_seq_queue *q) | |||
| 290 | tid.device = SNDRV_TIMER_GLOBAL_SYSTEM; | 290 | tid.device = SNDRV_TIMER_GLOBAL_SYSTEM; |
| 291 | err = snd_timer_open(&t, str, &tid, q->queue); | 291 | err = snd_timer_open(&t, str, &tid, q->queue); |
| 292 | } | 292 | } |
| 293 | if (err < 0) { | 293 | } |
| 294 | snd_printk(KERN_ERR "seq fatal error: cannot create timer (%i)\n", err); | 294 | if (err < 0) { |
| 295 | return err; | 295 | snd_printk(KERN_ERR "seq fatal error: cannot create timer (%i)\n", err); |
| 296 | } | 296 | return err; |
| 297 | } | 297 | } |
| 298 | t->callback = snd_seq_timer_interrupt; | 298 | t->callback = snd_seq_timer_interrupt; |
| 299 | t->callback_data = q; | 299 | t->callback_data = q; |
diff --git a/sound/oss/sequencer.c b/sound/oss/sequencer.c index 30bcfe470f83..4ff60a6427d9 100644 --- a/sound/oss/sequencer.c +++ b/sound/oss/sequencer.c | |||
| @@ -545,6 +545,9 @@ static void seq_chn_common_event(unsigned char *event_rec) | |||
| 545 | case MIDI_PGM_CHANGE: | 545 | case MIDI_PGM_CHANGE: |
| 546 | if (seq_mode == SEQ_2) | 546 | if (seq_mode == SEQ_2) |
| 547 | { | 547 | { |
| 548 | if (chn > 15) | ||
| 549 | break; | ||
| 550 | |||
| 548 | synth_devs[dev]->chn_info[chn].pgm_num = p1; | 551 | synth_devs[dev]->chn_info[chn].pgm_num = p1; |
| 549 | if ((int) dev >= num_synths) | 552 | if ((int) dev >= num_synths) |
| 550 | synth_devs[dev]->set_instr(dev, chn, p1); | 553 | synth_devs[dev]->set_instr(dev, chn, p1); |
| @@ -596,6 +599,9 @@ static void seq_chn_common_event(unsigned char *event_rec) | |||
| 596 | case MIDI_PITCH_BEND: | 599 | case MIDI_PITCH_BEND: |
| 597 | if (seq_mode == SEQ_2) | 600 | if (seq_mode == SEQ_2) |
| 598 | { | 601 | { |
| 602 | if (chn > 15) | ||
| 603 | break; | ||
| 604 | |||
| 599 | synth_devs[dev]->chn_info[chn].bender_value = w14; | 605 | synth_devs[dev]->chn_info[chn].bender_value = w14; |
| 600 | 606 | ||
| 601 | if ((int) dev < num_synths) | 607 | if ((int) dev < num_synths) |
diff --git a/sound/pci/asihpi/asihpi.c b/sound/pci/asihpi/asihpi.c index 3536b076b529..0aabfedeecba 100644 --- a/sound/pci/asihpi/asihpi.c +++ b/sound/pci/asihpi/asihpi.c | |||
| @@ -2549,7 +2549,7 @@ static int snd_asihpi_sampleclock_add(struct snd_card_asihpi *asihpi, | |||
| 2549 | 2549 | ||
| 2550 | static int snd_card_asihpi_mixer_new(struct snd_card_asihpi *asihpi) | 2550 | static int snd_card_asihpi_mixer_new(struct snd_card_asihpi *asihpi) |
| 2551 | { | 2551 | { |
| 2552 | struct snd_card *card = asihpi->card; | 2552 | struct snd_card *card; |
| 2553 | unsigned int idx = 0; | 2553 | unsigned int idx = 0; |
| 2554 | unsigned int subindex = 0; | 2554 | unsigned int subindex = 0; |
| 2555 | int err; | 2555 | int err; |
| @@ -2557,6 +2557,7 @@ static int snd_card_asihpi_mixer_new(struct snd_card_asihpi *asihpi) | |||
| 2557 | 2557 | ||
| 2558 | if (snd_BUG_ON(!asihpi)) | 2558 | if (snd_BUG_ON(!asihpi)) |
| 2559 | return -EINVAL; | 2559 | return -EINVAL; |
| 2560 | card = asihpi->card; | ||
| 2560 | strcpy(card->mixername, "Asihpi Mixer"); | 2561 | strcpy(card->mixername, "Asihpi Mixer"); |
| 2561 | 2562 | ||
| 2562 | err = | 2563 | err = |
diff --git a/sound/pci/hda/hda_codec.c b/sound/pci/hda/hda_codec.c index 97c68dd24ef5..a9ebcf9e3710 100644 --- a/sound/pci/hda/hda_codec.c +++ b/sound/pci/hda/hda_codec.c | |||
| @@ -494,7 +494,7 @@ static unsigned int get_num_conns(struct hda_codec *codec, hda_nid_t nid) | |||
| 494 | 494 | ||
| 495 | int snd_hda_get_num_raw_conns(struct hda_codec *codec, hda_nid_t nid) | 495 | int snd_hda_get_num_raw_conns(struct hda_codec *codec, hda_nid_t nid) |
| 496 | { | 496 | { |
| 497 | return get_num_conns(codec, nid) & AC_CLIST_LENGTH; | 497 | return snd_hda_get_raw_connections(codec, nid, NULL, 0); |
| 498 | } | 498 | } |
| 499 | 499 | ||
| 500 | /** | 500 | /** |
| @@ -517,9 +517,6 @@ int snd_hda_get_raw_connections(struct hda_codec *codec, hda_nid_t nid, | |||
| 517 | hda_nid_t prev_nid; | 517 | hda_nid_t prev_nid; |
| 518 | int null_count = 0; | 518 | int null_count = 0; |
| 519 | 519 | ||
| 520 | if (snd_BUG_ON(!conn_list || max_conns <= 0)) | ||
| 521 | return -EINVAL; | ||
| 522 | |||
| 523 | parm = get_num_conns(codec, nid); | 520 | parm = get_num_conns(codec, nid); |
| 524 | if (!parm) | 521 | if (!parm) |
| 525 | return 0; | 522 | return 0; |
| @@ -545,7 +542,8 @@ int snd_hda_get_raw_connections(struct hda_codec *codec, hda_nid_t nid, | |||
| 545 | AC_VERB_GET_CONNECT_LIST, 0); | 542 | AC_VERB_GET_CONNECT_LIST, 0); |
| 546 | if (parm == -1 && codec->bus->rirb_error) | 543 | if (parm == -1 && codec->bus->rirb_error) |
| 547 | return -EIO; | 544 | return -EIO; |
| 548 | conn_list[0] = parm & mask; | 545 | if (conn_list) |
| 546 | conn_list[0] = parm & mask; | ||
| 549 | return 1; | 547 | return 1; |
| 550 | } | 548 | } |
| 551 | 549 | ||
| @@ -580,14 +578,20 @@ int snd_hda_get_raw_connections(struct hda_codec *codec, hda_nid_t nid, | |||
| 580 | continue; | 578 | continue; |
| 581 | } | 579 | } |
| 582 | for (n = prev_nid + 1; n <= val; n++) { | 580 | for (n = prev_nid + 1; n <= val; n++) { |
| 581 | if (conn_list) { | ||
| 582 | if (conns >= max_conns) | ||
| 583 | return -ENOSPC; | ||
| 584 | conn_list[conns] = n; | ||
| 585 | } | ||
| 586 | conns++; | ||
| 587 | } | ||
| 588 | } else { | ||
| 589 | if (conn_list) { | ||
| 583 | if (conns >= max_conns) | 590 | if (conns >= max_conns) |
| 584 | return -ENOSPC; | 591 | return -ENOSPC; |
| 585 | conn_list[conns++] = n; | 592 | conn_list[conns] = val; |
| 586 | } | 593 | } |
| 587 | } else { | 594 | conns++; |
| 588 | if (conns >= max_conns) | ||
| 589 | return -ENOSPC; | ||
| 590 | conn_list[conns++] = val; | ||
| 591 | } | 595 | } |
| 592 | prev_nid = val; | 596 | prev_nid = val; |
| 593 | } | 597 | } |
diff --git a/sound/pci/hda/patch_ca0132.c b/sound/pci/hda/patch_ca0132.c index eefc4563b2f9..0792b5725f9c 100644 --- a/sound/pci/hda/patch_ca0132.c +++ b/sound/pci/hda/patch_ca0132.c | |||
| @@ -3239,7 +3239,7 @@ static int ca0132_set_vipsource(struct hda_codec *codec, int val) | |||
| 3239 | struct ca0132_spec *spec = codec->spec; | 3239 | struct ca0132_spec *spec = codec->spec; |
| 3240 | unsigned int tmp; | 3240 | unsigned int tmp; |
| 3241 | 3241 | ||
| 3242 | if (!dspload_is_loaded(codec)) | 3242 | if (spec->dsp_state != DSP_DOWNLOADED) |
| 3243 | return 0; | 3243 | return 0; |
| 3244 | 3244 | ||
| 3245 | /* if CrystalVoice if off, vipsource should be 0 */ | 3245 | /* if CrystalVoice if off, vipsource should be 0 */ |
| @@ -4267,11 +4267,12 @@ static void ca0132_refresh_widget_caps(struct hda_codec *codec) | |||
| 4267 | */ | 4267 | */ |
| 4268 | static void ca0132_setup_defaults(struct hda_codec *codec) | 4268 | static void ca0132_setup_defaults(struct hda_codec *codec) |
| 4269 | { | 4269 | { |
| 4270 | struct ca0132_spec *spec = codec->spec; | ||
| 4270 | unsigned int tmp; | 4271 | unsigned int tmp; |
| 4271 | int num_fx; | 4272 | int num_fx; |
| 4272 | int idx, i; | 4273 | int idx, i; |
| 4273 | 4274 | ||
| 4274 | if (!dspload_is_loaded(codec)) | 4275 | if (spec->dsp_state != DSP_DOWNLOADED) |
| 4275 | return; | 4276 | return; |
| 4276 | 4277 | ||
| 4277 | /* out, in effects + voicefx */ | 4278 | /* out, in effects + voicefx */ |
| @@ -4351,12 +4352,16 @@ static bool ca0132_download_dsp_images(struct hda_codec *codec) | |||
| 4351 | return false; | 4352 | return false; |
| 4352 | 4353 | ||
| 4353 | dsp_os_image = (struct dsp_image_seg *)(fw_entry->data); | 4354 | dsp_os_image = (struct dsp_image_seg *)(fw_entry->data); |
| 4354 | dspload_image(codec, dsp_os_image, 0, 0, true, 0); | 4355 | if (dspload_image(codec, dsp_os_image, 0, 0, true, 0)) { |
| 4356 | pr_err("ca0132 dspload_image failed.\n"); | ||
| 4357 | goto exit_download; | ||
| 4358 | } | ||
| 4359 | |||
| 4355 | dsp_loaded = dspload_wait_loaded(codec); | 4360 | dsp_loaded = dspload_wait_loaded(codec); |
| 4356 | 4361 | ||
| 4362 | exit_download: | ||
| 4357 | release_firmware(fw_entry); | 4363 | release_firmware(fw_entry); |
| 4358 | 4364 | ||
| 4359 | |||
| 4360 | return dsp_loaded; | 4365 | return dsp_loaded; |
| 4361 | } | 4366 | } |
| 4362 | 4367 | ||
| @@ -4367,16 +4372,13 @@ static void ca0132_download_dsp(struct hda_codec *codec) | |||
| 4367 | #ifndef CONFIG_SND_HDA_CODEC_CA0132_DSP | 4372 | #ifndef CONFIG_SND_HDA_CODEC_CA0132_DSP |
| 4368 | return; /* NOP */ | 4373 | return; /* NOP */ |
| 4369 | #endif | 4374 | #endif |
| 4370 | spec->dsp_state = DSP_DOWNLOAD_INIT; | ||
| 4371 | 4375 | ||
| 4372 | if (spec->dsp_state == DSP_DOWNLOAD_INIT) { | 4376 | chipio_enable_clocks(codec); |
| 4373 | chipio_enable_clocks(codec); | 4377 | spec->dsp_state = DSP_DOWNLOADING; |
| 4374 | spec->dsp_state = DSP_DOWNLOADING; | 4378 | if (!ca0132_download_dsp_images(codec)) |
| 4375 | if (!ca0132_download_dsp_images(codec)) | 4379 | spec->dsp_state = DSP_DOWNLOAD_FAILED; |
| 4376 | spec->dsp_state = DSP_DOWNLOAD_FAILED; | 4380 | else |
| 4377 | else | 4381 | spec->dsp_state = DSP_DOWNLOADED; |
| 4378 | spec->dsp_state = DSP_DOWNLOADED; | ||
| 4379 | } | ||
| 4380 | 4382 | ||
| 4381 | if (spec->dsp_state == DSP_DOWNLOADED) | 4383 | if (spec->dsp_state == DSP_DOWNLOADED) |
| 4382 | ca0132_set_dsp_msr(codec, true); | 4384 | ca0132_set_dsp_msr(codec, true); |
diff --git a/sound/pci/hda/patch_cirrus.c b/sound/pci/hda/patch_cirrus.c index 72ebb8a36b13..60d08f669f0c 100644 --- a/sound/pci/hda/patch_cirrus.c +++ b/sound/pci/hda/patch_cirrus.c | |||
| @@ -506,6 +506,8 @@ static int patch_cs420x(struct hda_codec *codec) | |||
| 506 | if (!spec) | 506 | if (!spec) |
| 507 | return -ENOMEM; | 507 | return -ENOMEM; |
| 508 | 508 | ||
| 509 | spec->gen.automute_hook = cs_automute; | ||
| 510 | |||
| 509 | snd_hda_pick_fixup(codec, cs420x_models, cs420x_fixup_tbl, | 511 | snd_hda_pick_fixup(codec, cs420x_models, cs420x_fixup_tbl, |
| 510 | cs420x_fixups); | 512 | cs420x_fixups); |
| 511 | snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE); | 513 | snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE); |
| @@ -893,6 +895,8 @@ static int patch_cs4210(struct hda_codec *codec) | |||
| 893 | if (!spec) | 895 | if (!spec) |
| 894 | return -ENOMEM; | 896 | return -ENOMEM; |
| 895 | 897 | ||
| 898 | spec->gen.automute_hook = cs_automute; | ||
| 899 | |||
| 896 | snd_hda_pick_fixup(codec, cs421x_models, cs421x_fixup_tbl, | 900 | snd_hda_pick_fixup(codec, cs421x_models, cs421x_fixup_tbl, |
| 897 | cs421x_fixups); | 901 | cs421x_fixups); |
| 898 | snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE); | 902 | snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE); |
diff --git a/sound/pci/hda/patch_sigmatel.c b/sound/pci/hda/patch_sigmatel.c index 83d5335ac348..dafe04ae8c72 100644 --- a/sound/pci/hda/patch_sigmatel.c +++ b/sound/pci/hda/patch_sigmatel.c | |||
| @@ -815,6 +815,29 @@ static int find_mute_led_cfg(struct hda_codec *codec, int default_polarity) | |||
| 815 | return 0; | 815 | return 0; |
| 816 | } | 816 | } |
| 817 | 817 | ||
| 818 | /* check whether a built-in speaker is included in parsed pins */ | ||
| 819 | static bool has_builtin_speaker(struct hda_codec *codec) | ||
| 820 | { | ||
| 821 | struct sigmatel_spec *spec = codec->spec; | ||
| 822 | hda_nid_t *nid_pin; | ||
| 823 | int nids, i; | ||
| 824 | |||
| 825 | if (spec->gen.autocfg.line_out_type == AUTO_PIN_SPEAKER_OUT) { | ||
| 826 | nid_pin = spec->gen.autocfg.line_out_pins; | ||
| 827 | nids = spec->gen.autocfg.line_outs; | ||
| 828 | } else { | ||
| 829 | nid_pin = spec->gen.autocfg.speaker_pins; | ||
| 830 | nids = spec->gen.autocfg.speaker_outs; | ||
| 831 | } | ||
| 832 | |||
| 833 | for (i = 0; i < nids; i++) { | ||
| 834 | unsigned int def_conf = snd_hda_codec_get_pincfg(codec, nid_pin[i]); | ||
| 835 | if (snd_hda_get_input_pin_attr(def_conf) == INPUT_PIN_ATTR_INT) | ||
| 836 | return true; | ||
| 837 | } | ||
| 838 | return false; | ||
| 839 | } | ||
| 840 | |||
| 818 | /* | 841 | /* |
| 819 | * PC beep controls | 842 | * PC beep controls |
| 820 | */ | 843 | */ |
| @@ -3890,6 +3913,12 @@ static int patch_stac92hd73xx(struct hda_codec *codec) | |||
| 3890 | return err; | 3913 | return err; |
| 3891 | } | 3914 | } |
| 3892 | 3915 | ||
| 3916 | /* Don't GPIO-mute speakers if there are no internal speakers, because | ||
| 3917 | * the GPIO might be necessary for Headphone | ||
| 3918 | */ | ||
| 3919 | if (spec->eapd_switch && !has_builtin_speaker(codec)) | ||
| 3920 | spec->eapd_switch = 0; | ||
| 3921 | |||
| 3893 | codec->proc_widget_hook = stac92hd7x_proc_hook; | 3922 | codec->proc_widget_hook = stac92hd7x_proc_hook; |
| 3894 | 3923 | ||
| 3895 | snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE); | 3924 | snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE); |
diff --git a/sound/usb/card.c b/sound/usb/card.c index 803953a9bff3..2da8ad75fd96 100644 --- a/sound/usb/card.c +++ b/sound/usb/card.c | |||
| @@ -244,6 +244,21 @@ static int snd_usb_create_streams(struct snd_usb_audio *chip, int ctrlif) | |||
| 244 | usb_ifnum_to_if(dev, ctrlif)->intf_assoc; | 244 | usb_ifnum_to_if(dev, ctrlif)->intf_assoc; |
| 245 | 245 | ||
| 246 | if (!assoc) { | 246 | if (!assoc) { |
| 247 | /* | ||
| 248 | * Firmware writers cannot count to three. So to find | ||
| 249 | * the IAD on the NuForce UDH-100, also check the next | ||
| 250 | * interface. | ||
| 251 | */ | ||
| 252 | struct usb_interface *iface = | ||
| 253 | usb_ifnum_to_if(dev, ctrlif + 1); | ||
| 254 | if (iface && | ||
| 255 | iface->intf_assoc && | ||
| 256 | iface->intf_assoc->bFunctionClass == USB_CLASS_AUDIO && | ||
| 257 | iface->intf_assoc->bFunctionProtocol == UAC_VERSION_2) | ||
| 258 | assoc = iface->intf_assoc; | ||
| 259 | } | ||
| 260 | |||
| 261 | if (!assoc) { | ||
| 247 | snd_printk(KERN_ERR "Audio class v2 interfaces need an interface association\n"); | 262 | snd_printk(KERN_ERR "Audio class v2 interfaces need an interface association\n"); |
| 248 | return -EINVAL; | 263 | return -EINVAL; |
| 249 | } | 264 | } |
