diff options
author | Ondrej Zary <linux@rainbow-software.org> | 2011-03-19 11:33:14 -0400 |
---|---|---|
committer | Takashi Iwai <tiwai@suse.de> | 2011-03-21 07:44:14 -0400 |
commit | 1872f589951caee1afd7cd2ea6729ac892de9ddf (patch) | |
tree | 11acde5b1d3e360141828014d6d9ec865816073d /sound/pci/es1968.c | |
parent | f8960d61bc8ba945b07a4de1288aac5d52f8607b (diff) |
ALSA: es1968: add radio (tea575x tuner) support
Add TEA5757 radio tuner support to es1968 driver. This is found at least on
MediaForte SF64-PCE2 sound cards.
Signed-off-by: Ondrej Zary <linux@rainbow-software.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Diffstat (limited to 'sound/pci/es1968.c')
-rw-r--r-- | sound/pci/es1968.c | 125 |
1 files changed, 125 insertions, 0 deletions
diff --git a/sound/pci/es1968.c b/sound/pci/es1968.c index 7c17f45d876d..faf9138970ae 100644 --- a/sound/pci/es1968.c +++ b/sound/pci/es1968.c | |||
@@ -112,6 +112,10 @@ | |||
112 | #include <sound/ac97_codec.h> | 112 | #include <sound/ac97_codec.h> |
113 | #include <sound/initval.h> | 113 | #include <sound/initval.h> |
114 | 114 | ||
115 | #ifdef CONFIG_SND_ES1968_RADIO | ||
116 | #include <sound/tea575x-tuner.h> | ||
117 | #endif | ||
118 | |||
115 | #define CARD_NAME "ESS Maestro1/2" | 119 | #define CARD_NAME "ESS Maestro1/2" |
116 | #define DRIVER_NAME "ES1968" | 120 | #define DRIVER_NAME "ES1968" |
117 | 121 | ||
@@ -553,6 +557,10 @@ struct es1968 { | |||
553 | spinlock_t ac97_lock; | 557 | spinlock_t ac97_lock; |
554 | struct tasklet_struct hwvol_tq; | 558 | struct tasklet_struct hwvol_tq; |
555 | #endif | 559 | #endif |
560 | |||
561 | #ifdef CONFIG_SND_ES1968_RADIO | ||
562 | struct snd_tea575x tea; | ||
563 | #endif | ||
556 | }; | 564 | }; |
557 | 565 | ||
558 | static irqreturn_t snd_es1968_interrupt(int irq, void *dev_id); | 566 | static irqreturn_t snd_es1968_interrupt(int irq, void *dev_id); |
@@ -2571,6 +2579,111 @@ static int __devinit snd_es1968_input_register(struct es1968 *chip) | |||
2571 | } | 2579 | } |
2572 | #endif /* CONFIG_SND_ES1968_INPUT */ | 2580 | #endif /* CONFIG_SND_ES1968_INPUT */ |
2573 | 2581 | ||
2582 | #ifdef CONFIG_SND_ES1968_RADIO | ||
2583 | #define GPIO_DATA 0x60 | ||
2584 | #define IO_MASK 4 /* mask register offset from GPIO_DATA | ||
2585 | bits 1=unmask write to given bit */ | ||
2586 | #define IO_DIR 8 /* direction register offset from GPIO_DATA | ||
2587 | bits 0/1=read/write direction */ | ||
2588 | /* mask bits for GPIO lines */ | ||
2589 | #define STR_DATA 0x0040 /* GPIO6 */ | ||
2590 | #define STR_CLK 0x0080 /* GPIO7 */ | ||
2591 | #define STR_WREN 0x0100 /* GPIO8 */ | ||
2592 | #define STR_MOST 0x0200 /* GPIO9 */ | ||
2593 | |||
2594 | static void snd_es1968_tea575x_write(struct snd_tea575x *tea, unsigned int val) | ||
2595 | { | ||
2596 | struct es1968 *chip = tea->private_data; | ||
2597 | unsigned long io = chip->io_port + GPIO_DATA; | ||
2598 | u16 l, bits; | ||
2599 | u16 omask, odir; | ||
2600 | |||
2601 | omask = inw(io + IO_MASK); | ||
2602 | odir = (inw(io + IO_DIR) & ~STR_DATA) | (STR_CLK | STR_WREN); | ||
2603 | outw(odir | STR_DATA, io + IO_DIR); | ||
2604 | outw(~(STR_DATA | STR_CLK | STR_WREN), io + IO_MASK); | ||
2605 | udelay(16); | ||
2606 | |||
2607 | for (l = 25; l; l--) { | ||
2608 | bits = ((val >> 18) & STR_DATA) | STR_WREN; | ||
2609 | val <<= 1; /* shift data */ | ||
2610 | outw(bits, io); /* start strobe */ | ||
2611 | udelay(2); | ||
2612 | outw(bits | STR_CLK, io); /* HI level */ | ||
2613 | udelay(2); | ||
2614 | outw(bits, io); /* LO level */ | ||
2615 | udelay(4); | ||
2616 | } | ||
2617 | |||
2618 | if (!tea->mute) | ||
2619 | outw(0, io); | ||
2620 | |||
2621 | udelay(4); | ||
2622 | outw(omask, io + IO_MASK); | ||
2623 | outw(odir, io + IO_DIR); | ||
2624 | msleep(125); | ||
2625 | } | ||
2626 | |||
2627 | static unsigned int snd_es1968_tea575x_read(struct snd_tea575x *tea) | ||
2628 | { | ||
2629 | struct es1968 *chip = tea->private_data; | ||
2630 | unsigned long io = chip->io_port + GPIO_DATA; | ||
2631 | u16 l, rdata; | ||
2632 | u32 data = 0; | ||
2633 | u16 omask; | ||
2634 | |||
2635 | omask = inw(io + IO_MASK); | ||
2636 | outw(~(STR_CLK | STR_WREN), io + IO_MASK); | ||
2637 | outw(0, io); | ||
2638 | udelay(16); | ||
2639 | |||
2640 | for (l = 24; l--;) { | ||
2641 | outw(STR_CLK, io); /* HI state */ | ||
2642 | udelay(2); | ||
2643 | if (!l) | ||
2644 | tea->tuned = inw(io) & STR_MOST ? 0 : 1; | ||
2645 | outw(0, io); /* LO state */ | ||
2646 | udelay(2); | ||
2647 | data <<= 1; /* shift data */ | ||
2648 | rdata = inw(io); | ||
2649 | if (!l) | ||
2650 | tea->stereo = (rdata & STR_MOST) ? 0 : 1; | ||
2651 | else if (l && rdata & STR_DATA) | ||
2652 | data++; | ||
2653 | udelay(2); | ||
2654 | } | ||
2655 | |||
2656 | if (tea->mute) | ||
2657 | outw(STR_WREN, io); | ||
2658 | |||
2659 | udelay(4); | ||
2660 | outw(omask, io + IO_MASK); | ||
2661 | |||
2662 | return data & 0x3ffe; | ||
2663 | } | ||
2664 | |||
2665 | static void snd_es1968_tea575x_mute(struct snd_tea575x *tea, unsigned int mute) | ||
2666 | { | ||
2667 | struct es1968 *chip = tea->private_data; | ||
2668 | unsigned long io = chip->io_port + GPIO_DATA; | ||
2669 | u16 omask; | ||
2670 | |||
2671 | omask = inw(io + IO_MASK); | ||
2672 | outw(~STR_WREN, io + IO_MASK); | ||
2673 | tea->mute = mute; | ||
2674 | outw(tea->mute ? STR_WREN : 0, io); | ||
2675 | udelay(4); | ||
2676 | outw(omask, io + IO_MASK); | ||
2677 | msleep(125); | ||
2678 | } | ||
2679 | |||
2680 | static struct snd_tea575x_ops snd_es1968_tea_ops = { | ||
2681 | .write = snd_es1968_tea575x_write, | ||
2682 | .read = snd_es1968_tea575x_read, | ||
2683 | .mute = snd_es1968_tea575x_mute, | ||
2684 | }; | ||
2685 | #endif | ||
2686 | |||
2574 | static int snd_es1968_free(struct es1968 *chip) | 2687 | static int snd_es1968_free(struct es1968 *chip) |
2575 | { | 2688 | { |
2576 | #ifdef CONFIG_SND_ES1968_INPUT | 2689 | #ifdef CONFIG_SND_ES1968_INPUT |
@@ -2585,6 +2698,10 @@ static int snd_es1968_free(struct es1968 *chip) | |||
2585 | outw(0, chip->io_port + ESM_PORT_HOST_IRQ); /* disable IRQ */ | 2698 | outw(0, chip->io_port + ESM_PORT_HOST_IRQ); /* disable IRQ */ |
2586 | } | 2699 | } |
2587 | 2700 | ||
2701 | #ifdef CONFIG_SND_ES1968_RADIO | ||
2702 | snd_tea575x_exit(&chip->tea); | ||
2703 | #endif | ||
2704 | |||
2588 | if (chip->irq >= 0) | 2705 | if (chip->irq >= 0) |
2589 | free_irq(chip->irq, chip); | 2706 | free_irq(chip->irq, chip); |
2590 | snd_es1968_free_gameport(chip); | 2707 | snd_es1968_free_gameport(chip); |
@@ -2723,6 +2840,14 @@ static int __devinit snd_es1968_create(struct snd_card *card, | |||
2723 | 2840 | ||
2724 | snd_card_set_dev(card, &pci->dev); | 2841 | snd_card_set_dev(card, &pci->dev); |
2725 | 2842 | ||
2843 | #ifdef CONFIG_SND_ES1968_RADIO | ||
2844 | chip->tea.card = card; | ||
2845 | chip->tea.freq_fixup = 10700; | ||
2846 | chip->tea.private_data = chip; | ||
2847 | chip->tea.ops = &snd_es1968_tea_ops; | ||
2848 | snd_tea575x_init(&chip->tea); | ||
2849 | #endif | ||
2850 | |||
2726 | *chip_ret = chip; | 2851 | *chip_ret = chip; |
2727 | 2852 | ||
2728 | return 0; | 2853 | return 0; |