|
@@ -44,8 +44,12 @@ module_param(debug,int,0644);
|
|
MODULE_PARM_DESC(debug,"enable debug messages [mpeg]");
|
|
MODULE_PARM_DESC(debug,"enable debug messages [mpeg]");
|
|
|
|
|
|
#define dprintk(level,fmt, arg...) if (debug >= level) \
|
|
#define dprintk(level,fmt, arg...) if (debug >= level) \
|
|
- printk(KERN_DEBUG "%s/2: " fmt, dev->core->name , ## arg)
|
|
|
|
|
|
+ printk(KERN_DEBUG "%s/2-mpeg: " fmt, dev->core->name, ## arg)
|
|
|
|
|
|
|
|
+#define mpeg_dbg(level,fmt, arg...) if (debug >= level) \
|
|
|
|
+ printk(KERN_DEBUG "%s/2-mpeg: " fmt, core->name, ## arg)
|
|
|
|
+
|
|
|
|
+static LIST_HEAD(cx8802_devlist);
|
|
/* ------------------------------------------------------------------ */
|
|
/* ------------------------------------------------------------------ */
|
|
|
|
|
|
static int cx8802_start_dma(struct cx8802_dev *dev,
|
|
static int cx8802_start_dma(struct cx8802_dev *dev,
|
|
@@ -65,17 +69,13 @@ static int cx8802_start_dma(struct cx8802_dev *dev,
|
|
|
|
|
|
/* FIXME: this needs a review.
|
|
/* FIXME: this needs a review.
|
|
* also: move to cx88-blackbird + cx88-dvb source files? */
|
|
* also: move to cx88-blackbird + cx88-dvb source files? */
|
|
- if (cx88_boards[core->board].mpeg == (CX88_MPEG_DVB | CX88_MPEG_BLACKBIRD) ) {
|
|
|
|
- /* Report a warning until the mini driver patch is applied,
|
|
|
|
- * else the following conditions will set the dma registers incorrectly.
|
|
|
|
- * This will be removed in the next major patch and changes to the conditions
|
|
|
|
- * will be made.
|
|
|
|
- */
|
|
|
|
- printk(KERN_INFO "%s() board->(CX88_MPEG_DVB | CX88_MPEG_BLACKBIRD) is invalid\n", __FUNCTION__);
|
|
|
|
- return -EINVAL;
|
|
|
|
- }
|
|
|
|
|
|
|
|
- if (cx88_boards[core->board].mpeg & CX88_MPEG_DVB) {
|
|
|
|
|
|
+ dprintk( 1, "core->active_type_id = 0x%08x\n", core->active_type_id);
|
|
|
|
+
|
|
|
|
+ if ( (core->active_type_id == CX88_MPEG_DVB) &&
|
|
|
|
+ (cx88_boards[core->board].mpeg & CX88_MPEG_DVB) ) {
|
|
|
|
+
|
|
|
|
+ dprintk( 1, "cx8802_start_dma doing .dvb\n");
|
|
/* negedge driven & software reset */
|
|
/* negedge driven & software reset */
|
|
cx_write(TS_GEN_CNTRL, 0x0040 | dev->ts_gen_cntrl);
|
|
cx_write(TS_GEN_CNTRL, 0x0040 | dev->ts_gen_cntrl);
|
|
udelay(100);
|
|
udelay(100);
|
|
@@ -93,15 +93,17 @@ static int cx8802_start_dma(struct cx8802_dev *dev,
|
|
cx_write(MO_PINMUX_IO, 0x88); /* Enable MPEG parallel IO and video signal pins */
|
|
cx_write(MO_PINMUX_IO, 0x88); /* Enable MPEG parallel IO and video signal pins */
|
|
udelay(100);
|
|
udelay(100);
|
|
break;
|
|
break;
|
|
|
|
+ case CX88_BOARD_HAUPPAUGE_HVR1300:
|
|
|
|
+ break;
|
|
default:
|
|
default:
|
|
cx_write(TS_SOP_STAT, 0x00);
|
|
cx_write(TS_SOP_STAT, 0x00);
|
|
break;
|
|
break;
|
|
}
|
|
}
|
|
cx_write(TS_GEN_CNTRL, dev->ts_gen_cntrl);
|
|
cx_write(TS_GEN_CNTRL, dev->ts_gen_cntrl);
|
|
udelay(100);
|
|
udelay(100);
|
|
- }
|
|
|
|
-
|
|
|
|
- if (cx88_boards[core->board].mpeg & CX88_MPEG_BLACKBIRD) {
|
|
|
|
|
|
+ } else if ( (core->active_type_id == CX88_MPEG_BLACKBIRD) &&
|
|
|
|
+ (cx88_boards[core->board].mpeg & CX88_MPEG_BLACKBIRD) ) {
|
|
|
|
+ dprintk( 1, "cx8802_start_dma doing .blackbird\n");
|
|
cx_write(MO_PINMUX_IO, 0x88); /* enable MPEG parallel IO */
|
|
cx_write(MO_PINMUX_IO, 0x88); /* enable MPEG parallel IO */
|
|
|
|
|
|
cx_write(TS_GEN_CNTRL, 0x46); /* punctured clock TS & posedge driven & software reset */
|
|
cx_write(TS_GEN_CNTRL, 0x46); /* punctured clock TS & posedge driven & software reset */
|
|
@@ -112,6 +114,10 @@ static int cx8802_start_dma(struct cx8802_dev *dev,
|
|
|
|
|
|
cx_write(TS_GEN_CNTRL, 0x06); /* punctured clock TS & posedge driven */
|
|
cx_write(TS_GEN_CNTRL, 0x06); /* punctured clock TS & posedge driven */
|
|
udelay(100);
|
|
udelay(100);
|
|
|
|
+ } else {
|
|
|
|
+ printk( "%s() Failed. Unsupported value in .mpeg (0x%08x)\n", __FUNCTION__,
|
|
|
|
+ cx88_boards[core->board].mpeg );
|
|
|
|
+ return -EINVAL;
|
|
}
|
|
}
|
|
|
|
|
|
/* reset counter */
|
|
/* reset counter */
|
|
@@ -542,8 +548,311 @@ int cx8802_resume_common(struct pci_dev *pci_dev)
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+struct cx8802_dev * cx8802_get_device(struct inode *inode)
|
|
|
|
+{
|
|
|
|
+ int minor = iminor(inode);
|
|
|
|
+ struct cx8802_dev *h = NULL;
|
|
|
|
+ struct list_head *list;
|
|
|
|
+
|
|
|
|
+ list_for_each(list,&cx8802_devlist) {
|
|
|
|
+ h = list_entry(list, struct cx8802_dev, devlist);
|
|
|
|
+ if (h->mpeg_dev->minor == minor)
|
|
|
|
+ return h;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return NULL;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+struct cx8802_driver * cx8802_get_driver(struct cx8802_dev *dev, enum cx88_board_type btype)
|
|
|
|
+{
|
|
|
|
+ struct cx8802_dev *h = NULL;
|
|
|
|
+ struct cx8802_driver *d = NULL;
|
|
|
|
+ struct list_head *list;
|
|
|
|
+ struct list_head *list2;
|
|
|
|
+
|
|
|
|
+ list_for_each(list,&cx8802_devlist) {
|
|
|
|
+ h = list_entry(list, struct cx8802_dev, devlist);
|
|
|
|
+
|
|
|
|
+ list_for_each(list2, &h->drvlist.devlist) {
|
|
|
|
+ d = list_entry(list2, struct cx8802_driver, devlist);
|
|
|
|
+
|
|
|
|
+ /* only unregister the correct driver type */
|
|
|
|
+ if (d->type_id == btype) {
|
|
|
|
+ return d;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return NULL;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* Driver asked for hardware access. */
|
|
|
|
+int cx8802_request_acquire(struct cx8802_driver *drv)
|
|
|
|
+{
|
|
|
|
+ struct cx88_core *core = drv->core;
|
|
|
|
+
|
|
|
|
+ /* Fail a request for hardware if the device is busy. */
|
|
|
|
+ if (core->active_type_id != CX88_BOARD_NONE)
|
|
|
|
+ return -EBUSY;
|
|
|
|
+
|
|
|
|
+ if (drv->advise_acquire)
|
|
|
|
+ {
|
|
|
|
+ core->active_type_id = drv->type_id;
|
|
|
|
+ drv->advise_acquire(drv);
|
|
|
|
+
|
|
|
|
+ mpeg_dbg(1,"%s() Post acquire GPIO=%x\n", __FUNCTION__, cx_read(MO_GP0_IO));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* Driver asked to release hardware. */
|
|
|
|
+int cx8802_request_release(struct cx8802_driver *drv)
|
|
|
|
+{
|
|
|
|
+ struct cx88_core *core = drv->core;
|
|
|
|
+
|
|
|
|
+ if (drv->advise_release)
|
|
|
|
+ {
|
|
|
|
+ drv->advise_release(drv);
|
|
|
|
+ core->active_type_id = CX88_BOARD_NONE;
|
|
|
|
+ mpeg_dbg(1,"%s() Post release GPIO=%x\n", __FUNCTION__, cx_read(MO_GP0_IO));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int cx8802_check_driver(struct cx8802_driver *drv)
|
|
|
|
+{
|
|
|
|
+ if (drv == NULL)
|
|
|
|
+ return -ENODEV;
|
|
|
|
+
|
|
|
|
+ if ((drv->type_id != CX88_MPEG_DVB) &&
|
|
|
|
+ (drv->type_id != CX88_MPEG_BLACKBIRD))
|
|
|
|
+ return -EINVAL;
|
|
|
|
+
|
|
|
|
+ if ((drv->hw_access != CX8802_DRVCTL_SHARED) &&
|
|
|
|
+ (drv->hw_access != CX8802_DRVCTL_EXCLUSIVE))
|
|
|
|
+ return -EINVAL;
|
|
|
|
+
|
|
|
|
+ if ((drv->probe == NULL) ||
|
|
|
|
+ (drv->remove == NULL) ||
|
|
|
|
+ (drv->advise_acquire == NULL) ||
|
|
|
|
+ (drv->advise_release == NULL))
|
|
|
|
+ return -EINVAL;
|
|
|
|
+
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+int cx8802_register_driver(struct cx8802_driver *drv)
|
|
|
|
+{
|
|
|
|
+ struct cx8802_dev *h;
|
|
|
|
+ struct cx8802_driver *driver;
|
|
|
|
+ struct list_head *list;
|
|
|
|
+ int err = 0, i = 0;
|
|
|
|
+
|
|
|
|
+ printk(KERN_INFO "%s() ->registering driver type=%s access=%s\n", __FUNCTION__ ,
|
|
|
|
+ drv->type_id == CX88_MPEG_DVB ? "dvb" : "blackbird",
|
|
|
|
+ drv->hw_access == CX8802_DRVCTL_SHARED ? "shared" : "exclusive");
|
|
|
|
+
|
|
|
|
+ if ((err = cx8802_check_driver(drv)) != 0) {
|
|
|
|
+ printk(KERN_INFO "%s() cx8802_driver is invalid\n", __FUNCTION__ );
|
|
|
|
+ return err;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ list_for_each(list,&cx8802_devlist) {
|
|
|
|
+ i++;
|
|
|
|
+ h = list_entry(list, struct cx8802_dev, devlist);
|
|
|
|
+
|
|
|
|
+ printk(KERN_INFO "CORE %s: subsystem: %04x:%04x, board: %s [card=%d]\n",
|
|
|
|
+ h->core->name,h->pci->subsystem_vendor,
|
|
|
|
+ h->pci->subsystem_device,cx88_boards[h->core->board].name,
|
|
|
|
+ h->core->board);
|
|
|
|
+
|
|
|
|
+ /* Bring up a new struct for each driver instance */
|
|
|
|
+ driver = kzalloc(sizeof(*drv),GFP_KERNEL);
|
|
|
|
+ if (driver == NULL)
|
|
|
|
+ return -ENOMEM;
|
|
|
|
+
|
|
|
|
+ /* Snapshot of the driver registration data */
|
|
|
|
+ drv->core = h->core;
|
|
|
|
+ drv->suspend = cx8802_suspend_common;
|
|
|
|
+ drv->resume = cx8802_resume_common;
|
|
|
|
+ drv->request_acquire = cx8802_request_acquire;
|
|
|
|
+ drv->request_release = cx8802_request_release;
|
|
|
|
+ memcpy(driver, drv, sizeof(*driver));
|
|
|
|
+
|
|
|
|
+ err = drv->probe(driver);
|
|
|
|
+ if (err == 0) {
|
|
|
|
+ mutex_lock(&drv->core->lock);
|
|
|
|
+ list_add_tail(&driver->devlist,&h->drvlist.devlist);
|
|
|
|
+ mutex_unlock(&drv->core->lock);
|
|
|
|
+ } else {
|
|
|
|
+ printk(KERN_ERR "%s() ->probe failed err = %d\n", __FUNCTION__, err);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ }
|
|
|
|
+ if (i == 0)
|
|
|
|
+ err = -ENODEV;
|
|
|
|
+
|
|
|
|
+ return err;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+int cx8802_unregister_driver(struct cx8802_driver *drv)
|
|
|
|
+{
|
|
|
|
+ struct cx8802_dev *h;
|
|
|
|
+ struct cx8802_driver *d;
|
|
|
|
+ struct list_head *list;
|
|
|
|
+ struct list_head *list2, *q;
|
|
|
|
+ int err = 0, i = 0;
|
|
|
|
+
|
|
|
|
+ printk(KERN_INFO "%s() ->unregistering driver type=%s\n", __FUNCTION__ ,
|
|
|
|
+ drv->type_id == CX88_MPEG_DVB ? "dvb" : "blackbird");
|
|
|
|
+
|
|
|
|
+ list_for_each(list,&cx8802_devlist) {
|
|
|
|
+ i++;
|
|
|
|
+ h = list_entry(list, struct cx8802_dev, devlist);
|
|
|
|
+
|
|
|
|
+ printk(KERN_INFO "CORE %s: subsystem: %04x:%04x, board: %s [card=%d]\n",
|
|
|
|
+ h->core->name,h->pci->subsystem_vendor,
|
|
|
|
+ h->pci->subsystem_device,cx88_boards[h->core->board].name,
|
|
|
|
+ h->core->board);
|
|
|
|
+
|
|
|
|
+ list_for_each_safe(list2, q, &h->drvlist.devlist) {
|
|
|
|
+ d = list_entry(list2, struct cx8802_driver, devlist);
|
|
|
|
+
|
|
|
|
+ /* only unregister the correct driver type */
|
|
|
|
+ if (d->type_id != drv->type_id)
|
|
|
|
+ continue;
|
|
|
|
+
|
|
|
|
+ err = d->remove(d);
|
|
|
|
+ if (err == 0) {
|
|
|
|
+ mutex_lock(&drv->core->lock);
|
|
|
|
+ list_del(list2);
|
|
|
|
+ mutex_unlock(&drv->core->lock);
|
|
|
|
+ } else
|
|
|
|
+ printk(KERN_ERR "%s() ->remove failed err = %d\n", __FUNCTION__, err);
|
|
|
|
+
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return err;
|
|
|
|
+}
|
|
|
|
+
|
|
/* ----------------------------------------------------------- */
|
|
/* ----------------------------------------------------------- */
|
|
|
|
+static int __devinit cx8802_probe(struct pci_dev *pci_dev,
|
|
|
|
+ const struct pci_device_id *pci_id)
|
|
|
|
+{
|
|
|
|
+ struct cx8802_dev *dev;
|
|
|
|
+ struct cx88_core *core;
|
|
|
|
+ int err;
|
|
|
|
+
|
|
|
|
+ /* general setup */
|
|
|
|
+ core = cx88_core_get(pci_dev);
|
|
|
|
+ if (NULL == core)
|
|
|
|
+ return -EINVAL;
|
|
|
|
+
|
|
|
|
+ printk("%s/2: cx2388x 8802 Driver Manager\n", core->name);
|
|
|
|
+
|
|
|
|
+ err = -ENODEV;
|
|
|
|
+ if (!cx88_boards[core->board].mpeg)
|
|
|
|
+ goto fail_core;
|
|
|
|
+
|
|
|
|
+ err = -ENOMEM;
|
|
|
|
+ dev = kzalloc(sizeof(*dev),GFP_KERNEL);
|
|
|
|
+ if (NULL == dev)
|
|
|
|
+ goto fail_core;
|
|
|
|
+ dev->pci = pci_dev;
|
|
|
|
+ dev->core = core;
|
|
|
|
+
|
|
|
|
+ err = cx8802_init_common(dev);
|
|
|
|
+ if (err != 0)
|
|
|
|
+ goto fail_free;
|
|
|
|
+
|
|
|
|
+ INIT_LIST_HEAD(&dev->drvlist.devlist);
|
|
|
|
+ list_add_tail(&dev->devlist,&cx8802_devlist);
|
|
|
|
|
|
|
|
+ /* Maintain a reference so cx88-video can query the 8802 device. */
|
|
|
|
+ core->dvbdev = dev;
|
|
|
|
+ return 0;
|
|
|
|
+
|
|
|
|
+ fail_free:
|
|
|
|
+ kfree(dev);
|
|
|
|
+ fail_core:
|
|
|
|
+ cx88_core_put(core,pci_dev);
|
|
|
|
+ return err;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void __devexit cx8802_remove(struct pci_dev *pci_dev)
|
|
|
|
+{
|
|
|
|
+ struct cx8802_dev *dev;
|
|
|
|
+ struct cx8802_driver *h;
|
|
|
|
+ struct list_head *list;
|
|
|
|
+
|
|
|
|
+ dev = pci_get_drvdata(pci_dev);
|
|
|
|
+
|
|
|
|
+ dprintk( 1, "%s\n", __FUNCTION__);
|
|
|
|
+
|
|
|
|
+ list_for_each(list,&dev->drvlist.devlist) {
|
|
|
|
+ h = list_entry(list, struct cx8802_driver, devlist);
|
|
|
|
+ dprintk( 1, " ->driver\n");
|
|
|
|
+ if (h->remove == NULL) {
|
|
|
|
+ printk(KERN_ERR "%s .. skipping driver, no probe function\n", __FUNCTION__);
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+ printk(KERN_INFO "%s .. Removing driver type %d\n", __FUNCTION__, h->type_id);
|
|
|
|
+ cx8802_unregister_driver(h);
|
|
|
|
+ list_del(&dev->drvlist.devlist);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* Destroy any 8802 reference. */
|
|
|
|
+ dev->core->dvbdev = NULL;
|
|
|
|
+
|
|
|
|
+ /* common */
|
|
|
|
+ cx8802_fini_common(dev);
|
|
|
|
+ cx88_core_put(dev->core,dev->pci);
|
|
|
|
+ kfree(dev);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static struct pci_device_id cx8802_pci_tbl[] = {
|
|
|
|
+ {
|
|
|
|
+ .vendor = 0x14f1,
|
|
|
|
+ .device = 0x8802,
|
|
|
|
+ .subvendor = PCI_ANY_ID,
|
|
|
|
+ .subdevice = PCI_ANY_ID,
|
|
|
|
+ },{
|
|
|
|
+ /* --- end of list --- */
|
|
|
|
+ }
|
|
|
|
+};
|
|
|
|
+MODULE_DEVICE_TABLE(pci, cx8802_pci_tbl);
|
|
|
|
+
|
|
|
|
+static struct pci_driver cx8802_pci_driver = {
|
|
|
|
+ .name = "cx88-mpeg driver manager",
|
|
|
|
+ .id_table = cx8802_pci_tbl,
|
|
|
|
+ .probe = cx8802_probe,
|
|
|
|
+ .remove = __devexit_p(cx8802_remove),
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+static int cx8802_init(void)
|
|
|
|
+{
|
|
|
|
+ printk(KERN_INFO "cx2388x cx88-mpeg Driver Manager version %d.%d.%d loaded\n",
|
|
|
|
+ (CX88_VERSION_CODE >> 16) & 0xff,
|
|
|
|
+ (CX88_VERSION_CODE >> 8) & 0xff,
|
|
|
|
+ CX88_VERSION_CODE & 0xff);
|
|
|
|
+#ifdef SNAPSHOT
|
|
|
|
+ printk(KERN_INFO "cx2388x: snapshot date %04d-%02d-%02d\n",
|
|
|
|
+ SNAPSHOT/10000, (SNAPSHOT/100)%100, SNAPSHOT%100);
|
|
|
|
+#endif
|
|
|
|
+ return pci_register_driver(&cx8802_pci_driver);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void cx8802_fini(void)
|
|
|
|
+{
|
|
|
|
+ pci_unregister_driver(&cx8802_pci_driver);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+module_init(cx8802_init);
|
|
|
|
+module_exit(cx8802_fini);
|
|
EXPORT_SYMBOL(cx8802_buf_prepare);
|
|
EXPORT_SYMBOL(cx8802_buf_prepare);
|
|
EXPORT_SYMBOL(cx8802_buf_queue);
|
|
EXPORT_SYMBOL(cx8802_buf_queue);
|
|
EXPORT_SYMBOL(cx8802_cancel_buffers);
|
|
EXPORT_SYMBOL(cx8802_cancel_buffers);
|
|
@@ -551,9 +860,10 @@ EXPORT_SYMBOL(cx8802_cancel_buffers);
|
|
EXPORT_SYMBOL(cx8802_init_common);
|
|
EXPORT_SYMBOL(cx8802_init_common);
|
|
EXPORT_SYMBOL(cx8802_fini_common);
|
|
EXPORT_SYMBOL(cx8802_fini_common);
|
|
|
|
|
|
-EXPORT_SYMBOL(cx8802_suspend_common);
|
|
|
|
-EXPORT_SYMBOL(cx8802_resume_common);
|
|
|
|
-
|
|
|
|
|
|
+EXPORT_SYMBOL(cx8802_register_driver);
|
|
|
|
+EXPORT_SYMBOL(cx8802_unregister_driver);
|
|
|
|
+EXPORT_SYMBOL(cx8802_get_device);
|
|
|
|
+EXPORT_SYMBOL(cx8802_get_driver);
|
|
/* ----------------------------------------------------------- */
|
|
/* ----------------------------------------------------------- */
|
|
/*
|
|
/*
|
|
* Local variables:
|
|
* Local variables:
|