summaryrefslogtreecommitdiff
path: root/linux/drivers
diff options
context:
space:
mode:
Diffstat (limited to 'linux/drivers')
-rw-r--r--linux/drivers/media/video/cx88/cx88-alsa.c221
1 files changed, 207 insertions, 14 deletions
diff --git a/linux/drivers/media/video/cx88/cx88-alsa.c b/linux/drivers/media/video/cx88/cx88-alsa.c
index 67c885e31..062ea54e3 100644
--- a/linux/drivers/media/video/cx88/cx88-alsa.c
+++ b/linux/drivers/media/video/cx88/cx88-alsa.c
@@ -1,5 +1,5 @@
/*
- * $Id: cx88-alsa.c,v 1.6 2005/08/13 22:59:49 mchehab Exp $
+ * $Id: cx88-alsa.c,v 1.7 2005/08/15 01:49:36 mchehab Exp $
*
* Support for audio capture
* PCI function #1 of the cx2388x.
@@ -40,6 +40,10 @@
#include "cx88.h"
#include "cx88-reg.h"
+#define dprintk(level,fmt, arg...) if (debug >= level) \
+ printk(KERN_DEBUG "%s/1: " fmt, chip->core->name , ## arg)
+
+
/****************************************************************************
Data type declarations - Can be moded to a header file later
****************************************************************************/
@@ -60,7 +64,7 @@ enum { DEVICE_DIGITAL, DEVICE_ANALOG };
struct cx88_audio_dev {
struct cx88_core *core;
struct cx88_buffer *buf;
- struct cx88_dmaqueue *q;
+ struct cx88_dmaqueue q;
/* pci i/o */
struct pci_dev *pci;
@@ -111,10 +115,177 @@ MODULE_LICENSE("GPL");
MODULE_SUPPORTED_DEVICE("{{Conexant,23881},"
"{{Conexant,23882},"
"{{Conexant,23883}");
+static unsigned int debug = 0;
+module_param(debug,int,0644);
+MODULE_PARM_DESC(debug,"enable debug messages");
/****************************************************************************
- Module useful unused functions
+ Module specific funtions
****************************************************************************/
+
+/*
+ * BOARD Specific: Sets audio DMA
+ */
+
+int cx88_start_audio_dma(snd_cx88_card_t *chip)
+{
+ struct cx88_core *core=chip->core;
+
+ /* setup fifo + format - out channel */
+ cx88_sram_channel_setup(core, &cx88_sram_channels[SRAM_CH26],
+ chip->buf->bpl, chip->buf->risc.dma);
+
+ /* FIXME reset counter */
+ cx_write(MO_VIDY_GPCNTRL,GP_COUNT_CONTROL_RESET);
+
+ /* enable irqs */
+ cx_set(MO_PCI_INTMSK, chip->core->pci_irqmask | 0x02);
+
+ /* Enables corresponding bits at AUD_INT_STAT */
+#if 1
+ cx_set(MO_AUD_INTMSK, (1<<21)||(1<<19));
+#endif
+
+ /* start dma */
+ cx_set(MO_DEV_CNTRL2, (1<<5)); /* Enables Risc Processor */
+ cx_set(MO_AUD_DMACNTRL, 0x11); /* audio downstream FIFO and RISC enable */
+
+ return 0;
+}
+
+/*
+ * BOARD Specific: Resets audio DMA
+ */
+int cx88_stop_audio_dma(snd_cx88_card_t *chip)
+{
+ struct cx88_core *core=chip->core;
+ /* stop dma */
+ cx_clear(MO_AUD_DMACNTRL, 0x11);
+
+ /* disable irqs */
+ cx_clear(MO_PCI_INTMSK, 0x000002);
+
+ cx_set(MO_AUD_INTMSK, 0);
+ return 0;
+}
+
+#define MAX_IRQ_LOOP 10
+
+static void cx8801_timeout(unsigned long data)
+{
+ snd_cx88_card_t *chip = (snd_cx88_card_t *)data;
+
+ dprintk(0, "cx88_alsa: %s\n",__FUNCTION__);
+
+ if (debug)
+ cx88_sram_channel_dump(chip->core, &cx88_sram_channels[SRAM_CH26]);
+ cx88_stop_audio_dma(chip);
+#if 0
+ do_cancel_buffers(dev,"timeout",1);
+#endif
+}
+
+/* FIXME: Wrong values*/
+static char *cx88_aud_irqs[32] = {
+ "y_risci1", "u_risci1", "v_risci1", "vbi_risc1",
+ "y_risci2", "u_risci2", "v_risci2", "vbi_risc2",
+ "y_oflow", "u_oflow", "v_oflow", "vbi_oflow",
+ "y_sync", "u_sync", "v_sync", "vbi_sync",
+ "opc_err", "par_err", "rip_err", "pci_abort",
+};
+
+
+static void cx8801_aud_irq(snd_cx88_card_t *chip)
+{
+ struct cx88_core *core = chip->core;
+ u32 status, mask;
+#if 0
+ u32 count;
+#endif
+ status = cx_read(MO_AUD_INTSTAT);
+ mask = cx_read(MO_AUD_INTMSK);
+ if (0 == (status & mask))
+ return;
+ cx_write(MO_AUD_INTSTAT, status);
+ if (debug || (status & mask & ~0xff))
+ cx88_print_irqbits(core->name, "irq aud",
+ cx88_aud_irqs, status, mask);
+#if 0 /* FIXME */
+ /* risc op code error */
+ if (status & (1 << 16)) {
+ printk(KERN_WARNING "%s/0: video risc op code error\n",core->name);
+ cx_clear(MO_VID_DMACNTRL, 0x11);
+ cx_clear(VID_CAPTURE_CONTROL, 0x06);
+ cx88_sram_channel_dump(dev->core, &cx88_sram_channels[SRAM_CH21]);
+ }
+
+ /* risc1 y */
+ if (status & 0x01) {
+ spin_lock(&dev->slock);
+ count = cx_read(MO_VIDY_GPCNT);
+ cx88_wakeup(dev->core, &dev->vidq, count);
+ spin_unlock(&dev->slock);
+ }
+
+ /* risc1 vbi */
+ if (status & 0x08) {
+ spin_lock(&dev->slock);
+ count = cx_read(MO_VBI_GPCNT);
+ cx88_wakeup(dev->core, &dev->vbiq, count);
+ spin_unlock(&dev->slock);
+ }
+
+ /* risc2 y */
+ if (status & 0x10) {
+ dprintk(2,"stopper video\n");
+ spin_lock(&dev->slock);
+ restart_video_queue(dev,&dev->vidq);
+ spin_unlock(&dev->slock);
+ }
+
+ /* risc2 vbi */
+ if (status & 0x80) {
+ dprintk(2,"stopper vbi\n");
+ spin_lock(&dev->slock);
+ cx8800_restart_vbi_queue(dev,&dev->vbiq);
+ spin_unlock(&dev->slock);
+ }
+#endif
+}
+
+static irqreturn_t cx8801_irq(int irq, void *dev_id, struct pt_regs *regs)
+{
+ snd_cx88_card_t *chip = dev_id;
+ struct cx88_core *core = chip->core;
+ u32 status;
+ int loop, handled = 0;
+
+ for (loop = 0; loop < MAX_IRQ_LOOP; loop++) {
+ status = cx_read(MO_PCI_INTSTAT) & (core->pci_irqmask | 0x02);
+ if (0 == status)
+ goto out;
+ dprintk( 1, "cx8801_irq\n" );
+ dprintk( 1, " loop: %d/%d\n", loop, MAX_IRQ_LOOP );
+ dprintk( 1, " status: %d\n", status );
+ handled = 1;
+ cx_write(MO_PCI_INTSTAT, status);
+
+ if (status & core->pci_irqmask)
+ cx88_core_irq(core,status);
+ if (status & 0x02)
+ cx8801_aud_irq(chip);
+ };
+ if (MAX_IRQ_LOOP == loop) {
+ dprintk( 0, "clearing mask\n" );
+ dprintk(1,"%s/0: irq loop -- clearing mask\n",
+ core->name);
+ cx_write(MO_PCI_INTMSK,0);
+ }
+
+ out:
+ return IRQ_RETVAL(handled);
+}
+
/*
=====================> FIXME
*/
@@ -524,7 +695,6 @@ static int snd_cx88_dev_free(snd_device_t *device)
*/
static int devno=0;
-
static int __devinit snd_cx88_create(snd_card_t *card, struct pci_dev *pci,
snd_cx88_card_t **rchip)
{
@@ -548,6 +718,13 @@ static int __devinit snd_cx88_create(snd_card_t *card, struct pci_dev *pci,
}
memset(chip,0,sizeof(*chip));
+ if (!pci_dma_supported(pci,0xffffffff)) {
+ dprintk(0, "%s/1: Oops: no 32bit PCI DMA ???\n",core->name);
+ err = -EIO;
+ kfree (chip);
+ return err;
+ }
+
/* pci init */
chip->card = card;
chip->pci = pci;
@@ -561,26 +738,42 @@ static int __devinit snd_cx88_create(snd_card_t *card, struct pci_dev *pci,
return err;
}
chip->core = core;
-#if 1
- chip->dig_rate=48000;
+#if 1 /* Should be tested if it is wright */
+ chip->dig_rate=32000;
#endif
+#if 1 /* From cx88-mpeg.c */
+
+ /* init dma queue */
+ INIT_LIST_HEAD(&chip->q.active);
+ INIT_LIST_HEAD(&chip->q.queued);
+ chip->q.timeout.function = cx8801_timeout;
+ chip->q.timeout.data = (unsigned long)chip;
+ init_timer(&chip->q.timeout);
+ cx88_risc_stopper(chip->pci,&chip->q.stopper,
+ MO_AUD_DMACNTRL, 0x11,0x00);
+
+ /* get irq */
+ err = request_irq(chip->pci->irq, cx8801_irq,
+ SA_SHIRQ | SA_INTERRUPT, chip->core->name, chip);
+ if (err < 0) {
+ dprintk(0, "%s: can't get IRQ %d\n",
+ chip->core->name, chip->pci->irq);
+ return err;
+ }
+ cx_set(MO_PCI_INTMSK, core->pci_irqmask);
+#endif
/* print pci info */
pci_read_config_byte(pci, PCI_CLASS_REVISION, &chip->pci_rev);
pci_read_config_byte(pci, PCI_LATENCY_TIMER, &chip->pci_lat);
- printk(KERN_INFO "ALSA %s/%i: found at %s, rev: %d, irq: %d, "
+ dprintk(1,"ALSA %s/%i: found at %s, rev: %d, irq: %d, "
"latency: %d, mmio: 0x%lx\n", core->name, devno,
pci_name(pci), chip->pci_rev, pci->irq,
chip->pci_lat,pci_resource_start(pci,0));
pci_set_master(pci);
- if (!pci_dma_supported(pci,0xffffffff)) {
- printk("%s/1: Oops: no 32bit PCI DMA ???\n",core->name);
- err = -EIO;
- kfree (chip);
- return err;
- }
+ synchronize_irq(chip->irq);
err = snd_device_new(card, SNDRV_DEV_LOWLEVEL, chip, &ops);
if (err < 0) {
@@ -634,7 +827,7 @@ static int __devinit cx88_audio_initdev(struct pci_dev *pci,
card->shortname, pci_resource_start(pci, 0));
strcpy (card->mixername, "CX88");
- printk ("%s/%i: Alsa support for cx2388x boards\n",
+ dprintk (0, "%s/%i: Alsa support for cx2388x boards\n",
card->driver,devno);
err = snd_card_register(card);