2 comedi/drivers/usbdux.c
3 Copyright (C) 2003-2007 Bernd Porr, Bernd.Porr@f2s.com
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
17 Description: University of Stirling USB DAQ & INCITE Technology Limited
18 Devices: [ITL] USB-DUX (usbdux.o)
19 Author: Bernd Porr <BerndPorr@f2s.com>
22 Configuration options:
23 You have to upload firmware with the -i option. The
24 firmware is usually installed under /usr/share/usb or
25 /usr/local/share/usb or /lib/firmware.
27 Connection scheme for the counter at the digital port:
28 0=/CLK0, 1=UP/DOWN0, 2=RESET0, 4=/CLK1, 5=UP/DOWN1, 6=RESET1.
29 The sampling rate of the counter is approximately 500Hz.
31 Please note that under USB2.0 the length of the channel list determines
32 the max sampling rate. If you sample only one channel you get 8kHz
33 sampling rate. If you sample two channels you get 4kHz and so on.
36 * I must give credit here to Chris Baugher who
37 * wrote the driver for AT-MIO-16d. I used some parts of this
38 * driver. I also must give credits to David Brownell
39 * who supported me with the USB development.
45 * 0.94: D/A output should work now with any channel list combinations
46 * 0.95: .owner commented out for kernel vers below 2.4.19
47 * sanity checks in ai/ao_cmd
48 * 0.96: trying to get it working with 2.6, moved all memory alloc to comedi's
49 * attach final USB IDs
50 * moved memory allocation completely to the corresponding comedi
51 * functions firmware upload is by fxload and no longer by comedi (due to
53 * 0.97: USB IDs received, adjusted table
54 * 0.98: SMP, locking, memory alloc: moved all usb memory alloc
55 * to the usb subsystem and moved all comedi related memory
57 * | kernel | registration | usbdux-usb | usbdux-comedi | comedi |
58 * 0.99: USB 2.0: changed protocol to isochronous transfer
59 * IRQ transfer is too buggy and too risky in 2.0
60 * for the high speed ISO transfer is now a working version
62 * 0.99b: Increased the iso transfer buffer for high sp.to 10 buffers. Some VIA
63 * chipsets miss out IRQs. Deeper buffering is needed.
64 * 1.00: full USB 2.0 support for the A/D converter. Now: max 8kHz sampling
66 * Firmware vers 1.00 is needed for this.
67 * Two 16 bit up/down/reset counter with a sampling rate of 1kHz
68 * And loads of cleaning up, in particular streamlining the
70 * 1.1: moved EP4 transfers to EP1 to make space for a PWM output on EP4
71 * 1.2: added PWM support via EP4
72 * 2.0: PWM seems to be stable and is not interfering with the other functions
73 * 2.1: changed PWM API
74 * 2.2: added firmware kernel request to fix an udev problem
75 * 2.3: corrected a bug in bulk timeouts which were far too short
76 * 2.4: fixed a bug which causes the driver to hang when it ran out of data.
77 * Thanks to Jan-Matthias Braun and Ian to spot the bug and fix it.
81 #include <linux/kernel.h>
82 #include <linux/module.h>
83 #include <linux/slab.h>
84 #include <linux/input.h>
85 #include <linux/usb.h>
86 #include <linux/fcntl.h>
87 #include <linux/compiler.h>
89 #include "../comedidev.h"
91 #include "comedi_fc.h"
93 /* constants for firmware upload and download */
94 #define USBDUX_FIRMWARE "usbdux_firmware.bin"
95 #define USBDUX_FIRMWARE_MAX_LEN 0x2000
96 #define USBDUX_FIRMWARE_CMD 0xa0
97 #define VENDOR_DIR_IN 0xc0
98 #define VENDOR_DIR_OUT 0x40
99 #define USBDUX_CPU_CS 0xe600
101 /* usbdux bulk transfer commands */
102 #define USBDUX_CMD_MULT_AI 0
103 #define USBDUX_CMD_AO 1
104 #define USBDUX_CMD_DIO_CFG 2
105 #define USBDUX_CMD_DIO_BITS 3
106 #define USBDUX_CMD_SINGLE_AI 4
107 #define USBDUX_CMD_TIMER_RD 5
108 #define USBDUX_CMD_TIMER_WR 6
109 #define USBDUX_CMD_PWM_ON 7
110 #define USBDUX_CMD_PWM_OFF 8
112 /* timeout for the USB-transfer in ms */
113 #define BULK_TIMEOUT 1000
115 /* 300Hz max frequ under PWM */
116 #define MIN_PWM_PERIOD ((long)(1E9/300))
118 /* Default PWM frequency */
119 #define PWM_DEFAULT_PERIOD ((long)(1E9/100))
121 /* Size of one A/D value */
122 #define SIZEADIN ((sizeof(uint16_t)))
125 * Size of the input-buffer IN BYTES
126 * Always multiple of 8 for 8 microframes which is needed in the highspeed mode
128 #define SIZEINBUF ((8*SIZEADIN))
131 #define SIZEINSNBUF 16
133 /* size of one value for the D/A converter: channel and value */
134 #define SIZEDAOUT ((sizeof(uint8_t)+sizeof(uint16_t)))
137 * Size of the output-buffer in bytes
138 * Actually only the first 4 triplets are used but for the
139 * high speed mode we need to pad it to 8 (microframes).
141 #define SIZEOUTBUF ((8*SIZEDAOUT))
144 * Size of the buffer for the dux commands: just now max size is determined
145 * by the analogue out + command byte + panic bytes...
147 #define SIZEOFDUXBUFFER ((8*SIZEDAOUT+2))
149 /* Number of in-URBs which receive the data: min=2 */
150 #define NUMOFINBUFFERSFULL 5
152 /* Number of out-URBs which send the data: min=2 */
153 #define NUMOFOUTBUFFERSFULL 5
155 /* Number of in-URBs which receive the data: min=5 */
156 /* must have more buffers due to buggy USB ctr */
157 #define NUMOFINBUFFERSHIGH 10
159 /* Number of out-URBs which send the data: min=5 */
160 /* must have more buffers due to buggy USB ctr */
161 #define NUMOFOUTBUFFERSHIGH 10
163 /* number of retries to get the right dux command */
166 static const struct comedi_lrange range_usbdux_ai_range
= {
169 BIP_RANGE(4.096 / 2),
175 static const struct comedi_lrange range_usbdux_ao_range
= {
182 struct usbdux_private
{
183 /* actual number of in-buffers */
185 /* actual number of out-buffers */
187 /* ISO-transfer handling: buffers */
188 struct urb
**ai_urbs
;
189 struct urb
**ao_urbs
;
190 /* pwm-transfer handling */
193 unsigned int pwm_period
;
194 /* PWM internal delay for the GPIF in the FX2 */
196 /* size of the PWM buffer which holds the bit pattern */
198 /* input buffer for the ISO-transfer */
200 /* input buffer for single insn */
203 unsigned int high_speed
:1;
204 unsigned int ai_cmd_running
:1;
205 unsigned int ao_cmd_running
:1;
206 unsigned int pwm_cmd_running
:1;
208 /* number of samples to acquire */
211 /* time between samples in units of the timer */
212 unsigned int ai_timer
;
213 unsigned int ao_timer
;
214 /* counter between aquisitions */
215 unsigned int ai_counter
;
216 unsigned int ao_counter
;
217 /* interval in frames/uframes */
218 unsigned int ai_interval
;
220 uint8_t *dux_commands
;
221 struct semaphore sem
;
224 static void usbdux_unlink_urbs(struct urb
**urbs
, int num_urbs
)
228 for (i
= 0; i
< num_urbs
; i
++)
229 usb_kill_urb(urbs
[i
]);
232 static void usbdux_ai_stop(struct comedi_device
*dev
, int do_unlink
)
234 struct usbdux_private
*devpriv
= dev
->private;
236 if (do_unlink
&& devpriv
->ai_urbs
)
237 usbdux_unlink_urbs(devpriv
->ai_urbs
, devpriv
->n_ai_urbs
);
239 devpriv
->ai_cmd_running
= 0;
242 static int usbdux_ai_cancel(struct comedi_device
*dev
,
243 struct comedi_subdevice
*s
)
245 struct usbdux_private
*devpriv
= dev
->private;
247 /* prevent other CPUs from submitting new commands just now */
249 /* unlink only if the urb really has been submitted */
250 usbdux_ai_stop(dev
, devpriv
->ai_cmd_running
);
256 /* analogue IN - interrupt service routine */
257 static void usbduxsub_ai_isoc_irq(struct urb
*urb
)
259 struct comedi_device
*dev
= urb
->context
;
260 struct comedi_subdevice
*s
= dev
->read_subdev
;
261 struct usbdux_private
*devpriv
= dev
->private;
262 struct comedi_cmd
*cmd
= &s
->async
->cmd
;
265 /* first we test if something unusual has just happened */
266 switch (urb
->status
) {
268 /* copy the result in the transfer buffer */
269 memcpy(devpriv
->in_buf
, urb
->transfer_buffer
, SIZEINBUF
);
272 /* error in the ISOchronous data */
273 /* we don't copy the data into the transfer buffer */
274 /* and recycle the last data byte */
275 dev_dbg(dev
->class_dev
, "CRC error in ISO IN stream\n");
282 /* happens after an unlink command */
283 if (devpriv
->ai_cmd_running
) {
284 s
->async
->events
|= COMEDI_CB_EOA
;
285 s
->async
->events
|= COMEDI_CB_ERROR
;
286 comedi_event(dev
, s
);
287 /* stop the transfer w/o unlink */
288 usbdux_ai_stop(dev
, 0);
293 /* a real error on the bus */
294 /* pass error to comedi if we are really running a command */
295 if (devpriv
->ai_cmd_running
) {
296 dev_err(dev
->class_dev
,
297 "Non-zero urb status received in ai intr context: %d\n",
299 s
->async
->events
|= COMEDI_CB_EOA
;
300 s
->async
->events
|= COMEDI_CB_ERROR
;
301 comedi_event(dev
, s
);
302 /* don't do an unlink here */
303 usbdux_ai_stop(dev
, 0);
309 * at this point we are reasonably sure that nothing dodgy has happened
310 * are we running a command?
312 if (unlikely(!devpriv
->ai_cmd_running
)) {
314 * not running a command, do not continue execution if no
315 * asynchronous command is running in particular not resubmit
320 urb
->dev
= comedi_to_usb_dev(dev
);
322 /* resubmit the urb */
323 err
= usb_submit_urb(urb
, GFP_ATOMIC
);
324 if (unlikely(err
< 0)) {
325 dev_err(dev
->class_dev
,
326 "urb resubmit failed in int-context! err=%d\n", err
);
327 if (err
== -EL2NSYNC
)
328 dev_err(dev
->class_dev
,
329 "buggy USB host controller or bug in IRQ handler!\n");
330 s
->async
->events
|= COMEDI_CB_EOA
;
331 s
->async
->events
|= COMEDI_CB_ERROR
;
332 comedi_event(dev
, s
);
333 /* don't do an unlink here */
334 usbdux_ai_stop(dev
, 0);
338 devpriv
->ai_counter
--;
339 if (likely(devpriv
->ai_counter
> 0))
342 /* timer zero, transfer measurements to comedi */
343 devpriv
->ai_counter
= devpriv
->ai_timer
;
345 /* test, if we transmit only a fixed number of samples */
346 if (cmd
->stop_src
== TRIG_COUNT
) {
347 /* not continuous, fixed number of samples */
348 devpriv
->ai_sample_count
--;
349 /* all samples received? */
350 if (devpriv
->ai_sample_count
< 0) {
351 /* prevent a resubmit next time */
352 usbdux_ai_stop(dev
, 0);
353 /* say comedi that the acquistion is over */
354 s
->async
->events
|= COMEDI_CB_EOA
;
355 comedi_event(dev
, s
);
359 /* get the data from the USB bus and hand it over to comedi */
360 for (i
= 0; i
< cmd
->chanlist_len
; i
++) {
361 unsigned int range
= CR_RANGE(cmd
->chanlist
[i
]);
362 uint16_t val
= le16_to_cpu(devpriv
->in_buf
[i
]);
364 /* bipolar data is two's-complement */
365 if (comedi_range_is_bipolar(s
, range
))
366 val
^= ((s
->maxdata
+ 1) >> 1);
369 err
= comedi_buf_put(s
, val
);
370 if (unlikely(err
== 0)) {
371 /* buffer overflow */
372 usbdux_ai_stop(dev
, 0);
376 /* tell comedi that data is there */
377 s
->async
->events
|= COMEDI_CB_BLOCK
| COMEDI_CB_EOS
;
378 comedi_event(dev
, s
);
381 static void usbdux_ao_stop(struct comedi_device
*dev
, int do_unlink
)
383 struct usbdux_private
*devpriv
= dev
->private;
385 if (do_unlink
&& devpriv
->ao_urbs
)
386 usbdux_unlink_urbs(devpriv
->ao_urbs
, devpriv
->n_ao_urbs
);
388 devpriv
->ao_cmd_running
= 0;
391 static int usbdux_ao_cancel(struct comedi_device
*dev
,
392 struct comedi_subdevice
*s
)
394 struct usbdux_private
*devpriv
= dev
->private;
396 /* prevent other CPUs from submitting a command just now */
398 /* unlink only if it is really running */
399 usbdux_ao_stop(dev
, devpriv
->ao_cmd_running
);
405 static void usbduxsub_ao_isoc_irq(struct urb
*urb
)
407 struct comedi_device
*dev
= urb
->context
;
408 struct comedi_subdevice
*s
= dev
->write_subdev
;
409 struct usbdux_private
*devpriv
= dev
->private;
410 struct comedi_cmd
*cmd
= &s
->async
->cmd
;
415 switch (urb
->status
) {
424 /* after an unlink command, unplug, ... etc */
425 /* no unlink needed here. Already shutting down. */
426 if (devpriv
->ao_cmd_running
) {
427 s
->async
->events
|= COMEDI_CB_EOA
;
428 comedi_event(dev
, s
);
429 usbdux_ao_stop(dev
, 0);
435 if (devpriv
->ao_cmd_running
) {
436 dev_err(dev
->class_dev
,
437 "Non-zero urb status received in ao intr context: %d\n",
439 s
->async
->events
|= COMEDI_CB_ERROR
;
440 s
->async
->events
|= COMEDI_CB_EOA
;
441 comedi_event(dev
, s
);
442 /* we do an unlink if we are in the high speed mode */
443 usbdux_ao_stop(dev
, 0);
448 /* are we actually running? */
449 if (!devpriv
->ao_cmd_running
)
452 /* normal operation: executing a command in this subdevice */
453 devpriv
->ao_counter
--;
454 if ((int)devpriv
->ao_counter
<= 0) {
456 devpriv
->ao_counter
= devpriv
->ao_timer
;
458 /* handle non continous acquisition */
459 if (cmd
->stop_src
== TRIG_COUNT
) {
460 /* fixed number of samples */
461 devpriv
->ao_sample_count
--;
462 if (devpriv
->ao_sample_count
< 0) {
463 /* all samples transmitted */
464 usbdux_ao_stop(dev
, 0);
465 s
->async
->events
|= COMEDI_CB_EOA
;
466 comedi_event(dev
, s
);
467 /* no resubmit of the urb */
472 /* transmit data to the USB bus */
473 datap
= urb
->transfer_buffer
;
474 *datap
++ = cmd
->chanlist_len
;
475 for (i
= 0; i
< cmd
->chanlist_len
; i
++) {
476 unsigned int chan
= CR_CHAN(cmd
->chanlist
[i
]);
479 ret
= comedi_buf_get(s
, &val
);
481 dev_err(dev
->class_dev
, "buffer underflow\n");
482 s
->async
->events
|= (COMEDI_CB_EOA
|
485 /* pointer to the DA */
486 *datap
++ = val
& 0xff;
487 *datap
++ = (val
>> 8) & 0xff;
488 *datap
++ = chan
<< 6;
489 s
->readback
[chan
] = val
;
491 s
->async
->events
|= COMEDI_CB_BLOCK
;
492 comedi_event(dev
, s
);
495 urb
->transfer_buffer_length
= SIZEOUTBUF
;
496 urb
->dev
= comedi_to_usb_dev(dev
);
498 if (devpriv
->ao_cmd_running
) {
499 if (devpriv
->high_speed
)
500 urb
->interval
= 8; /* uframes */
502 urb
->interval
= 1; /* frames */
503 urb
->number_of_packets
= 1;
504 urb
->iso_frame_desc
[0].offset
= 0;
505 urb
->iso_frame_desc
[0].length
= SIZEOUTBUF
;
506 urb
->iso_frame_desc
[0].status
= 0;
507 ret
= usb_submit_urb(urb
, GFP_ATOMIC
);
509 dev_err(dev
->class_dev
,
510 "ao urb resubm failed in int-cont. ret=%d",
512 if (ret
== -EL2NSYNC
)
513 dev_err(dev
->class_dev
,
514 "buggy USB host controller or bug in IRQ handling!\n");
516 s
->async
->events
|= COMEDI_CB_EOA
;
517 s
->async
->events
|= COMEDI_CB_ERROR
;
518 comedi_event(dev
, s
);
519 /* don't do an unlink here */
520 usbdux_ao_stop(dev
, 0);
525 static int usbdux_submit_urbs(struct comedi_device
*dev
,
526 struct urb
**urbs
, int num_urbs
,
529 struct usb_device
*usb
= comedi_to_usb_dev(dev
);
530 struct usbdux_private
*devpriv
= dev
->private;
535 /* Submit all URBs and start the transfer on the bus */
536 for (i
= 0; i
< num_urbs
; i
++) {
539 /* in case of a resubmission after an unlink... */
541 urb
->interval
= devpriv
->ai_interval
;
545 urb
->transfer_flags
= URB_ISO_ASAP
;
547 ret
= usb_submit_urb(urb
, GFP_ATOMIC
);
554 static int usbdux_ai_cmdtest(struct comedi_device
*dev
,
555 struct comedi_subdevice
*s
, struct comedi_cmd
*cmd
)
557 struct usbdux_private
*this_usbduxsub
= dev
->private;
559 unsigned int tmp_timer
;
561 /* Step 1 : check if triggers are trivially valid */
563 err
|= cfc_check_trigger_src(&cmd
->start_src
, TRIG_NOW
| TRIG_INT
);
564 err
|= cfc_check_trigger_src(&cmd
->scan_begin_src
, TRIG_TIMER
);
565 err
|= cfc_check_trigger_src(&cmd
->convert_src
, TRIG_NOW
);
566 err
|= cfc_check_trigger_src(&cmd
->scan_end_src
, TRIG_COUNT
);
567 err
|= cfc_check_trigger_src(&cmd
->stop_src
, TRIG_COUNT
| TRIG_NONE
);
572 /* Step 2a : make sure trigger sources are unique */
574 err
|= cfc_check_trigger_is_unique(cmd
->start_src
);
575 err
|= cfc_check_trigger_is_unique(cmd
->stop_src
);
577 /* Step 2b : and mutually compatible */
582 /* Step 3: check if arguments are trivially valid */
584 err
|= cfc_check_trigger_arg_is(&cmd
->start_arg
, 0);
586 if (cmd
->scan_begin_src
== TRIG_FOLLOW
) /* internal trigger */
587 err
|= cfc_check_trigger_arg_is(&cmd
->scan_begin_arg
, 0);
589 if (cmd
->scan_begin_src
== TRIG_TIMER
) {
590 if (this_usbduxsub
->high_speed
) {
592 * In high speed mode microframes are possible.
593 * However, during one microframe we can roughly
594 * sample one channel. Thus, the more channels
595 * are in the channel list the more time we need.
598 /* find a power of 2 for the number of channels */
599 while (i
< (cmd
->chanlist_len
))
602 err
|= cfc_check_trigger_arg_min(&cmd
->scan_begin_arg
,
604 /* now calc the real sampling rate with all the
607 ((unsigned int)(cmd
->scan_begin_arg
/ 125000)) *
611 /* 1kHz scans every USB frame */
612 err
|= cfc_check_trigger_arg_min(&cmd
->scan_begin_arg
,
615 * calc the real sampling rate with the rounding errors
617 tmp_timer
= ((unsigned int)(cmd
->scan_begin_arg
/
620 err
|= cfc_check_trigger_arg_is(&cmd
->scan_begin_arg
,
624 err
|= cfc_check_trigger_arg_is(&cmd
->scan_end_arg
, cmd
->chanlist_len
);
626 if (cmd
->stop_src
== TRIG_COUNT
) {
627 /* any count is allowed */
630 err
|= cfc_check_trigger_arg_is(&cmd
->stop_arg
, 0);
640 * creates the ADC command for the MAX1271
641 * range is the range value from comedi
643 static uint8_t create_adc_command(unsigned int chan
, unsigned int range
)
645 uint8_t p
= (range
<= 1);
646 uint8_t r
= ((range
% 2) == 0);
648 return (chan
<< 4) | ((p
== 1) << 2) | ((r
== 1) << 3);
651 static int send_dux_commands(struct comedi_device
*dev
, unsigned int cmd_type
)
653 struct usb_device
*usb
= comedi_to_usb_dev(dev
);
654 struct usbdux_private
*devpriv
= dev
->private;
657 devpriv
->dux_commands
[0] = cmd_type
;
659 return usb_bulk_msg(usb
, usb_sndbulkpipe(usb
, 1),
660 devpriv
->dux_commands
, SIZEOFDUXBUFFER
,
661 &nsent
, BULK_TIMEOUT
);
664 static int receive_dux_commands(struct comedi_device
*dev
, unsigned int command
)
666 struct usb_device
*usb
= comedi_to_usb_dev(dev
);
667 struct usbdux_private
*devpriv
= dev
->private;
672 for (i
= 0; i
< RETRIES
; i
++) {
673 ret
= usb_bulk_msg(usb
, usb_rcvbulkpipe(usb
, 8),
674 devpriv
->insn_buf
, SIZEINSNBUF
,
675 &nrec
, BULK_TIMEOUT
);
678 if (le16_to_cpu(devpriv
->insn_buf
[0]) == command
)
681 /* command not received */
685 static int usbdux_ai_inttrig(struct comedi_device
*dev
,
686 struct comedi_subdevice
*s
,
687 unsigned int trig_num
)
689 struct usbdux_private
*devpriv
= dev
->private;
690 struct comedi_cmd
*cmd
= &s
->async
->cmd
;
693 if (trig_num
!= cmd
->start_arg
)
698 if (!devpriv
->ai_cmd_running
) {
699 devpriv
->ai_cmd_running
= 1;
700 ret
= usbdux_submit_urbs(dev
, devpriv
->ai_urbs
,
701 devpriv
->n_ai_urbs
, 1);
703 devpriv
->ai_cmd_running
= 0;
706 s
->async
->inttrig
= NULL
;
716 static int usbdux_ai_cmd(struct comedi_device
*dev
, struct comedi_subdevice
*s
)
718 struct usbdux_private
*devpriv
= dev
->private;
719 struct comedi_cmd
*cmd
= &s
->async
->cmd
;
720 int len
= cmd
->chanlist_len
;
724 /* block other CPUs from starting an ai_cmd */
727 if (devpriv
->ai_cmd_running
)
730 /* set current channel of the running acquisition to zero */
731 s
->async
->cur_chan
= 0;
733 devpriv
->dux_commands
[1] = len
;
734 for (i
= 0; i
< len
; ++i
) {
735 unsigned int chan
= CR_CHAN(cmd
->chanlist
[i
]);
736 unsigned int range
= CR_RANGE(cmd
->chanlist
[i
]);
738 devpriv
->dux_commands
[i
+ 2] = create_adc_command(chan
, range
);
741 ret
= send_dux_commands(dev
, USBDUX_CMD_MULT_AI
);
745 if (devpriv
->high_speed
) {
747 * every channel gets a time window of 125us. Thus, if we
748 * sample all 8 channels we need 1ms. If we sample only one
749 * channel we need only 125us
751 devpriv
->ai_interval
= 1;
752 /* find a power of 2 for the interval */
753 while (devpriv
->ai_interval
< len
)
754 devpriv
->ai_interval
*= 2;
756 devpriv
->ai_timer
= cmd
->scan_begin_arg
/
757 (125000 * devpriv
->ai_interval
);
759 /* interval always 1ms */
760 devpriv
->ai_interval
= 1;
761 devpriv
->ai_timer
= cmd
->scan_begin_arg
/ 1000000;
763 if (devpriv
->ai_timer
< 1) {
768 devpriv
->ai_counter
= devpriv
->ai_timer
;
770 if (cmd
->stop_src
== TRIG_COUNT
) {
771 /* data arrives as one packet */
772 devpriv
->ai_sample_count
= cmd
->stop_arg
;
774 /* continous acquisition */
775 devpriv
->ai_sample_count
= 0;
778 if (cmd
->start_src
== TRIG_NOW
) {
779 /* enable this acquisition operation */
780 devpriv
->ai_cmd_running
= 1;
781 ret
= usbdux_submit_urbs(dev
, devpriv
->ai_urbs
,
782 devpriv
->n_ai_urbs
, 1);
784 devpriv
->ai_cmd_running
= 0;
785 /* fixme: unlink here?? */
788 s
->async
->inttrig
= NULL
;
791 /* don't enable the acquision operation */
792 /* wait for an internal signal */
793 s
->async
->inttrig
= usbdux_ai_inttrig
;
802 /* Mode 0 is used to get a single conversion on demand */
803 static int usbdux_ai_insn_read(struct comedi_device
*dev
,
804 struct comedi_subdevice
*s
,
805 struct comedi_insn
*insn
,
808 struct usbdux_private
*devpriv
= dev
->private;
809 unsigned int chan
= CR_CHAN(insn
->chanspec
);
810 unsigned int range
= CR_RANGE(insn
->chanspec
);
817 if (devpriv
->ai_cmd_running
)
820 /* set command for the first channel */
821 devpriv
->dux_commands
[1] = create_adc_command(chan
, range
);
824 ret
= send_dux_commands(dev
, USBDUX_CMD_SINGLE_AI
);
828 for (i
= 0; i
< insn
->n
; i
++) {
829 ret
= receive_dux_commands(dev
, USBDUX_CMD_SINGLE_AI
);
833 val
= le16_to_cpu(devpriv
->insn_buf
[1]);
835 /* bipolar data is two's-complement */
836 if (comedi_range_is_bipolar(s
, range
))
837 val
^= ((s
->maxdata
+ 1) >> 1);
845 return ret
? ret
: insn
->n
;
848 static int usbdux_ao_insn_read(struct comedi_device
*dev
,
849 struct comedi_subdevice
*s
,
850 struct comedi_insn
*insn
,
853 struct usbdux_private
*devpriv
= dev
->private;
857 ret
= comedi_readback_insn_read(dev
, s
, insn
, data
);
863 static int usbdux_ao_insn_write(struct comedi_device
*dev
,
864 struct comedi_subdevice
*s
,
865 struct comedi_insn
*insn
,
868 struct usbdux_private
*devpriv
= dev
->private;
869 unsigned int chan
= CR_CHAN(insn
->chanspec
);
870 unsigned int val
= s
->readback
[chan
];
871 uint16_t *p
= (uint16_t *)&devpriv
->dux_commands
[2];
877 if (devpriv
->ao_cmd_running
)
880 /* number of channels: 1 */
881 devpriv
->dux_commands
[1] = 1;
883 devpriv
->dux_commands
[4] = chan
<< 6;
885 for (i
= 0; i
< insn
->n
; i
++) {
888 /* one 16 bit value */
889 *p
= cpu_to_le16(val
);
891 ret
= send_dux_commands(dev
, USBDUX_CMD_AO
);
895 s
->readback
[chan
] = val
;
901 return ret
? ret
: insn
->n
;
904 static int usbdux_ao_inttrig(struct comedi_device
*dev
,
905 struct comedi_subdevice
*s
,
906 unsigned int trig_num
)
908 struct usbdux_private
*devpriv
= dev
->private;
909 struct comedi_cmd
*cmd
= &s
->async
->cmd
;
912 if (trig_num
!= cmd
->start_arg
)
917 if (!devpriv
->ao_cmd_running
) {
918 devpriv
->ao_cmd_running
= 1;
919 ret
= usbdux_submit_urbs(dev
, devpriv
->ao_urbs
,
920 devpriv
->n_ao_urbs
, 0);
922 devpriv
->ao_cmd_running
= 0;
925 s
->async
->inttrig
= NULL
;
935 static int usbdux_ao_cmdtest(struct comedi_device
*dev
,
936 struct comedi_subdevice
*s
, struct comedi_cmd
*cmd
)
938 struct usbdux_private
*this_usbduxsub
= dev
->private;
945 /* Step 1 : check if triggers are trivially valid */
947 err
|= cfc_check_trigger_src(&cmd
->start_src
, TRIG_NOW
| TRIG_INT
);
949 if (0) { /* (this_usbduxsub->high_speed) */
950 /* the sampling rate is set by the coversion rate */
953 /* start a new scan (output at once) with a timer */
956 err
|= cfc_check_trigger_src(&cmd
->scan_begin_src
, flags
);
958 if (0) { /* (this_usbduxsub->high_speed) */
960 * in usb-2.0 only one conversion it transmitted
966 * all conversion events happen simultaneously with
971 err
|= cfc_check_trigger_src(&cmd
->convert_src
, flags
);
973 err
|= cfc_check_trigger_src(&cmd
->scan_end_src
, TRIG_COUNT
);
974 err
|= cfc_check_trigger_src(&cmd
->stop_src
, TRIG_COUNT
| TRIG_NONE
);
979 /* Step 2a : make sure trigger sources are unique */
981 err
|= cfc_check_trigger_is_unique(cmd
->start_src
);
982 err
|= cfc_check_trigger_is_unique(cmd
->stop_src
);
984 /* Step 2b : and mutually compatible */
989 /* Step 3: check if arguments are trivially valid */
991 err
|= cfc_check_trigger_arg_is(&cmd
->start_arg
, 0);
993 if (cmd
->scan_begin_src
== TRIG_FOLLOW
) /* internal trigger */
994 err
|= cfc_check_trigger_arg_is(&cmd
->scan_begin_arg
, 0);
996 if (cmd
->scan_begin_src
== TRIG_TIMER
)
997 err
|= cfc_check_trigger_arg_min(&cmd
->scan_begin_arg
,
1000 /* not used now, is for later use */
1001 if (cmd
->convert_src
== TRIG_TIMER
)
1002 err
|= cfc_check_trigger_arg_min(&cmd
->convert_arg
, 125000);
1004 err
|= cfc_check_trigger_arg_is(&cmd
->scan_end_arg
, cmd
->chanlist_len
);
1006 if (cmd
->stop_src
== TRIG_COUNT
) {
1007 /* any count is allowed */
1010 err
|= cfc_check_trigger_arg_is(&cmd
->stop_arg
, 0);
1019 static int usbdux_ao_cmd(struct comedi_device
*dev
, struct comedi_subdevice
*s
)
1021 struct usbdux_private
*devpriv
= dev
->private;
1022 struct comedi_cmd
*cmd
= &s
->async
->cmd
;
1025 down(&devpriv
->sem
);
1027 if (devpriv
->ao_cmd_running
)
1030 /* set current channel of the running acquisition to zero */
1031 s
->async
->cur_chan
= 0;
1033 /* we count in steps of 1ms (125us) */
1034 /* 125us mode not used yet */
1035 if (0) { /* (devpriv->high_speed) */
1037 /* timing of the conversion itself: every 125 us */
1038 devpriv
->ao_timer
= cmd
->convert_arg
/ 125000;
1041 /* timing of the scan: we get all channels at once */
1042 devpriv
->ao_timer
= cmd
->scan_begin_arg
/ 1000000;
1043 if (devpriv
->ao_timer
< 1) {
1049 devpriv
->ao_counter
= devpriv
->ao_timer
;
1051 if (cmd
->stop_src
== TRIG_COUNT
) {
1052 /* not continuous */
1054 /* high speed also scans everything at once */
1055 if (0) { /* (devpriv->high_speed) */
1056 devpriv
->ao_sample_count
= cmd
->stop_arg
*
1059 /* there's no scan as the scan has been */
1060 /* perf inside the FX2 */
1061 /* data arrives as one packet */
1062 devpriv
->ao_sample_count
= cmd
->stop_arg
;
1065 /* continous acquisition */
1066 devpriv
->ao_sample_count
= 0;
1069 if (cmd
->start_src
== TRIG_NOW
) {
1070 /* enable this acquisition operation */
1071 devpriv
->ao_cmd_running
= 1;
1072 ret
= usbdux_submit_urbs(dev
, devpriv
->ao_urbs
,
1073 devpriv
->n_ao_urbs
, 0);
1075 devpriv
->ao_cmd_running
= 0;
1076 /* fixme: unlink here?? */
1079 s
->async
->inttrig
= NULL
;
1082 /* submit the urbs later */
1083 /* wait for an internal signal */
1084 s
->async
->inttrig
= usbdux_ao_inttrig
;
1093 static int usbdux_dio_insn_config(struct comedi_device
*dev
,
1094 struct comedi_subdevice
*s
,
1095 struct comedi_insn
*insn
,
1100 ret
= comedi_dio_insn_config(dev
, s
, insn
, data
, 0);
1105 * We don't tell the firmware here as it would take 8 frames
1106 * to submit the information. We do it in the insn_bits.
1111 static int usbdux_dio_insn_bits(struct comedi_device
*dev
,
1112 struct comedi_subdevice
*s
,
1113 struct comedi_insn
*insn
,
1117 struct usbdux_private
*devpriv
= dev
->private;
1120 down(&devpriv
->sem
);
1122 comedi_dio_update_state(s
, data
);
1124 /* Always update the hardware. See the (*insn_config). */
1125 devpriv
->dux_commands
[1] = s
->io_bits
;
1126 devpriv
->dux_commands
[2] = s
->state
;
1129 * This command also tells the firmware to return
1130 * the digital input lines.
1132 ret
= send_dux_commands(dev
, USBDUX_CMD_DIO_BITS
);
1135 ret
= receive_dux_commands(dev
, USBDUX_CMD_DIO_BITS
);
1139 data
[1] = le16_to_cpu(devpriv
->insn_buf
[1]);
1144 return ret
? ret
: insn
->n
;
1147 static int usbdux_counter_read(struct comedi_device
*dev
,
1148 struct comedi_subdevice
*s
,
1149 struct comedi_insn
*insn
,
1152 struct usbdux_private
*devpriv
= dev
->private;
1153 unsigned int chan
= CR_CHAN(insn
->chanspec
);
1157 down(&devpriv
->sem
);
1159 for (i
= 0; i
< insn
->n
; i
++) {
1160 ret
= send_dux_commands(dev
, USBDUX_CMD_TIMER_RD
);
1162 goto counter_read_exit
;
1163 ret
= receive_dux_commands(dev
, USBDUX_CMD_TIMER_RD
);
1165 goto counter_read_exit
;
1167 data
[i
] = le16_to_cpu(devpriv
->insn_buf
[chan
+ 1]);
1173 return ret
? ret
: insn
->n
;
1176 static int usbdux_counter_write(struct comedi_device
*dev
,
1177 struct comedi_subdevice
*s
,
1178 struct comedi_insn
*insn
,
1181 struct usbdux_private
*devpriv
= dev
->private;
1182 unsigned int chan
= CR_CHAN(insn
->chanspec
);
1183 uint16_t *p
= (uint16_t *)&devpriv
->dux_commands
[2];
1187 down(&devpriv
->sem
);
1189 devpriv
->dux_commands
[1] = chan
;
1191 for (i
= 0; i
< insn
->n
; i
++) {
1192 *p
= cpu_to_le16(data
[i
]);
1194 ret
= send_dux_commands(dev
, USBDUX_CMD_TIMER_WR
);
1201 return ret
? ret
: insn
->n
;
1204 static int usbdux_counter_config(struct comedi_device
*dev
,
1205 struct comedi_subdevice
*s
,
1206 struct comedi_insn
*insn
, unsigned int *data
)
1208 /* nothing to do so far */
1212 static void usbduxsub_unlink_pwm_urbs(struct comedi_device
*dev
)
1214 struct usbdux_private
*devpriv
= dev
->private;
1216 usb_kill_urb(devpriv
->pwm_urb
);
1219 static void usbdux_pwm_stop(struct comedi_device
*dev
, int do_unlink
)
1221 struct usbdux_private
*devpriv
= dev
->private;
1224 usbduxsub_unlink_pwm_urbs(dev
);
1226 devpriv
->pwm_cmd_running
= 0;
1229 static int usbdux_pwm_cancel(struct comedi_device
*dev
,
1230 struct comedi_subdevice
*s
)
1232 struct usbdux_private
*devpriv
= dev
->private;
1235 down(&devpriv
->sem
);
1236 /* unlink only if it is really running */
1237 usbdux_pwm_stop(dev
, devpriv
->pwm_cmd_running
);
1238 ret
= send_dux_commands(dev
, USBDUX_CMD_PWM_OFF
);
1244 static void usbduxsub_pwm_irq(struct urb
*urb
)
1246 struct comedi_device
*dev
= urb
->context
;
1247 struct usbdux_private
*devpriv
= dev
->private;
1250 switch (urb
->status
) {
1260 * after an unlink command, unplug, ... etc
1261 * no unlink needed here. Already shutting down.
1263 if (devpriv
->pwm_cmd_running
)
1264 usbdux_pwm_stop(dev
, 0);
1270 if (devpriv
->pwm_cmd_running
) {
1271 dev_err(dev
->class_dev
,
1272 "Non-zero urb status received in pwm intr context: %d\n",
1274 usbdux_pwm_stop(dev
, 0);
1279 /* are we actually running? */
1280 if (!devpriv
->pwm_cmd_running
)
1283 urb
->transfer_buffer_length
= devpriv
->pwm_buf_sz
;
1284 urb
->dev
= comedi_to_usb_dev(dev
);
1286 if (devpriv
->pwm_cmd_running
) {
1287 ret
= usb_submit_urb(urb
, GFP_ATOMIC
);
1289 dev_err(dev
->class_dev
,
1290 "pwm urb resubm failed in int-cont. ret=%d",
1292 if (ret
== -EL2NSYNC
)
1293 dev_err(dev
->class_dev
,
1294 "buggy USB host controller or bug in IRQ handling!\n");
1296 /* don't do an unlink here */
1297 usbdux_pwm_stop(dev
, 0);
1302 static int usbduxsub_submit_pwm_urbs(struct comedi_device
*dev
)
1304 struct usb_device
*usb
= comedi_to_usb_dev(dev
);
1305 struct usbdux_private
*devpriv
= dev
->private;
1306 struct urb
*urb
= devpriv
->pwm_urb
;
1308 /* in case of a resubmission after an unlink... */
1309 usb_fill_bulk_urb(urb
, usb
, usb_sndbulkpipe(usb
, 4),
1310 urb
->transfer_buffer
,
1311 devpriv
->pwm_buf_sz
,
1315 return usb_submit_urb(urb
, GFP_ATOMIC
);
1318 static int usbdux_pwm_period(struct comedi_device
*dev
,
1319 struct comedi_subdevice
*s
,
1320 unsigned int period
)
1322 struct usbdux_private
*devpriv
= dev
->private;
1325 if (period
< MIN_PWM_PERIOD
)
1328 fx2delay
= (period
/ (6 * 512 * 1000 / 33)) - 6;
1332 devpriv
->pwm_delay
= fx2delay
;
1333 devpriv
->pwm_period
= period
;
1338 static int usbdux_pwm_start(struct comedi_device
*dev
,
1339 struct comedi_subdevice
*s
)
1341 struct usbdux_private
*devpriv
= dev
->private;
1344 down(&devpriv
->sem
);
1346 if (devpriv
->pwm_cmd_running
)
1347 goto pwm_start_exit
;
1349 devpriv
->dux_commands
[1] = devpriv
->pwm_delay
;
1350 ret
= send_dux_commands(dev
, USBDUX_CMD_PWM_ON
);
1352 goto pwm_start_exit
;
1354 /* initialise the buffer */
1355 memset(devpriv
->pwm_urb
->transfer_buffer
, 0, devpriv
->pwm_buf_sz
);
1357 devpriv
->pwm_cmd_running
= 1;
1358 ret
= usbduxsub_submit_pwm_urbs(dev
);
1360 devpriv
->pwm_cmd_running
= 0;
1368 static void usbdux_pwm_pattern(struct comedi_device
*dev
,
1369 struct comedi_subdevice
*s
,
1374 struct usbdux_private
*devpriv
= dev
->private;
1375 char pwm_mask
= (1 << chan
); /* DIO bit for the PWM data */
1376 char sgn_mask
= (16 << chan
); /* DIO bit for the sign */
1377 char *buf
= (char *)(devpriv
->pwm_urb
->transfer_buffer
);
1378 int szbuf
= devpriv
->pwm_buf_sz
;
1381 for (i
= 0; i
< szbuf
; i
++) {
1395 static int usbdux_pwm_write(struct comedi_device
*dev
,
1396 struct comedi_subdevice
*s
,
1397 struct comedi_insn
*insn
,
1400 unsigned int chan
= CR_CHAN(insn
->chanspec
);
1403 * It doesn't make sense to support more than one value here
1404 * because it would just overwrite the PWM buffer.
1410 * The sign is set via a special INSN only, this gives us 8 bits
1411 * for normal operation, sign is 0 by default.
1413 usbdux_pwm_pattern(dev
, s
, chan
, data
[0], 0);
1418 static int usbdux_pwm_config(struct comedi_device
*dev
,
1419 struct comedi_subdevice
*s
,
1420 struct comedi_insn
*insn
,
1423 struct usbdux_private
*devpriv
= dev
->private;
1424 unsigned int chan
= CR_CHAN(insn
->chanspec
);
1427 case INSN_CONFIG_ARM
:
1429 * if not zero the PWM is limited to a certain time which is
1430 * not supported here
1434 return usbdux_pwm_start(dev
, s
);
1435 case INSN_CONFIG_DISARM
:
1436 return usbdux_pwm_cancel(dev
, s
);
1437 case INSN_CONFIG_GET_PWM_STATUS
:
1438 data
[1] = devpriv
->pwm_cmd_running
;
1440 case INSN_CONFIG_PWM_SET_PERIOD
:
1441 return usbdux_pwm_period(dev
, s
, data
[1]);
1442 case INSN_CONFIG_PWM_GET_PERIOD
:
1443 data
[1] = devpriv
->pwm_period
;
1445 case INSN_CONFIG_PWM_SET_H_BRIDGE
:
1448 * data[2] = sign (for a relay)
1450 usbdux_pwm_pattern(dev
, s
, chan
, data
[1], (data
[2] != 0));
1452 case INSN_CONFIG_PWM_GET_H_BRIDGE
:
1453 /* values are not kept in this driver, nothing to return here */
1459 static int usbdux_firmware_upload(struct comedi_device
*dev
,
1460 const u8
*data
, size_t size
,
1461 unsigned long context
)
1463 struct usb_device
*usb
= comedi_to_usb_dev(dev
);
1471 if (size
> USBDUX_FIRMWARE_MAX_LEN
) {
1472 dev_err(dev
->class_dev
,
1473 "usbdux firmware binary it too large for FX2.\n");
1477 /* we generate a local buffer for the firmware */
1478 buf
= kmemdup(data
, size
, GFP_KERNEL
);
1482 /* we need a malloc'ed buffer for usb_control_msg() */
1483 tmp
= kmalloc(1, GFP_KERNEL
);
1489 /* stop the current firmware on the device */
1490 *tmp
= 1; /* 7f92 to one */
1491 ret
= usb_control_msg(usb
, usb_sndctrlpipe(usb
, 0),
1492 USBDUX_FIRMWARE_CMD
,
1494 USBDUX_CPU_CS
, 0x0000,
1498 dev_err(dev
->class_dev
, "can not stop firmware\n");
1502 /* upload the new firmware to the device */
1503 ret
= usb_control_msg(usb
, usb_sndctrlpipe(usb
, 0),
1504 USBDUX_FIRMWARE_CMD
,
1510 dev_err(dev
->class_dev
, "firmware upload failed\n");
1514 /* start the new firmware on the device */
1515 *tmp
= 0; /* 7f92 to zero */
1516 ret
= usb_control_msg(usb
, usb_sndctrlpipe(usb
, 0),
1517 USBDUX_FIRMWARE_CMD
,
1519 USBDUX_CPU_CS
, 0x0000,
1523 dev_err(dev
->class_dev
, "can not start firmware\n");
1531 static int usbdux_alloc_usb_buffers(struct comedi_device
*dev
)
1533 struct usb_device
*usb
= comedi_to_usb_dev(dev
);
1534 struct usbdux_private
*devpriv
= dev
->private;
1538 devpriv
->dux_commands
= kzalloc(SIZEOFDUXBUFFER
, GFP_KERNEL
);
1539 devpriv
->in_buf
= kzalloc(SIZEINBUF
, GFP_KERNEL
);
1540 devpriv
->insn_buf
= kzalloc(SIZEINSNBUF
, GFP_KERNEL
);
1541 devpriv
->ai_urbs
= kcalloc(devpriv
->n_ai_urbs
, sizeof(void *),
1543 devpriv
->ao_urbs
= kcalloc(devpriv
->n_ao_urbs
, sizeof(void *),
1545 if (!devpriv
->dux_commands
|| !devpriv
->in_buf
|| !devpriv
->insn_buf
||
1546 !devpriv
->ai_urbs
|| !devpriv
->ao_urbs
)
1549 for (i
= 0; i
< devpriv
->n_ai_urbs
; i
++) {
1550 /* one frame: 1ms */
1551 urb
= usb_alloc_urb(1, GFP_KERNEL
);
1554 devpriv
->ai_urbs
[i
] = urb
;
1558 urb
->pipe
= usb_rcvisocpipe(usb
, 6);
1559 urb
->transfer_flags
= URB_ISO_ASAP
;
1560 urb
->transfer_buffer
= kzalloc(SIZEINBUF
, GFP_KERNEL
);
1561 if (!urb
->transfer_buffer
)
1564 urb
->complete
= usbduxsub_ai_isoc_irq
;
1565 urb
->number_of_packets
= 1;
1566 urb
->transfer_buffer_length
= SIZEINBUF
;
1567 urb
->iso_frame_desc
[0].offset
= 0;
1568 urb
->iso_frame_desc
[0].length
= SIZEINBUF
;
1571 for (i
= 0; i
< devpriv
->n_ao_urbs
; i
++) {
1572 /* one frame: 1ms */
1573 urb
= usb_alloc_urb(1, GFP_KERNEL
);
1576 devpriv
->ao_urbs
[i
] = urb
;
1580 urb
->pipe
= usb_sndisocpipe(usb
, 2);
1581 urb
->transfer_flags
= URB_ISO_ASAP
;
1582 urb
->transfer_buffer
= kzalloc(SIZEOUTBUF
, GFP_KERNEL
);
1583 if (!urb
->transfer_buffer
)
1586 urb
->complete
= usbduxsub_ao_isoc_irq
;
1587 urb
->number_of_packets
= 1;
1588 urb
->transfer_buffer_length
= SIZEOUTBUF
;
1589 urb
->iso_frame_desc
[0].offset
= 0;
1590 urb
->iso_frame_desc
[0].length
= SIZEOUTBUF
;
1591 if (devpriv
->high_speed
)
1592 urb
->interval
= 8; /* uframes */
1594 urb
->interval
= 1; /* frames */
1598 if (devpriv
->pwm_buf_sz
) {
1599 urb
= usb_alloc_urb(0, GFP_KERNEL
);
1602 devpriv
->pwm_urb
= urb
;
1604 /* max bulk ep size in high speed */
1605 urb
->transfer_buffer
= kzalloc(devpriv
->pwm_buf_sz
,
1607 if (!urb
->transfer_buffer
)
1614 static void usbdux_free_usb_buffers(struct comedi_device
*dev
)
1616 struct usbdux_private
*devpriv
= dev
->private;
1620 urb
= devpriv
->pwm_urb
;
1622 kfree(urb
->transfer_buffer
);
1625 if (devpriv
->ao_urbs
) {
1626 for (i
= 0; i
< devpriv
->n_ao_urbs
; i
++) {
1627 urb
= devpriv
->ao_urbs
[i
];
1629 kfree(urb
->transfer_buffer
);
1633 kfree(devpriv
->ao_urbs
);
1635 if (devpriv
->ai_urbs
) {
1636 for (i
= 0; i
< devpriv
->n_ai_urbs
; i
++) {
1637 urb
= devpriv
->ai_urbs
[i
];
1639 kfree(urb
->transfer_buffer
);
1643 kfree(devpriv
->ai_urbs
);
1645 kfree(devpriv
->insn_buf
);
1646 kfree(devpriv
->in_buf
);
1647 kfree(devpriv
->dux_commands
);
1650 static int usbdux_auto_attach(struct comedi_device
*dev
,
1651 unsigned long context_unused
)
1653 struct usb_interface
*intf
= comedi_to_usb_interface(dev
);
1654 struct usb_device
*usb
= comedi_to_usb_dev(dev
);
1655 struct usbdux_private
*devpriv
;
1656 struct comedi_subdevice
*s
;
1659 devpriv
= comedi_alloc_devpriv(dev
, sizeof(*devpriv
));
1663 sema_init(&devpriv
->sem
, 1);
1665 usb_set_intfdata(intf
, devpriv
);
1667 devpriv
->high_speed
= (usb
->speed
== USB_SPEED_HIGH
);
1668 if (devpriv
->high_speed
) {
1669 devpriv
->n_ai_urbs
= NUMOFINBUFFERSHIGH
;
1670 devpriv
->n_ao_urbs
= NUMOFOUTBUFFERSHIGH
;
1671 devpriv
->pwm_buf_sz
= 512;
1673 devpriv
->n_ai_urbs
= NUMOFINBUFFERSFULL
;
1674 devpriv
->n_ao_urbs
= NUMOFOUTBUFFERSFULL
;
1677 ret
= usbdux_alloc_usb_buffers(dev
);
1681 /* setting to alternate setting 3: enabling iso ep and bulk ep. */
1682 ret
= usb_set_interface(usb
, intf
->altsetting
->desc
.bInterfaceNumber
,
1685 dev_err(dev
->class_dev
,
1686 "could not set alternate setting 3 in high speed\n");
1690 ret
= comedi_load_firmware(dev
, &usb
->dev
, USBDUX_FIRMWARE
,
1691 usbdux_firmware_upload
, 0);
1695 ret
= comedi_alloc_subdevices(dev
, (devpriv
->high_speed
) ? 5 : 4);
1699 /* Analog Input subdevice */
1700 s
= &dev
->subdevices
[0];
1701 dev
->read_subdev
= s
;
1702 s
->type
= COMEDI_SUBD_AI
;
1703 s
->subdev_flags
= SDF_READABLE
| SDF_GROUND
| SDF_CMD_READ
;
1705 s
->maxdata
= 0x0fff;
1706 s
->len_chanlist
= 8;
1707 s
->range_table
= &range_usbdux_ai_range
;
1708 s
->insn_read
= usbdux_ai_insn_read
;
1709 s
->do_cmdtest
= usbdux_ai_cmdtest
;
1710 s
->do_cmd
= usbdux_ai_cmd
;
1711 s
->cancel
= usbdux_ai_cancel
;
1713 /* Analog Output subdevice */
1714 s
= &dev
->subdevices
[1];
1715 dev
->write_subdev
= s
;
1716 s
->type
= COMEDI_SUBD_AO
;
1717 s
->subdev_flags
= SDF_WRITABLE
| SDF_GROUND
| SDF_CMD_WRITE
;
1719 s
->maxdata
= 0x0fff;
1720 s
->len_chanlist
= s
->n_chan
;
1721 s
->range_table
= &range_usbdux_ao_range
;
1722 s
->do_cmdtest
= usbdux_ao_cmdtest
;
1723 s
->do_cmd
= usbdux_ao_cmd
;
1724 s
->cancel
= usbdux_ao_cancel
;
1725 s
->insn_read
= usbdux_ao_insn_read
;
1726 s
->insn_write
= usbdux_ao_insn_write
;
1728 ret
= comedi_alloc_subdev_readback(s
);
1732 /* Digital I/O subdevice */
1733 s
= &dev
->subdevices
[2];
1734 s
->type
= COMEDI_SUBD_DIO
;
1735 s
->subdev_flags
= SDF_READABLE
| SDF_WRITABLE
;
1738 s
->range_table
= &range_digital
;
1739 s
->insn_bits
= usbdux_dio_insn_bits
;
1740 s
->insn_config
= usbdux_dio_insn_config
;
1742 /* Counter subdevice */
1743 s
= &dev
->subdevices
[3];
1744 s
->type
= COMEDI_SUBD_COUNTER
;
1745 s
->subdev_flags
= SDF_WRITABLE
| SDF_READABLE
;
1747 s
->maxdata
= 0xffff;
1748 s
->insn_read
= usbdux_counter_read
;
1749 s
->insn_write
= usbdux_counter_write
;
1750 s
->insn_config
= usbdux_counter_config
;
1752 if (devpriv
->high_speed
) {
1754 s
= &dev
->subdevices
[4];
1755 s
->type
= COMEDI_SUBD_PWM
;
1756 s
->subdev_flags
= SDF_WRITABLE
| SDF_PWM_HBRIDGE
;
1758 s
->maxdata
= devpriv
->pwm_buf_sz
;
1759 s
->insn_write
= usbdux_pwm_write
;
1760 s
->insn_config
= usbdux_pwm_config
;
1762 usbdux_pwm_period(dev
, s
, PWM_DEFAULT_PERIOD
);
1768 static void usbdux_detach(struct comedi_device
*dev
)
1770 struct usb_interface
*intf
= comedi_to_usb_interface(dev
);
1771 struct usbdux_private
*devpriv
= dev
->private;
1773 usb_set_intfdata(intf
, NULL
);
1778 down(&devpriv
->sem
);
1780 /* force unlink all urbs */
1781 usbdux_pwm_stop(dev
, 1);
1782 usbdux_ao_stop(dev
, 1);
1783 usbdux_ai_stop(dev
, 1);
1785 usbdux_free_usb_buffers(dev
);
1790 static struct comedi_driver usbdux_driver
= {
1791 .driver_name
= "usbdux",
1792 .module
= THIS_MODULE
,
1793 .auto_attach
= usbdux_auto_attach
,
1794 .detach
= usbdux_detach
,
1797 static int usbdux_usb_probe(struct usb_interface
*intf
,
1798 const struct usb_device_id
*id
)
1800 return comedi_usb_auto_config(intf
, &usbdux_driver
, 0);
1803 static const struct usb_device_id usbdux_usb_table
[] = {
1804 { USB_DEVICE(0x13d8, 0x0001) },
1805 { USB_DEVICE(0x13d8, 0x0002) },
1808 MODULE_DEVICE_TABLE(usb
, usbdux_usb_table
);
1810 static struct usb_driver usbdux_usb_driver
= {
1812 .probe
= usbdux_usb_probe
,
1813 .disconnect
= comedi_usb_auto_unconfig
,
1814 .id_table
= usbdux_usb_table
,
1816 module_comedi_usb_driver(usbdux_driver
, usbdux_usb_driver
);
1818 MODULE_AUTHOR("Bernd Porr, BerndPorr@f2s.com");
1819 MODULE_DESCRIPTION("Stirling/ITL USB-DUX -- Bernd.Porr@f2s.com");
1820 MODULE_LICENSE("GPL");
1821 MODULE_FIRMWARE(USBDUX_FIRMWARE
);