summaryrefslogtreecommitdiff
path: root/linux/drivers/media/video/cx88/cx88-blackbird.c
diff options
context:
space:
mode:
Diffstat (limited to 'linux/drivers/media/video/cx88/cx88-blackbird.c')
-rw-r--r--linux/drivers/media/video/cx88/cx88-blackbird.c135
1 files changed, 53 insertions, 82 deletions
diff --git a/linux/drivers/media/video/cx88/cx88-blackbird.c b/linux/drivers/media/video/cx88/cx88-blackbird.c
index 0725ea76c..47772322b 100644
--- a/linux/drivers/media/video/cx88/cx88-blackbird.c
+++ b/linux/drivers/media/video/cx88/cx88-blackbird.c
@@ -1,5 +1,5 @@
/*
- * $Id: cx88-blackbird.c,v 1.2 2004/07/30 13:43:39 kraxel Exp $
+ * $Id: cx88-blackbird.c,v 1.3 2004/07/30 15:26:01 kraxel Exp $
*
* Support for a cx23416 mpeg encoder via cx2388x host port.
* "blackbird" reference design.
@@ -28,6 +28,8 @@
#include <linux/init.h>
#include <linux/device.h>
#include <linux/fs.h>
+#include <linux/delay.h>
+#include <linux/firmware.h>
#include "cx88.h"
@@ -84,13 +86,10 @@ MODULE_LICENSE("GPL");
#define IVTV_REG_VPU (0x9058 /*| IVTV_REG_OFFSET*/)
#define IVTV_REG_APU (0xA064 /*| IVTV_REG_OFFSET*/)
-#if 0
+/* ------------------------------------------------------------------ */
-/* initialize the dma for the mpegport, called from cx88-video.c:cx8800_initdev */
-int cx8800_mpegport_init_dma(struct cx8800_dev *dev)
+static void host_setup(struct cx88_core *core)
{
- dprintk(1,"cx88_mpegport_init_dma\n");
-
/* toggle reset of the host */
cx_write(MO_GPHST_SOFT_RST, 1);
udelay(100);
@@ -104,59 +103,10 @@ int cx8800_mpegport_init_dma(struct cx8800_dev *dev)
cx_write(MO_GPHST_HDSHK, 0);
cx_write(MO_GPHST_MUX16, 0x44448888U);
cx_write(MO_GPHST_MODE, 0);
-
- cx_write(MO_PINMUX_IO, 0x88); /* enable MPEG parallel IO */
-
- cx_write(MO_DEV_CNTRL2, 0x20); /* enable the RISC controller */
- cx_set(MO_PCI_INTMSK, 0x4); /* enable ts_int */
-
- cx_write(TS_F2_CMD_STAT_MM, 0x2900106); /* F2_CMD_STAT_MM defaults + master + memory space */
- cx_write(TS_GEN_CNTRL, 0x46); /* punctured clock TS & posedge driven & software reset */
- cx_write(MO_TS_LNGTH, MD_TS_LNGHT_VAL);
-
- udelay(100);
- INIT_LIST_HEAD(&dev->mpegq.active);
- INIT_LIST_HEAD(&dev->mpegq.queued);
- dev->mpegq.timeout.function = cx8800_mpegport_timeout;
- dev->mpegq.timeout.data = (unsigned long)dev;
- init_timer(&dev->mpegq.timeout);
- cx88_risc_stopper(dev->pci,&dev->mpegq.stopper,
- MO_TS_DMACNTRL,0x11,0x00);
- udelay(100);
-
- cx_write(MO_TS_INTMSK, 0);
- cx_set(MO_TS_INTSTAT, 0);
- cx_write(MO_TS_INTMSK, 0x1f1101); /* all except the irq2 bit */
-
- cx_write(MO_TS_GPCNTRL, GP_COUNT_CONTROL_RESET); /* reset gp counter to 0 */
- cx_write(TS_HW_SOP_CNTRL, 0x408); /* mpeg start byte */
- //cx_write(TS_HW_SOP_CNTRL, 0x2F0BC0); /* mpeg start byte ts: 0x2F0BC0 ? */
- cx_write(TS_VALERR_CNTRL, 0x2000);
-
- cx_write(TS_GEN_CNTRL, 0x06); /* punctured clock TS & posedge driven */
- udelay(100);
-
- return 0;
}
/* ------------------------------------------------------------------ */
-int wait_ready_gpio0_bit1_high(struct cx8800_dev *dev)
-{
- int timeout = 1000;
- do { timeout--; udelay(1); } while (!(cx_readb(MO_GP0_IO) & 2) && (timeout >= 0));
- if (timeout < 0) return -1;
- return 0;
-}
-
-int wait_ready_gpio0_bit1_low(struct cx8800_dev *dev)
-{
- int timeout = 1000;
- do { timeout--; udelay(1); } while ((cx_readb(MO_GP0_IO) & 2) && (timeout >= 0));
- if (timeout < 0) return -1;
- return 0;
-}
-
#define P1_MDATA0 0x390000
#define P1_MDATA1 0x390001
#define P1_MDATA2 0x390002
@@ -172,13 +122,25 @@ int wait_ready_gpio0_bit1_low(struct cx8800_dev *dev)
#define P1_RADDR1 0x39000D
#define P1_RRDWR 0x39000E
-/* Warning: address is dword address (4 bytes) */
-static int memory_write(struct cx8800_dev *dev, int address, int value)
+static int wait_ready_gpio0_bit1(struct cx88_core *core, u32 state)
{
- int retval;
+ unsigned long timeout = jiffies + msecs_to_jiffies(1);
+ u32 gpio0,need;
+
+ need = state ? 2 : 0;
+ for (;;) {
+ gpio0 = cx_read(MO_GP0_IO) & 2;
+ if (need == gpio0)
+ return 0;
+ if (time_after(jiffies,timeout))
+ return -1;
+ udelay(1);
+ }
+}
- //retval = wait_ready_gpio0_bit1_low(dev);
-
+static int memory_write(struct cx88_core *core, u32 address, u32 value)
+{
+ /* Warning: address is dword address (4 bytes) */
cx_writeb(P1_MDATA0, (unsigned int)value);
cx_writeb(P1_MDATA1, (unsigned int)(value >> 8));
cx_writeb(P1_MDATA2, (unsigned int)(value >> 16));
@@ -186,23 +148,25 @@ static int memory_write(struct cx8800_dev *dev, int address, int value)
cx_writeb(P1_MADDR2, (unsigned int)(address >> 16) | 0x40);
cx_writeb(P1_MADDR1, (unsigned int)(address >> 8));
cx_writeb(P1_MADDR0, (unsigned int)address);
+ cx_read(P1_MDATA0);
+ cx_read(P1_MADDR0);
- retval = wait_ready_gpio0_bit1_high(dev);
- return retval;
+ return wait_ready_gpio0_bit1(core,1);
}
-/* Warning: address is dword address (4 bytes) */
-static int memory_read(struct cx8800_dev *dev, int address, int *value)
+static int memory_read(struct cx88_core *core, u32 address, u32 *value)
{
- int retval;
+ int retval;
+ u32 val;
+ /* Warning: address is dword address (4 bytes) */
cx_writeb(P1_MADDR2, (unsigned int)(address >> 16) & ~0xC0);
cx_writeb(P1_MADDR1, (unsigned int)(address >> 8));
cx_writeb(P1_MADDR0, (unsigned int)address);
+ cx_read(P1_MADDR0);
- retval = wait_ready_gpio0_bit1_high(dev);
+ retval = wait_ready_gpio0_bit1(core,1);
- int val;
cx_writeb(P1_MDATA3, 0);
val = (unsigned char)cx_read(P1_MDATA3) << 24;
cx_writeb(P1_MDATA2, 0);
@@ -211,15 +175,13 @@ static int memory_read(struct cx8800_dev *dev, int address, int *value)
val |= (unsigned char)cx_read(P1_MDATA1) << 8;
cx_writeb(P1_MDATA0, 0);
val |= (unsigned char)cx_read(P1_MDATA0);
+
*value = val;
-
return retval;
}
-
-static int register_write(struct cx8800_dev *dev, int address, int value)
+static int register_write(struct cx88_core *core, u32 address, u32 value)
{
- int retval;
cx_writeb(P1_RDATA0, (unsigned int)value);
cx_writeb(P1_RDATA1, (unsigned int)(value >> 8));
cx_writeb(P1_RDATA2, (unsigned int)(value >> 16));
@@ -227,51 +189,60 @@ static int register_write(struct cx8800_dev *dev, int address, int value)
cx_writeb(P1_RADDR0, (unsigned int)address);
cx_writeb(P1_RADDR1, (unsigned int)(address >> 8));
cx_writeb(P1_RRDWR, 1);
+ cx_read(P1_RDATA0);
+ cx_read(P1_RADDR0);
- retval = wait_ready_gpio0_bit1_high(dev);
+ return wait_ready_gpio0_bit1(core,1);
+#if 0
udelay(1000); /* without this, things don't go right (subsequent memory_write()'s don't get through */
/* ? would this be safe here? set_current_state(TASK_INTERRUPTIBLE); schedule_timeout(1); */
- return retval;
+#endif
}
-static int register_read(struct cx8800_dev *dev, int address, int *value)
+static int register_read(struct cx88_core *core, u32 address, u32 *value)
{
int retval;
+ u32 val;
+
cx_writeb(P1_RADDR0, (unsigned int)address);
cx_writeb(P1_RADDR1, (unsigned int)(address >> 8));
cx_writeb(P1_RRDWR, 0);
+ cx_read(P1_RADDR0);
- retval = wait_ready_gpio0_bit1_high(dev);
-
- int val;
+ retval = wait_ready_gpio0_bit1(core,1);
val = (unsigned char)cx_read(P1_RDATA0);
val |= (unsigned char)cx_read(P1_RDATA1) << 8;
val |= (unsigned char)cx_read(P1_RDATA2) << 16;
val |= (unsigned char)cx_read(P1_RDATA3) << 24;
- *value = val;
+ *value = val;
return retval;
}
+/* ------------------------------------------------------------------ */
+
+#if 0
+
/* We don't need to call the API often, so using just one mailbox will probably suffice */
-int mpegport_api_cmd(struct cx8800_dev *dev, u32 command, u32 inputcnt, u32 outputcnt, ...)
+int blackbird_cmd(struct cx8802_dev *dev, u32 command, u32 inputcnt, u32 outputcnt, ...)
{
va_list args;
va_start(args, outputcnt);
int i;
u32 value;
+ u32 flag;
- dprintk(1,"API Command 0x%X\n", command);
+ dprintk(1,"%s: 0x%X\n", __FUNCTION__, command);
- /* this may not be 100% safe if we can't read any memory location without side effects */
+ /* this may not be 100% safe if we can't read any memory location
+ without side effects */
memory_read(dev, dev->mpegport_mailbox - 4, &value);
if (value != 0x12345678) {
dprintk(0, "Firmware and/or mailbox pointer not initialized or corrupted\n");
return -1;
}
- u32 flag;
memory_read(dev, dev->mpegport_mailbox, &flag);
if (flag) {
dprintk(0, "ERROR: Mailbox appears to be in use (%x)\n", flag);