1 // SPDX-License-Identifier: GPL-2.0-or-later
3 dmx3191d.c - driver for the Domex DMX3191D SCSI card.
4 Copyright (C) 2000 by Massimo Piccioni <dafastidio@libero.it>
5 Portions Copyright (C) 2004 by Christoph Hellwig <hch@lst.de>
7 Based on the generic NCR5380 driver by Drew Eckhardt et al.
11 #include <linux/init.h>
12 #include <linux/ioport.h>
13 #include <linux/kernel.h>
14 #include <linux/module.h>
15 #include <linux/pci.h>
16 #include <linux/interrupt.h>
19 #include <scsi/scsi_host.h>
22 * Definitions for the generic 5380 driver.
25 #define NCR5380_read(reg) inb(hostdata->base + (reg))
26 #define NCR5380_write(reg, value) outb(value, hostdata->base + (reg))
28 #define NCR5380_dma_xfer_len NCR5380_dma_xfer_none
29 #define NCR5380_dma_recv_setup NCR5380_dma_setup_none
30 #define NCR5380_dma_send_setup NCR5380_dma_setup_none
31 #define NCR5380_dma_residual NCR5380_dma_residual_none
33 #define NCR5380_implementation_fields /* none */
38 #define DMX3191D_DRIVER_NAME "dmx3191d"
39 #define DMX3191D_REGION_LEN 8
42 static const struct scsi_host_template dmx3191d_driver_template
= {
43 .module
= THIS_MODULE
,
44 .proc_name
= DMX3191D_DRIVER_NAME
,
45 .name
= "Domex DMX3191D",
47 .queuecommand
= NCR5380_queue_command
,
48 .eh_abort_handler
= NCR5380_abort
,
49 .eh_host_reset_handler
= NCR5380_host_reset
,
52 .sg_tablesize
= SG_ALL
,
54 .dma_boundary
= PAGE_SIZE
- 1,
55 .cmd_size
= sizeof(struct NCR5380_cmd
),
58 static int dmx3191d_probe_one(struct pci_dev
*pdev
,
59 const struct pci_device_id
*id
)
61 struct Scsi_Host
*shost
;
62 struct NCR5380_hostdata
*hostdata
;
66 if (pci_enable_device(pdev
))
69 io
= pci_resource_start(pdev
, 0);
70 if (!request_region(io
, DMX3191D_REGION_LEN
, DMX3191D_DRIVER_NAME
)) {
71 printk(KERN_ERR
"dmx3191: region 0x%lx-0x%lx already reserved\n",
72 io
, io
+ DMX3191D_REGION_LEN
);
73 goto out_disable_device
;
76 shost
= scsi_host_alloc(&dmx3191d_driver_template
,
77 sizeof(struct NCR5380_hostdata
));
79 goto out_release_region
;
81 hostdata
= shost_priv(shost
);
84 /* This card does not seem to raise an interrupt on pdev->irq.
85 * Steam-powered SCSI controllers run without an IRQ anyway.
89 error
= NCR5380_init(shost
, 0);
93 NCR5380_maybe_reset_bus(shost
);
95 pci_set_drvdata(pdev
, shost
);
97 error
= scsi_add_host(shost
, &pdev
->dev
);
101 scsi_scan_host(shost
);
107 scsi_host_put(shost
);
109 release_region(io
, DMX3191D_REGION_LEN
);
111 pci_disable_device(pdev
);
116 static void dmx3191d_remove_one(struct pci_dev
*pdev
)
118 struct Scsi_Host
*shost
= pci_get_drvdata(pdev
);
119 struct NCR5380_hostdata
*hostdata
= shost_priv(shost
);
120 unsigned long io
= hostdata
->base
;
122 scsi_remove_host(shost
);
125 scsi_host_put(shost
);
126 release_region(io
, DMX3191D_REGION_LEN
);
127 pci_disable_device(pdev
);
130 static struct pci_device_id dmx3191d_pci_tbl
[] = {
131 {PCI_VENDOR_ID_DOMEX
, PCI_DEVICE_ID_DOMEX_DMX3191D
,
132 PCI_ANY_ID
, PCI_ANY_ID
, 0, 0, 4},
135 MODULE_DEVICE_TABLE(pci
, dmx3191d_pci_tbl
);
137 static struct pci_driver dmx3191d_pci_driver
= {
138 .name
= DMX3191D_DRIVER_NAME
,
139 .id_table
= dmx3191d_pci_tbl
,
140 .probe
= dmx3191d_probe_one
,
141 .remove
= dmx3191d_remove_one
,
144 module_pci_driver(dmx3191d_pci_driver
);
146 MODULE_AUTHOR("Massimo Piccioni <dafastidio@libero.it>");
147 MODULE_DESCRIPTION("Domex DMX3191D SCSI driver");
148 MODULE_LICENSE("GPL");