range.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. /*
  2. module/range.c
  3. comedi routines for voltage ranges
  4. COMEDI - Linux Control and Measurement Device Interface
  5. Copyright (C) 1997-8 David A. Schleef <ds@schleef.org>
  6. This program is free software; you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation; either version 2 of the License, or
  9. (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.
  14. */
  15. #include <linux/uaccess.h>
  16. #include "comedidev.h"
  17. #include "comedi_internal.h"
  18. const struct comedi_lrange range_bipolar10 = { 1, {BIP_RANGE(10)} };
  19. EXPORT_SYMBOL_GPL(range_bipolar10);
  20. const struct comedi_lrange range_bipolar5 = { 1, {BIP_RANGE(5)} };
  21. EXPORT_SYMBOL_GPL(range_bipolar5);
  22. const struct comedi_lrange range_bipolar2_5 = { 1, {BIP_RANGE(2.5)} };
  23. EXPORT_SYMBOL_GPL(range_bipolar2_5);
  24. const struct comedi_lrange range_unipolar10 = { 1, {UNI_RANGE(10)} };
  25. EXPORT_SYMBOL_GPL(range_unipolar10);
  26. const struct comedi_lrange range_unipolar5 = { 1, {UNI_RANGE(5)} };
  27. EXPORT_SYMBOL_GPL(range_unipolar5);
  28. const struct comedi_lrange range_unipolar2_5 = { 1, {UNI_RANGE(2.5)} };
  29. EXPORT_SYMBOL_GPL(range_unipolar2_5);
  30. const struct comedi_lrange range_0_20mA = { 1, {RANGE_mA(0, 20)} };
  31. EXPORT_SYMBOL_GPL(range_0_20mA);
  32. const struct comedi_lrange range_4_20mA = { 1, {RANGE_mA(4, 20)} };
  33. EXPORT_SYMBOL_GPL(range_4_20mA);
  34. const struct comedi_lrange range_0_32mA = { 1, {RANGE_mA(0, 32)} };
  35. EXPORT_SYMBOL_GPL(range_0_32mA);
  36. const struct comedi_lrange range_unknown = { 1, {{0, 1000000, UNIT_none} } };
  37. EXPORT_SYMBOL_GPL(range_unknown);
  38. /*
  39. COMEDI_RANGEINFO
  40. range information ioctl
  41. arg:
  42. pointer to rangeinfo structure
  43. reads:
  44. range info structure
  45. writes:
  46. n struct comedi_krange structures to rangeinfo->range_ptr
  47. */
  48. int do_rangeinfo_ioctl(struct comedi_device *dev,
  49. struct comedi_rangeinfo __user *arg)
  50. {
  51. struct comedi_rangeinfo it;
  52. int subd, chan;
  53. const struct comedi_lrange *lr;
  54. struct comedi_subdevice *s;
  55. if (copy_from_user(&it, arg, sizeof(struct comedi_rangeinfo)))
  56. return -EFAULT;
  57. subd = (it.range_type >> 24) & 0xf;
  58. chan = (it.range_type >> 16) & 0xff;
  59. if (!dev->attached)
  60. return -EINVAL;
  61. if (subd >= dev->n_subdevices)
  62. return -EINVAL;
  63. s = &dev->subdevices[subd];
  64. if (s->range_table) {
  65. lr = s->range_table;
  66. } else if (s->range_table_list) {
  67. if (chan >= s->n_chan)
  68. return -EINVAL;
  69. lr = s->range_table_list[chan];
  70. } else {
  71. return -EINVAL;
  72. }
  73. if (RANGE_LENGTH(it.range_type) != lr->length) {
  74. dev_dbg(dev->class_dev,
  75. "wrong length %d should be %d (0x%08x)\n",
  76. RANGE_LENGTH(it.range_type),
  77. lr->length, it.range_type);
  78. return -EINVAL;
  79. }
  80. if (copy_to_user(it.range_ptr, lr->range,
  81. sizeof(struct comedi_krange) * lr->length))
  82. return -EFAULT;
  83. return 0;
  84. }
  85. static int aref_invalid(struct comedi_subdevice *s, unsigned int chanspec)
  86. {
  87. unsigned int aref;
  88. /* disable reporting invalid arefs... maybe someday */
  89. return 0;
  90. aref = CR_AREF(chanspec);
  91. switch (aref) {
  92. case AREF_DIFF:
  93. if (s->subdev_flags & SDF_DIFF)
  94. return 0;
  95. break;
  96. case AREF_COMMON:
  97. if (s->subdev_flags & SDF_COMMON)
  98. return 0;
  99. break;
  100. case AREF_GROUND:
  101. if (s->subdev_flags & SDF_GROUND)
  102. return 0;
  103. break;
  104. case AREF_OTHER:
  105. if (s->subdev_flags & SDF_OTHER)
  106. return 0;
  107. break;
  108. default:
  109. break;
  110. }
  111. dev_dbg(s->device->class_dev, "subdevice does not support aref %i",
  112. aref);
  113. return 1;
  114. }
  115. /**
  116. * comedi_check_chanlist() - Validate each element in a chanlist.
  117. * @s: comedi_subdevice struct
  118. * @n: number of elements in the chanlist
  119. * @chanlist: the chanlist to validate
  120. */
  121. int comedi_check_chanlist(struct comedi_subdevice *s, int n,
  122. unsigned int *chanlist)
  123. {
  124. struct comedi_device *dev = s->device;
  125. unsigned int chanspec;
  126. int chan, range_len, i;
  127. for (i = 0; i < n; i++) {
  128. chanspec = chanlist[i];
  129. chan = CR_CHAN(chanspec);
  130. if (s->range_table)
  131. range_len = s->range_table->length;
  132. else if (s->range_table_list && chan < s->n_chan)
  133. range_len = s->range_table_list[chan]->length;
  134. else
  135. range_len = 0;
  136. if (chan >= s->n_chan ||
  137. CR_RANGE(chanspec) >= range_len ||
  138. aref_invalid(s, chanspec)) {
  139. dev_warn(dev->class_dev,
  140. "bad chanlist[%d]=0x%08x chan=%d range length=%d\n",
  141. i, chanspec, chan, range_len);
  142. return -EINVAL;
  143. }
  144. }
  145. return 0;
  146. }
  147. EXPORT_SYMBOL_GPL(comedi_check_chanlist);