|  | /* | 
|  | Mantis PCI bridge driver | 
|  |  | 
|  | Copyright (C) Manu Abraham (abraham.manu@gmail.com) | 
|  |  | 
|  | This program is free software; you can redistribute it and/or modify | 
|  | it under the terms of the GNU General Public License as published by | 
|  | the Free Software Foundation; either version 2 of the License, or | 
|  | (at your option) any later version. | 
|  |  | 
|  | This program is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | GNU General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU General Public License | 
|  | along with this program; if not, write to the Free Software | 
|  | Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | 
|  | */ | 
|  |  | 
|  | #include <linux/module.h> | 
|  | #include <linux/moduleparam.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <asm/io.h> | 
|  | #include <asm/pgtable.h> | 
|  | #include <asm/page.h> | 
|  | #include <linux/kmod.h> | 
|  | #include <linux/vmalloc.h> | 
|  | #include <linux/init.h> | 
|  | #include <linux/device.h> | 
|  | #include <linux/pci.h> | 
|  |  | 
|  | #include <asm/irq.h> | 
|  | #include <linux/signal.h> | 
|  | #include <linux/sched.h> | 
|  | #include <linux/interrupt.h> | 
|  |  | 
|  | #include "dmxdev.h" | 
|  | #include "dvbdev.h" | 
|  | #include "dvb_demux.h" | 
|  | #include "dvb_frontend.h" | 
|  | #include "dvb_net.h" | 
|  |  | 
|  | #include <asm/irq.h> | 
|  | #include <linux/signal.h> | 
|  | #include <linux/sched.h> | 
|  | #include <linux/interrupt.h> | 
|  |  | 
|  | #include "mantis_common.h" | 
|  | #include "mantis_reg.h" | 
|  | #include "mantis_pci.h" | 
|  |  | 
|  | #define DRIVER_NAME		"Mantis Core" | 
|  |  | 
|  | int __devinit mantis_pci_init(struct mantis_pci *mantis) | 
|  | { | 
|  | u8 revision, latency; | 
|  | struct mantis_hwconfig *config	= mantis->hwconfig; | 
|  | struct pci_dev *pdev		= mantis->pdev; | 
|  | int err, ret = 0; | 
|  |  | 
|  | dprintk(MANTIS_ERROR, 0, "found a %s PCI %s device on (%02x:%02x.%x),\n", | 
|  | config->model_name, | 
|  | config->dev_type, | 
|  | mantis->pdev->bus->number, | 
|  | PCI_SLOT(mantis->pdev->devfn), | 
|  | PCI_FUNC(mantis->pdev->devfn)); | 
|  |  | 
|  | err = pci_enable_device(pdev); | 
|  | if (err != 0) { | 
|  | ret = -ENODEV; | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: PCI enable failed <%i>", err); | 
|  | goto fail0; | 
|  | } | 
|  |  | 
|  | err = pci_set_consistent_dma_mask(pdev, DMA_BIT_MASK(32)); | 
|  | if (err != 0) { | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: Unable to obtain 32 bit DMA <%i>", err); | 
|  | ret = -ENOMEM; | 
|  | goto fail1; | 
|  | } | 
|  |  | 
|  | pci_set_master(pdev); | 
|  |  | 
|  | if (!request_mem_region(pci_resource_start(pdev, 0), | 
|  | pci_resource_len(pdev, 0), | 
|  | DRIVER_NAME)) { | 
|  |  | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: BAR0 Request failed !"); | 
|  | ret = -ENODEV; | 
|  | goto fail1; | 
|  | } | 
|  |  | 
|  | mantis->mmio = ioremap(pci_resource_start(pdev, 0), | 
|  | pci_resource_len(pdev, 0)); | 
|  |  | 
|  | if (!mantis->mmio) { | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: BAR0 remap failed !"); | 
|  | ret = -ENODEV; | 
|  | goto fail2; | 
|  | } | 
|  |  | 
|  | pci_read_config_byte(pdev, PCI_LATENCY_TIMER, &latency); | 
|  | pci_read_config_byte(pdev, PCI_CLASS_REVISION, &revision); | 
|  | mantis->latency = latency; | 
|  | mantis->revision = revision; | 
|  |  | 
|  | dprintk(MANTIS_ERROR, 0, "    Mantis Rev %d [%04x:%04x], ", | 
|  | mantis->revision, | 
|  | mantis->pdev->subsystem_vendor, | 
|  | mantis->pdev->subsystem_device); | 
|  |  | 
|  | dprintk(MANTIS_ERROR, 0, | 
|  | "irq: %d, latency: %d\n    memory: 0x%lx, mmio: 0x%p\n", | 
|  | mantis->pdev->irq, | 
|  | mantis->latency, | 
|  | mantis->mantis_addr, | 
|  | mantis->mmio); | 
|  |  | 
|  | err = request_irq(pdev->irq, | 
|  | config->irq_handler, | 
|  | IRQF_SHARED, | 
|  | DRIVER_NAME, | 
|  | mantis); | 
|  |  | 
|  | if (err != 0) { | 
|  |  | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: IRQ registration failed ! <%d>", err); | 
|  | ret = -ENODEV; | 
|  | goto fail3; | 
|  | } | 
|  |  | 
|  | pci_set_drvdata(pdev, mantis); | 
|  | return ret; | 
|  |  | 
|  | /* Error conditions */ | 
|  | fail3: | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: <%d> I/O unmap", ret); | 
|  | if (mantis->mmio) | 
|  | iounmap(mantis->mmio); | 
|  |  | 
|  | fail2: | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: <%d> releasing regions", ret); | 
|  | release_mem_region(pci_resource_start(pdev, 0), | 
|  | pci_resource_len(pdev, 0)); | 
|  |  | 
|  | fail1: | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: <%d> disabling device", ret); | 
|  | pci_disable_device(pdev); | 
|  |  | 
|  | fail0: | 
|  | dprintk(MANTIS_ERROR, 1, "ERROR: <%d> exiting", ret); | 
|  | pci_set_drvdata(pdev, NULL); | 
|  | return ret; | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(mantis_pci_init); | 
|  |  | 
|  | void mantis_pci_exit(struct mantis_pci *mantis) | 
|  | { | 
|  | struct pci_dev *pdev = mantis->pdev; | 
|  |  | 
|  | dprintk(MANTIS_NOTICE, 1, " mem: 0x%p", mantis->mmio); | 
|  | free_irq(pdev->irq, mantis); | 
|  | if (mantis->mmio) { | 
|  | iounmap(mantis->mmio); | 
|  | release_mem_region(pci_resource_start(pdev, 0), | 
|  | pci_resource_len(pdev, 0)); | 
|  | } | 
|  |  | 
|  | pci_disable_device(pdev); | 
|  | pci_set_drvdata(pdev, NULL); | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(mantis_pci_exit); | 
|  |  | 
|  | MODULE_DESCRIPTION("Mantis PCI DTV bridge driver"); | 
|  | MODULE_AUTHOR("Manu Abraham"); | 
|  | MODULE_LICENSE("GPL"); |