1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43#include <linux/module.h>
44#include <linux/delay.h>
45#include <linux/interrupt.h>
46
47#include "../comedi_pci.h"
48
49#include "plx9080.h"
50
51
52
53
54#define FIRMWARE_REV_REG 0x00
55#define FEATURES_REG_PRESENT_BIT BIT(15)
56#define BOARD_CONTROL_REG 0x04
57#define BOARD_RESET_BIT BIT(0)
58#define TX_FIFO_RESET_BIT BIT(1)
59#define RX_FIFO_RESET_BIT BIT(2)
60#define TX_ENABLE_BIT BIT(4)
61#define RX_ENABLE_BIT BIT(5)
62#define DEMAND_DMA_DIRECTION_TX_BIT BIT(6)
63#define LINE_VALID_ON_STATUS_VALID_BIT BIT(7)
64#define START_TX_BIT BIT(8)
65#define CABLE_THROTTLE_ENABLE_BIT BIT(9)
66#define TEST_MODE_ENABLE_BIT BIT(31)
67#define BOARD_STATUS_REG 0x08
68#define COMMAND_LINE_STATUS_MASK (0x7f << 0)
69#define TX_IN_PROGRESS_BIT BIT(7)
70#define TX_NOT_EMPTY_BIT BIT(8)
71#define TX_NOT_ALMOST_EMPTY_BIT BIT(9)
72#define TX_NOT_ALMOST_FULL_BIT BIT(10)
73#define TX_NOT_FULL_BIT BIT(11)
74#define RX_NOT_EMPTY_BIT BIT(12)
75#define RX_NOT_ALMOST_EMPTY_BIT BIT(13)
76#define RX_NOT_ALMOST_FULL_BIT BIT(14)
77#define RX_NOT_FULL_BIT BIT(15)
78#define BOARD_JUMPER0_INSTALLED_BIT BIT(16)
79#define BOARD_JUMPER1_INSTALLED_BIT BIT(17)
80#define TX_OVERRUN_BIT BIT(21)
81#define RX_UNDERRUN_BIT BIT(22)
82#define RX_OVERRUN_BIT BIT(23)
83#define TX_PROG_ALMOST_REG 0x0c
84#define RX_PROG_ALMOST_REG 0x10
85#define ALMOST_EMPTY_BITS(x) (((x) & 0xffff) << 0)
86#define ALMOST_FULL_BITS(x) (((x) & 0xff) << 16)
87#define FEATURES_REG 0x14
88#define FIFO_SIZE_PRESENT_BIT BIT(0)
89#define FIFO_WORDS_PRESENT_BIT BIT(1)
90#define LEVEL_EDGE_INTERRUPTS_PRESENT_BIT BIT(2)
91#define GPIO_SUPPORTED_BIT BIT(3)
92#define PLX_DMA_CH1_SUPPORTED_BIT BIT(4)
93#define OVERRUN_UNDERRUN_SUPPORTED_BIT BIT(5)
94#define FIFO_REG 0x18
95#define TX_STATUS_COUNT_REG 0x1c
96#define TX_LINE_VALID_COUNT_REG 0x20,
97#define TX_LINE_INVALID_COUNT_REG 0x24
98#define RX_STATUS_COUNT_REG 0x28
99#define RX_LINE_COUNT_REG 0x2c
100#define INTERRUPT_CONTROL_REG 0x30
101#define FRAME_VALID_START_INTR BIT(0)
102#define FRAME_VALID_END_INTR BIT(1)
103#define TX_FIFO_EMPTY_INTR BIT(8)
104#define TX_FIFO_ALMOST_EMPTY_INTR BIT(9)
105#define TX_FIFO_ALMOST_FULL_INTR BIT(10)
106#define TX_FIFO_FULL_INTR BIT(11)
107#define RX_EMPTY_INTR BIT(12)
108#define RX_ALMOST_EMPTY_INTR BIT(13)
109#define RX_ALMOST_FULL_INTR BIT(14)
110#define RX_FULL_INTR BIT(15)
111#define INTERRUPT_STATUS_REG 0x34
112#define TX_CLOCK_DIVIDER_REG 0x38
113#define TX_FIFO_SIZE_REG 0x40
114#define RX_FIFO_SIZE_REG 0x44
115#define FIFO_SIZE_MASK (0xfffff << 0)
116#define TX_FIFO_WORDS_REG 0x48
117#define RX_FIFO_WORDS_REG 0x4c
118#define INTERRUPT_EDGE_LEVEL_REG 0x50
119#define INTERRUPT_POLARITY_REG 0x54
120
121#define TIMER_BASE 50
122#define DMA_BUFFER_SIZE 0x10000
123#define NUM_DMA_BUFFERS 4
124#define NUM_DMA_DESCRIPTORS 256
125
126struct hpdi_private {
127 void __iomem *plx9080_mmio;
128 uint32_t *dio_buffer[NUM_DMA_BUFFERS];
129
130 dma_addr_t dio_buffer_phys_addr[NUM_DMA_BUFFERS];
131
132
133
134
135 struct plx_dma_desc *dma_desc;
136
137 dma_addr_t dma_desc_phys_addr;
138 unsigned int num_dma_descriptors;
139
140 uint32_t *desc_dio_buffer[NUM_DMA_DESCRIPTORS];
141
142 unsigned int dma_desc_index;
143 unsigned int tx_fifo_size;
144 unsigned int rx_fifo_size;
145 unsigned long dio_count;
146
147 unsigned int block_size;
148};
149
150static void gsc_hpdi_drain_dma(struct comedi_device *dev, unsigned int channel)
151{
152 struct hpdi_private *devpriv = dev->private;
153 struct comedi_subdevice *s = dev->read_subdev;
154 struct comedi_cmd *cmd = &s->async->cmd;
155 unsigned int idx;
156 unsigned int start;
157 unsigned int desc;
158 unsigned int size;
159 unsigned int next;
160
161 if (channel)
162 next = readl(devpriv->plx9080_mmio + PLX_DMA1_PCI_ADDRESS_REG);
163 else
164 next = readl(devpriv->plx9080_mmio + PLX_DMA0_PCI_ADDRESS_REG);
165
166 idx = devpriv->dma_desc_index;
167 start = le32_to_cpu(devpriv->dma_desc[idx].pci_start_addr);
168
169 for (desc = 0; (next < start || next >= start + devpriv->block_size) &&
170 desc < devpriv->num_dma_descriptors; desc++) {
171
172 size = devpriv->block_size / sizeof(uint32_t);
173 if (cmd->stop_src == TRIG_COUNT) {
174 if (size > devpriv->dio_count)
175 size = devpriv->dio_count;
176 devpriv->dio_count -= size;
177 }
178 comedi_buf_write_samples(s, devpriv->desc_dio_buffer[idx],
179 size);
180 idx++;
181 idx %= devpriv->num_dma_descriptors;
182 start = le32_to_cpu(devpriv->dma_desc[idx].pci_start_addr);
183
184 devpriv->dma_desc_index = idx;
185 }
186
187}
188
189static irqreturn_t gsc_hpdi_interrupt(int irq, void *d)
190{
191 struct comedi_device *dev = d;
192 struct hpdi_private *devpriv = dev->private;
193 struct comedi_subdevice *s = dev->read_subdev;
194 struct comedi_async *async = s->async;
195 uint32_t hpdi_intr_status, hpdi_board_status;
196 uint32_t plx_status;
197 uint32_t plx_bits;
198 uint8_t dma0_status, dma1_status;
199 unsigned long flags;
200
201 if (!dev->attached)
202 return IRQ_NONE;
203
204 plx_status = readl(devpriv->plx9080_mmio + PLX_INTRCS_REG);
205 if ((plx_status & (ICS_DMA0_A | ICS_DMA1_A | ICS_LIA)) == 0)
206 return IRQ_NONE;
207
208 hpdi_intr_status = readl(dev->mmio + INTERRUPT_STATUS_REG);
209 hpdi_board_status = readl(dev->mmio + BOARD_STATUS_REG);
210
211 if (hpdi_intr_status)
212 writel(hpdi_intr_status, dev->mmio + INTERRUPT_STATUS_REG);
213
214
215 spin_lock_irqsave(&dev->spinlock, flags);
216 dma0_status = readb(devpriv->plx9080_mmio + PLX_DMA0_CS_REG);
217 if (plx_status & ICS_DMA0_A) {
218
219 writeb((dma0_status & PLX_DMA_EN_BIT) | PLX_CLEAR_DMA_INTR_BIT,
220 devpriv->plx9080_mmio + PLX_DMA0_CS_REG);
221
222 if (dma0_status & PLX_DMA_EN_BIT)
223 gsc_hpdi_drain_dma(dev, 0);
224 }
225 spin_unlock_irqrestore(&dev->spinlock, flags);
226
227
228 spin_lock_irqsave(&dev->spinlock, flags);
229 dma1_status = readb(devpriv->plx9080_mmio + PLX_DMA1_CS_REG);
230 if (plx_status & ICS_DMA1_A) {
231
232 writeb((dma1_status & PLX_DMA_EN_BIT) | PLX_CLEAR_DMA_INTR_BIT,
233 devpriv->plx9080_mmio + PLX_DMA1_CS_REG);
234 }
235 spin_unlock_irqrestore(&dev->spinlock, flags);
236
237
238 if (plx_status & ICS_LDIA) {
239
240 plx_bits = readl(devpriv->plx9080_mmio + PLX_DBR_OUT_REG);
241 writel(plx_bits, devpriv->plx9080_mmio + PLX_DBR_OUT_REG);
242 }
243
244 if (hpdi_board_status & RX_OVERRUN_BIT) {
245 dev_err(dev->class_dev, "rx fifo overrun\n");
246 async->events |= COMEDI_CB_ERROR;
247 }
248
249 if (hpdi_board_status & RX_UNDERRUN_BIT) {
250 dev_err(dev->class_dev, "rx fifo underrun\n");
251 async->events |= COMEDI_CB_ERROR;
252 }
253
254 if (devpriv->dio_count == 0)
255 async->events |= COMEDI_CB_EOA;
256
257 comedi_handle_events(dev, s);
258
259 return IRQ_HANDLED;
260}
261
262static void gsc_hpdi_abort_dma(struct comedi_device *dev, unsigned int channel)
263{
264 struct hpdi_private *devpriv = dev->private;
265 unsigned long flags;
266
267
268 spin_lock_irqsave(&dev->spinlock, flags);
269
270 plx9080_abort_dma(devpriv->plx9080_mmio, channel);
271
272 spin_unlock_irqrestore(&dev->spinlock, flags);
273}
274
275static int gsc_hpdi_cancel(struct comedi_device *dev,
276 struct comedi_subdevice *s)
277{
278 writel(0, dev->mmio + BOARD_CONTROL_REG);
279 writel(0, dev->mmio + INTERRUPT_CONTROL_REG);
280
281 gsc_hpdi_abort_dma(dev, 0);
282
283 return 0;
284}
285
286static int gsc_hpdi_cmd(struct comedi_device *dev,
287 struct comedi_subdevice *s)
288{
289 struct hpdi_private *devpriv = dev->private;
290 struct comedi_async *async = s->async;
291 struct comedi_cmd *cmd = &async->cmd;
292 unsigned long flags;
293 uint32_t bits;
294
295 if (s->io_bits)
296 return -EINVAL;
297
298 writel(RX_FIFO_RESET_BIT, dev->mmio + BOARD_CONTROL_REG);
299
300 gsc_hpdi_abort_dma(dev, 0);
301
302 devpriv->dma_desc_index = 0;
303
304
305
306
307
308
309
310 writel(0, devpriv->plx9080_mmio + PLX_DMA0_TRANSFER_SIZE_REG);
311 writel(0, devpriv->plx9080_mmio + PLX_DMA0_PCI_ADDRESS_REG);
312 writel(0, devpriv->plx9080_mmio + PLX_DMA0_LOCAL_ADDRESS_REG);
313
314
315 bits = devpriv->dma_desc_phys_addr | PLX_DESC_IN_PCI_BIT |
316 PLX_INTR_TERM_COUNT | PLX_XFER_LOCAL_TO_PCI;
317 writel(bits, devpriv->plx9080_mmio + PLX_DMA0_DESCRIPTOR_REG);
318
319
320 spin_lock_irqsave(&dev->spinlock, flags);
321 writeb(PLX_DMA_EN_BIT | PLX_DMA_START_BIT | PLX_CLEAR_DMA_INTR_BIT,
322 devpriv->plx9080_mmio + PLX_DMA0_CS_REG);
323 spin_unlock_irqrestore(&dev->spinlock, flags);
324
325 if (cmd->stop_src == TRIG_COUNT)
326 devpriv->dio_count = cmd->stop_arg;
327 else
328 devpriv->dio_count = 1;
329
330
331 writel(RX_UNDERRUN_BIT | RX_OVERRUN_BIT, dev->mmio + BOARD_STATUS_REG);
332
333
334 writel(RX_FULL_INTR, dev->mmio + INTERRUPT_CONTROL_REG);
335
336 writel(RX_ENABLE_BIT, dev->mmio + BOARD_CONTROL_REG);
337
338 return 0;
339}
340
341static int gsc_hpdi_check_chanlist(struct comedi_device *dev,
342 struct comedi_subdevice *s,
343 struct comedi_cmd *cmd)
344{
345 int i;
346
347 for (i = 0; i < cmd->chanlist_len; i++) {
348 unsigned int chan = CR_CHAN(cmd->chanlist[i]);
349
350 if (chan != i) {
351 dev_dbg(dev->class_dev,
352 "chanlist must be ch 0 to 31 in order\n");
353 return -EINVAL;
354 }
355 }
356
357 return 0;
358}
359
360static int gsc_hpdi_cmd_test(struct comedi_device *dev,
361 struct comedi_subdevice *s,
362 struct comedi_cmd *cmd)
363{
364 int err = 0;
365
366 if (s->io_bits)
367 return -EINVAL;
368
369
370
371 err |= comedi_check_trigger_src(&cmd->start_src, TRIG_NOW);
372 err |= comedi_check_trigger_src(&cmd->scan_begin_src, TRIG_EXT);
373 err |= comedi_check_trigger_src(&cmd->convert_src, TRIG_NOW);
374 err |= comedi_check_trigger_src(&cmd->scan_end_src, TRIG_COUNT);
375 err |= comedi_check_trigger_src(&cmd->stop_src, TRIG_COUNT | TRIG_NONE);
376
377 if (err)
378 return 1;
379
380
381
382 err |= comedi_check_trigger_is_unique(cmd->stop_src);
383
384
385
386 if (err)
387 return 2;
388
389
390
391 err |= comedi_check_trigger_arg_is(&cmd->start_arg, 0);
392
393 if (!cmd->chanlist_len || !cmd->chanlist) {
394 cmd->chanlist_len = 32;
395 err |= -EINVAL;
396 }
397 err |= comedi_check_trigger_arg_is(&cmd->scan_end_arg,
398 cmd->chanlist_len);
399
400 if (cmd->stop_src == TRIG_COUNT)
401 err |= comedi_check_trigger_arg_min(&cmd->stop_arg, 1);
402 else
403 err |= comedi_check_trigger_arg_is(&cmd->stop_arg, 0);
404
405 if (err)
406 return 3;
407
408
409
410
411
412 if (cmd->chanlist && cmd->chanlist_len > 0)
413 err |= gsc_hpdi_check_chanlist(dev, s, cmd);
414
415 if (err)
416 return 5;
417
418 return 0;
419}
420
421
422static int gsc_hpdi_setup_dma_descriptors(struct comedi_device *dev,
423 unsigned int len)
424{
425 struct hpdi_private *devpriv = dev->private;
426 dma_addr_t phys_addr = devpriv->dma_desc_phys_addr;
427 uint32_t next_bits = PLX_DESC_IN_PCI_BIT | PLX_INTR_TERM_COUNT |
428 PLX_XFER_LOCAL_TO_PCI;
429 unsigned int offset = 0;
430 unsigned int idx = 0;
431 unsigned int i;
432
433 if (len > DMA_BUFFER_SIZE)
434 len = DMA_BUFFER_SIZE;
435 len -= len % sizeof(uint32_t);
436 if (len == 0)
437 return -EINVAL;
438
439 for (i = 0; i < NUM_DMA_DESCRIPTORS && idx < NUM_DMA_BUFFERS; i++) {
440 devpriv->dma_desc[i].pci_start_addr =
441 cpu_to_le32(devpriv->dio_buffer_phys_addr[idx] + offset);
442 devpriv->dma_desc[i].local_start_addr = cpu_to_le32(FIFO_REG);
443 devpriv->dma_desc[i].transfer_size = cpu_to_le32(len);
444 devpriv->dma_desc[i].next = cpu_to_le32((phys_addr +
445 (i + 1) * sizeof(devpriv->dma_desc[0])) | next_bits);
446
447 devpriv->desc_dio_buffer[i] = devpriv->dio_buffer[idx] +
448 (offset / sizeof(uint32_t));
449
450 offset += len;
451 if (len + offset > DMA_BUFFER_SIZE) {
452 offset = 0;
453 idx++;
454 }
455 }
456 devpriv->num_dma_descriptors = i;
457
458 devpriv->dma_desc[i - 1].next = cpu_to_le32(phys_addr | next_bits);
459
460 devpriv->block_size = len;
461
462 return len;
463}
464
465static int gsc_hpdi_dio_insn_config(struct comedi_device *dev,
466 struct comedi_subdevice *s,
467 struct comedi_insn *insn,
468 unsigned int *data)
469{
470 int ret;
471
472 switch (data[0]) {
473 case INSN_CONFIG_BLOCK_SIZE:
474 ret = gsc_hpdi_setup_dma_descriptors(dev, data[1]);
475 if (ret)
476 return ret;
477
478 data[1] = ret;
479 break;
480 default:
481 ret = comedi_dio_insn_config(dev, s, insn, data, 0xffffffff);
482 if (ret)
483 return ret;
484 break;
485 }
486
487 return insn->n;
488}
489
490static void gsc_hpdi_free_dma(struct comedi_device *dev)
491{
492 struct pci_dev *pcidev = comedi_to_pci_dev(dev);
493 struct hpdi_private *devpriv = dev->private;
494 int i;
495
496 if (!devpriv)
497 return;
498
499
500 for (i = 0; i < NUM_DMA_BUFFERS; i++) {
501 if (devpriv->dio_buffer[i])
502 pci_free_consistent(pcidev,
503 DMA_BUFFER_SIZE,
504 devpriv->dio_buffer[i],
505 devpriv->dio_buffer_phys_addr[i]);
506 }
507
508 if (devpriv->dma_desc)
509 pci_free_consistent(pcidev,
510 sizeof(struct plx_dma_desc) *
511 NUM_DMA_DESCRIPTORS,
512 devpriv->dma_desc,
513 devpriv->dma_desc_phys_addr);
514}
515
516static int gsc_hpdi_init(struct comedi_device *dev)
517{
518 struct hpdi_private *devpriv = dev->private;
519 uint32_t plx_intcsr_bits;
520
521
522 writel(BOARD_RESET_BIT, dev->mmio + BOARD_CONTROL_REG);
523 usleep_range(10, 1000);
524
525 writel(ALMOST_EMPTY_BITS(32) | ALMOST_FULL_BITS(32),
526 dev->mmio + RX_PROG_ALMOST_REG);
527 writel(ALMOST_EMPTY_BITS(32) | ALMOST_FULL_BITS(32),
528 dev->mmio + TX_PROG_ALMOST_REG);
529
530 devpriv->tx_fifo_size = readl(dev->mmio + TX_FIFO_SIZE_REG) &
531 FIFO_SIZE_MASK;
532 devpriv->rx_fifo_size = readl(dev->mmio + RX_FIFO_SIZE_REG) &
533 FIFO_SIZE_MASK;
534
535 writel(0, dev->mmio + INTERRUPT_CONTROL_REG);
536
537
538 plx_intcsr_bits =
539 ICS_AERR | ICS_PERR | ICS_PIE | ICS_PLIE | ICS_PAIE | ICS_LIE |
540 ICS_DMA0_E;
541 writel(plx_intcsr_bits, devpriv->plx9080_mmio + PLX_INTRCS_REG);
542
543 return 0;
544}
545
546static void gsc_hpdi_init_plx9080(struct comedi_device *dev)
547{
548 struct hpdi_private *devpriv = dev->private;
549 uint32_t bits;
550 void __iomem *plx_iobase = devpriv->plx9080_mmio;
551
552#ifdef __BIG_ENDIAN
553 bits = BIGEND_DMA0 | BIGEND_DMA1;
554#else
555 bits = 0;
556#endif
557 writel(bits, devpriv->plx9080_mmio + PLX_BIGEND_REG);
558
559 writel(0, devpriv->plx9080_mmio + PLX_INTRCS_REG);
560
561 gsc_hpdi_abort_dma(dev, 0);
562 gsc_hpdi_abort_dma(dev, 1);
563
564
565 bits = 0;
566
567 bits |= PLX_DMA_EN_READYIN_BIT;
568
569 bits |= PLX_EN_CHAIN_BIT;
570
571
572
573
574 bits |= PLX_EN_DMA_DONE_INTR_BIT;
575
576
577
578
579 bits |= PLX_LOCAL_ADDR_CONST_BIT;
580
581 bits |= PLX_DMA_INTR_PCI_BIT;
582
583 bits |= PLX_DEMAND_MODE_BIT;
584
585 bits |= PLX_DMA_LOCAL_BURST_EN_BIT;
586 bits |= PLX_LOCAL_BUS_32_WIDE_BITS;
587 writel(bits, plx_iobase + PLX_DMA0_MODE_REG);
588}
589
590static int gsc_hpdi_auto_attach(struct comedi_device *dev,
591 unsigned long context_unused)
592{
593 struct pci_dev *pcidev = comedi_to_pci_dev(dev);
594 struct hpdi_private *devpriv;
595 struct comedi_subdevice *s;
596 int i;
597 int retval;
598
599 dev->board_name = "pci-hpdi32";
600
601 devpriv = comedi_alloc_devpriv(dev, sizeof(*devpriv));
602 if (!devpriv)
603 return -ENOMEM;
604
605 retval = comedi_pci_enable(dev);
606 if (retval)
607 return retval;
608 pci_set_master(pcidev);
609
610 devpriv->plx9080_mmio = pci_ioremap_bar(pcidev, 0);
611 dev->mmio = pci_ioremap_bar(pcidev, 2);
612 if (!devpriv->plx9080_mmio || !dev->mmio) {
613 dev_warn(dev->class_dev, "failed to remap io memory\n");
614 return -ENOMEM;
615 }
616
617 gsc_hpdi_init_plx9080(dev);
618
619
620 if (request_irq(pcidev->irq, gsc_hpdi_interrupt, IRQF_SHARED,
621 dev->board_name, dev)) {
622 dev_warn(dev->class_dev,
623 "unable to allocate irq %u\n", pcidev->irq);
624 return -EINVAL;
625 }
626 dev->irq = pcidev->irq;
627
628 dev_dbg(dev->class_dev, " irq %u\n", dev->irq);
629
630
631 for (i = 0; i < NUM_DMA_BUFFERS; i++) {
632 devpriv->dio_buffer[i] =
633 pci_alloc_consistent(pcidev, DMA_BUFFER_SIZE,
634 &devpriv->dio_buffer_phys_addr[i]);
635 }
636
637 devpriv->dma_desc = pci_alloc_consistent(pcidev,
638 sizeof(struct plx_dma_desc) *
639 NUM_DMA_DESCRIPTORS,
640 &devpriv->dma_desc_phys_addr);
641 if (devpriv->dma_desc_phys_addr & 0xf) {
642 dev_warn(dev->class_dev,
643 " dma descriptors not quad-word aligned (bug)\n");
644 return -EIO;
645 }
646
647 retval = gsc_hpdi_setup_dma_descriptors(dev, 0x1000);
648 if (retval < 0)
649 return retval;
650
651 retval = comedi_alloc_subdevices(dev, 1);
652 if (retval)
653 return retval;
654
655
656 s = &dev->subdevices[0];
657 dev->read_subdev = s;
658 s->type = COMEDI_SUBD_DIO;
659 s->subdev_flags = SDF_READABLE | SDF_WRITABLE | SDF_LSAMPL |
660 SDF_CMD_READ;
661 s->n_chan = 32;
662 s->len_chanlist = 32;
663 s->maxdata = 1;
664 s->range_table = &range_digital;
665 s->insn_config = gsc_hpdi_dio_insn_config;
666 s->do_cmd = gsc_hpdi_cmd;
667 s->do_cmdtest = gsc_hpdi_cmd_test;
668 s->cancel = gsc_hpdi_cancel;
669
670 return gsc_hpdi_init(dev);
671}
672
673static void gsc_hpdi_detach(struct comedi_device *dev)
674{
675 struct hpdi_private *devpriv = dev->private;
676
677 if (dev->irq)
678 free_irq(dev->irq, dev);
679 if (devpriv) {
680 if (devpriv->plx9080_mmio) {
681 writel(0, devpriv->plx9080_mmio + PLX_INTRCS_REG);
682 iounmap(devpriv->plx9080_mmio);
683 }
684 if (dev->mmio)
685 iounmap(dev->mmio);
686 }
687 comedi_pci_disable(dev);
688 gsc_hpdi_free_dma(dev);
689}
690
691static struct comedi_driver gsc_hpdi_driver = {
692 .driver_name = "gsc_hpdi",
693 .module = THIS_MODULE,
694 .auto_attach = gsc_hpdi_auto_attach,
695 .detach = gsc_hpdi_detach,
696};
697
698static int gsc_hpdi_pci_probe(struct pci_dev *dev,
699 const struct pci_device_id *id)
700{
701 return comedi_pci_auto_config(dev, &gsc_hpdi_driver, id->driver_data);
702}
703
704static const struct pci_device_id gsc_hpdi_pci_table[] = {
705 { PCI_DEVICE_SUB(PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9080,
706 PCI_VENDOR_ID_PLX, 0x2400) },
707 { 0 }
708};
709MODULE_DEVICE_TABLE(pci, gsc_hpdi_pci_table);
710
711static struct pci_driver gsc_hpdi_pci_driver = {
712 .name = "gsc_hpdi",
713 .id_table = gsc_hpdi_pci_table,
714 .probe = gsc_hpdi_pci_probe,
715 .remove = comedi_pci_auto_unconfig,
716};
717module_comedi_pci_driver(gsc_hpdi_driver, gsc_hpdi_pci_driver);
718
719MODULE_AUTHOR("Comedi http://www.comedi.org");
720MODULE_DESCRIPTION("Comedi driver for General Standards PCI-HPDI32/PMC-HPDI32");
721MODULE_LICENSE("GPL");
722