3 comedi routines for voltage ranges
5 COMEDI - Linux Control and Measurement Device Interface
6 Copyright (C) 1997-8 David A. Schleef <ds@schleef.org>
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
24 #include "comedidev.h"
25 #include <asm/uaccess.h>
27 const struct comedi_lrange range_bipolar10
= { 1, {BIP_RANGE(10)} };
28 const struct comedi_lrange range_bipolar5
= { 1, {BIP_RANGE(5)} };
29 const struct comedi_lrange range_bipolar2_5
= { 1, {BIP_RANGE(2.5)} };
30 const struct comedi_lrange range_unipolar10
= { 1, {UNI_RANGE(10)} };
31 const struct comedi_lrange range_unipolar5
= { 1, {UNI_RANGE(5)} };
32 const struct comedi_lrange range_unknown
= { 1, {{0, 1000000, UNIT_none
}} };
36 range information ioctl
39 pointer to rangeinfo structure
45 n struct comedi_krange structures to rangeinfo->range_ptr
47 int do_rangeinfo_ioctl(struct comedi_device
*dev
, struct comedi_rangeinfo
*arg
)
49 struct comedi_rangeinfo it
;
51 const struct comedi_lrange
*lr
;
52 struct comedi_subdevice
*s
;
54 if (copy_from_user(&it
, arg
, sizeof(struct comedi_rangeinfo
)))
56 subd
= (it
.range_type
>> 24) & 0xf;
57 chan
= (it
.range_type
>> 16) & 0xff;
61 if (subd
>= dev
->n_subdevices
)
63 s
= dev
->subdevices
+ subd
;
66 } else if (s
->range_table_list
) {
67 if (chan
>= s
->n_chan
)
69 lr
= s
->range_table_list
[chan
];
74 if (RANGE_LENGTH(it
.range_type
) != lr
->length
) {
75 DPRINTK("wrong length %d should be %d (0x%08x)\n",
76 RANGE_LENGTH(it
.range_type
), lr
->length
, it
.range_type
);
80 if (copy_to_user(it
.range_ptr
, lr
->range
,
81 sizeof(struct comedi_krange
) * lr
->length
))
87 static int aref_invalid(struct comedi_subdevice
*s
, unsigned int chanspec
)
91 /* disable reporting invalid arefs... maybe someday */
94 aref
= CR_AREF(chanspec
);
97 if (s
->subdev_flags
& SDF_DIFF
)
101 if (s
->subdev_flags
& SDF_COMMON
)
105 if (s
->subdev_flags
& SDF_GROUND
)
109 if (s
->subdev_flags
& SDF_OTHER
)
115 DPRINTK("subdevice does not support aref %i", aref
);
120 This function checks each element in a channel/gain list to make
121 make sure it is valid.
123 int check_chanlist(struct comedi_subdevice
*s
, int n
, unsigned int *chanlist
)
128 if (s
->range_table
) {
129 for (i
= 0; i
< n
; i
++)
130 if (CR_CHAN(chanlist
[i
]) >= s
->n_chan
||
131 CR_RANGE(chanlist
[i
]) >= s
->range_table
->length
132 || aref_invalid(s
, chanlist
[i
])) {
133 printk("bad chanlist[%d]=0x%08x n_chan=%d range length=%d\n",
134 i
, chanlist
[i
], s
->n_chan
,
135 s
->range_table
->length
);
137 for (i
= 0; i
< n
; i
++)
138 printk("[%d]=0x%08x\n", i
, chanlist
[i
]);
142 } else if (s
->range_table_list
) {
143 for (i
= 0; i
< n
; i
++) {
144 chan
= CR_CHAN(chanlist
[i
]);
145 if (chan
>= s
->n_chan
||
146 CR_RANGE(chanlist
[i
]) >=
147 s
->range_table_list
[chan
]->length
148 || aref_invalid(s
, chanlist
[i
])) {
149 printk("bad chanlist[%d]=0x%08x\n", i
,
155 printk("comedi: (bug) no range type list!\n");