xref: /openbmc/linux/drivers/comedi/range.c (revision df0e68c1)
18ffdff6aSGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0+
28ffdff6aSGreg Kroah-Hartman /*
38ffdff6aSGreg Kroah-Hartman  * comedi/range.c
48ffdff6aSGreg Kroah-Hartman  * comedi routines for voltage ranges
58ffdff6aSGreg Kroah-Hartman  *
68ffdff6aSGreg Kroah-Hartman  * COMEDI - Linux Control and Measurement Device Interface
78ffdff6aSGreg Kroah-Hartman  * Copyright (C) 1997-8 David A. Schleef <ds@schleef.org>
88ffdff6aSGreg Kroah-Hartman  */
98ffdff6aSGreg Kroah-Hartman 
108ffdff6aSGreg Kroah-Hartman #include <linux/uaccess.h>
11*df0e68c1SIan Abbott #include <linux/comedi/comedidev.h>
128ffdff6aSGreg Kroah-Hartman #include "comedi_internal.h"
138ffdff6aSGreg Kroah-Hartman 
148ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_bipolar10 = { 1, {BIP_RANGE(10)} };
158ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_bipolar10);
168ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_bipolar5 = { 1, {BIP_RANGE(5)} };
178ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_bipolar5);
188ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_bipolar2_5 = { 1, {BIP_RANGE(2.5)} };
198ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_bipolar2_5);
208ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_unipolar10 = { 1, {UNI_RANGE(10)} };
218ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_unipolar10);
228ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_unipolar5 = { 1, {UNI_RANGE(5)} };
238ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_unipolar5);
248ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_unipolar2_5 = { 1, {UNI_RANGE(2.5)} };
258ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_unipolar2_5);
268ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_0_20mA = { 1, {RANGE_mA(0, 20)} };
278ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_0_20mA);
288ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_4_20mA = { 1, {RANGE_mA(4, 20)} };
298ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_4_20mA);
308ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_0_32mA = { 1, {RANGE_mA(0, 32)} };
318ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_0_32mA);
328ffdff6aSGreg Kroah-Hartman const struct comedi_lrange range_unknown = { 1, {{0, 1000000, UNIT_none} } };
338ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(range_unknown);
348ffdff6aSGreg Kroah-Hartman 
358ffdff6aSGreg Kroah-Hartman /*
368ffdff6aSGreg Kroah-Hartman  * COMEDI_RANGEINFO ioctl
378ffdff6aSGreg Kroah-Hartman  * range information
388ffdff6aSGreg Kroah-Hartman  *
398ffdff6aSGreg Kroah-Hartman  * arg:
408ffdff6aSGreg Kroah-Hartman  *	pointer to comedi_rangeinfo structure
418ffdff6aSGreg Kroah-Hartman  *
428ffdff6aSGreg Kroah-Hartman  * reads:
438ffdff6aSGreg Kroah-Hartman  *	comedi_rangeinfo structure
448ffdff6aSGreg Kroah-Hartman  *
458ffdff6aSGreg Kroah-Hartman  * writes:
468ffdff6aSGreg Kroah-Hartman  *	array of comedi_krange structures to rangeinfo->range_ptr pointer
478ffdff6aSGreg Kroah-Hartman  */
do_rangeinfo_ioctl(struct comedi_device * dev,struct comedi_rangeinfo * it)488ffdff6aSGreg Kroah-Hartman int do_rangeinfo_ioctl(struct comedi_device *dev,
498ffdff6aSGreg Kroah-Hartman 		       struct comedi_rangeinfo *it)
508ffdff6aSGreg Kroah-Hartman {
518ffdff6aSGreg Kroah-Hartman 	int subd, chan;
528ffdff6aSGreg Kroah-Hartman 	const struct comedi_lrange *lr;
538ffdff6aSGreg Kroah-Hartman 	struct comedi_subdevice *s;
548ffdff6aSGreg Kroah-Hartman 
558ffdff6aSGreg Kroah-Hartman 	subd = (it->range_type >> 24) & 0xf;
568ffdff6aSGreg Kroah-Hartman 	chan = (it->range_type >> 16) & 0xff;
578ffdff6aSGreg Kroah-Hartman 
588ffdff6aSGreg Kroah-Hartman 	if (!dev->attached)
598ffdff6aSGreg Kroah-Hartman 		return -EINVAL;
608ffdff6aSGreg Kroah-Hartman 	if (subd >= dev->n_subdevices)
618ffdff6aSGreg Kroah-Hartman 		return -EINVAL;
628ffdff6aSGreg Kroah-Hartman 	s = &dev->subdevices[subd];
638ffdff6aSGreg Kroah-Hartman 	if (s->range_table) {
648ffdff6aSGreg Kroah-Hartman 		lr = s->range_table;
658ffdff6aSGreg Kroah-Hartman 	} else if (s->range_table_list) {
668ffdff6aSGreg Kroah-Hartman 		if (chan >= s->n_chan)
678ffdff6aSGreg Kroah-Hartman 			return -EINVAL;
688ffdff6aSGreg Kroah-Hartman 		lr = s->range_table_list[chan];
698ffdff6aSGreg Kroah-Hartman 	} else {
708ffdff6aSGreg Kroah-Hartman 		return -EINVAL;
718ffdff6aSGreg Kroah-Hartman 	}
728ffdff6aSGreg Kroah-Hartman 
738ffdff6aSGreg Kroah-Hartman 	if (RANGE_LENGTH(it->range_type) != lr->length) {
748ffdff6aSGreg Kroah-Hartman 		dev_dbg(dev->class_dev,
758ffdff6aSGreg Kroah-Hartman 			"wrong length %d should be %d (0x%08x)\n",
768ffdff6aSGreg Kroah-Hartman 			RANGE_LENGTH(it->range_type),
778ffdff6aSGreg Kroah-Hartman 			lr->length, it->range_type);
788ffdff6aSGreg Kroah-Hartman 		return -EINVAL;
798ffdff6aSGreg Kroah-Hartman 	}
808ffdff6aSGreg Kroah-Hartman 
818ffdff6aSGreg Kroah-Hartman 	if (copy_to_user(it->range_ptr, lr->range,
828ffdff6aSGreg Kroah-Hartman 			 sizeof(struct comedi_krange) * lr->length))
838ffdff6aSGreg Kroah-Hartman 		return -EFAULT;
848ffdff6aSGreg Kroah-Hartman 
858ffdff6aSGreg Kroah-Hartman 	return 0;
868ffdff6aSGreg Kroah-Hartman }
878ffdff6aSGreg Kroah-Hartman 
888ffdff6aSGreg Kroah-Hartman /**
898ffdff6aSGreg Kroah-Hartman  * comedi_check_chanlist() - Validate each element in a chanlist.
908ffdff6aSGreg Kroah-Hartman  * @s: comedi_subdevice struct
918ffdff6aSGreg Kroah-Hartman  * @n: number of elements in the chanlist
928ffdff6aSGreg Kroah-Hartman  * @chanlist: the chanlist to validate
938ffdff6aSGreg Kroah-Hartman  *
948ffdff6aSGreg Kroah-Hartman  * Each element consists of a channel number, a range index, an analog
958ffdff6aSGreg Kroah-Hartman  * reference type and some flags, all packed into an unsigned int.
968ffdff6aSGreg Kroah-Hartman  *
978ffdff6aSGreg Kroah-Hartman  * This checks that the channel number and range index are supported by
988ffdff6aSGreg Kroah-Hartman  * the comedi subdevice.  It does not check whether the analog reference
998ffdff6aSGreg Kroah-Hartman  * type and the flags are supported.  Drivers that care should check those
1008ffdff6aSGreg Kroah-Hartman  * themselves.
1018ffdff6aSGreg Kroah-Hartman  *
1028ffdff6aSGreg Kroah-Hartman  * Return: %0 if all @chanlist elements are valid (success),
1038ffdff6aSGreg Kroah-Hartman  *         %-EINVAL if one or more elements are invalid.
1048ffdff6aSGreg Kroah-Hartman  */
comedi_check_chanlist(struct comedi_subdevice * s,int n,unsigned int * chanlist)1058ffdff6aSGreg Kroah-Hartman int comedi_check_chanlist(struct comedi_subdevice *s, int n,
1068ffdff6aSGreg Kroah-Hartman 			  unsigned int *chanlist)
1078ffdff6aSGreg Kroah-Hartman {
1088ffdff6aSGreg Kroah-Hartman 	struct comedi_device *dev = s->device;
1098ffdff6aSGreg Kroah-Hartman 	unsigned int chanspec;
1108ffdff6aSGreg Kroah-Hartman 	int chan, range_len, i;
1118ffdff6aSGreg Kroah-Hartman 
1128ffdff6aSGreg Kroah-Hartman 	for (i = 0; i < n; i++) {
1138ffdff6aSGreg Kroah-Hartman 		chanspec = chanlist[i];
1148ffdff6aSGreg Kroah-Hartman 		chan = CR_CHAN(chanspec);
1158ffdff6aSGreg Kroah-Hartman 		if (s->range_table)
1168ffdff6aSGreg Kroah-Hartman 			range_len = s->range_table->length;
1178ffdff6aSGreg Kroah-Hartman 		else if (s->range_table_list && chan < s->n_chan)
1188ffdff6aSGreg Kroah-Hartman 			range_len = s->range_table_list[chan]->length;
1198ffdff6aSGreg Kroah-Hartman 		else
1208ffdff6aSGreg Kroah-Hartman 			range_len = 0;
1218ffdff6aSGreg Kroah-Hartman 		if (chan >= s->n_chan ||
1228ffdff6aSGreg Kroah-Hartman 		    CR_RANGE(chanspec) >= range_len) {
1238ffdff6aSGreg Kroah-Hartman 			dev_warn(dev->class_dev,
1248ffdff6aSGreg Kroah-Hartman 				 "bad chanlist[%d]=0x%08x chan=%d range length=%d\n",
1258ffdff6aSGreg Kroah-Hartman 				 i, chanspec, chan, range_len);
1268ffdff6aSGreg Kroah-Hartman 			return -EINVAL;
1278ffdff6aSGreg Kroah-Hartman 		}
1288ffdff6aSGreg Kroah-Hartman 	}
1298ffdff6aSGreg Kroah-Hartman 	return 0;
1308ffdff6aSGreg Kroah-Hartman }
1318ffdff6aSGreg Kroah-Hartman EXPORT_SYMBOL_GPL(comedi_check_chanlist);
132