提交 853fc892 编写于 作者: H H Hartley Sweeten 提交者: Greg Kroah-Hartman

staging: comedi: rti800: fix rti800_ao_insn_write()

This board has two independent analog output channels. Each channel
has its own 12-bit D.A converter. Each channel can be set to output
a unipolar 10V or bipolar 10V signal. This selection is handled by
a jumper on the board and configured by the user with options[5] and
[7] passed during the attach of the board.

The two channels can also be configured with jumpers to use straight
binary or two's complement data. The user configures the data type
with options[6] and [8] when attaching.

Currently, this driver uses the dac0 selection, option[6], for both
channels. Fix the rti800_ao_insn_write() function to properly use
the configuration specified by the user.

Change the dac[01]_coding in the private data to a simple bool
array, dac_2comp.

Add some local vars to hold the registers offsets needed to write
to the DAC.

Only the last value written to the DAC needs to be cached for readback,
Signed-off-by: NH Hartley Sweeten <hsweeten@visionengravers.com>
Cc: Ian Abbott <abbotti@mev.co.uk>
Signed-off-by: NGreg Kroah-Hartman <gregkh@linuxfoundation.org>
上级 a4b279ba
...@@ -153,9 +153,7 @@ struct rti800_private { ...@@ -153,9 +153,7 @@ struct rti800_private {
enum { enum {
adc_2comp, adc_straight adc_2comp, adc_straight
} adc_coding; } adc_coding;
enum { bool dac_2comp[2];
dac_2comp, dac_straight
} dac0_coding, dac1_coding;
const struct comedi_lrange *ao_range_type_list[2]; const struct comedi_lrange *ao_range_type_list[2];
unsigned int ao_readback[2]; unsigned int ao_readback[2];
int muxgain_bits; int muxgain_bits;
...@@ -235,24 +233,28 @@ static int rti800_ao_insn_read(struct comedi_device *dev, ...@@ -235,24 +233,28 @@ static int rti800_ao_insn_read(struct comedi_device *dev,
static int rti800_ao_insn_write(struct comedi_device *dev, static int rti800_ao_insn_write(struct comedi_device *dev,
struct comedi_subdevice *s, struct comedi_subdevice *s,
struct comedi_insn *insn, unsigned int *data) struct comedi_insn *insn,
unsigned int *data)
{ {
struct rti800_private *devpriv = dev->private; struct rti800_private *devpriv = dev->private;
int chan = CR_CHAN(insn->chanspec); int chan = CR_CHAN(insn->chanspec);
int d; int reg_lo = chan ? RTI800_DAC1LO : RTI800_DAC0LO;
int reg_hi = chan ? RTI800_DAC1HI : RTI800_DAC0HI;
int val = devpriv->ao_readback[chan];
int i; int i;
for (i = 0; i < insn->n; i++) { for (i = 0; i < insn->n; i++) {
devpriv->ao_readback[chan] = d = data[i]; val = data[i];
if (devpriv->dac0_coding == dac_2comp) if (devpriv->dac_2comp[chan])
d ^= 0x800; val ^= 0x800;
outb(d & 0xff, outb(val & 0xff, dev->iobase + reg_lo);
dev->iobase + (chan ? RTI800_DAC1LO : RTI800_DAC0LO)); outb((val >> 8) & 0xff, dev->iobase + reg_hi);
outb(d >> 8,
dev->iobase + (chan ? RTI800_DAC1HI : RTI800_DAC0HI));
} }
return i;
devpriv->ao_readback[chan] = val;
return insn->n;
} }
static int rti800_di_insn_bits(struct comedi_device *dev, static int rti800_di_insn_bits(struct comedi_device *dev,
...@@ -325,8 +327,8 @@ static int rti800_attach(struct comedi_device *dev, struct comedi_devconfig *it) ...@@ -325,8 +327,8 @@ static int rti800_attach(struct comedi_device *dev, struct comedi_devconfig *it)
dev->private = devpriv; dev->private = devpriv;
devpriv->adc_coding = it->options[4]; devpriv->adc_coding = it->options[4];
devpriv->dac0_coding = it->options[6]; devpriv->dac_2comp[0] = (it->options[6] == 0);
devpriv->dac1_coding = it->options[8]; devpriv->dac_2comp[1] = (it->options[8] == 0);
devpriv->muxgain_bits = -1; devpriv->muxgain_bits = -1;
ret = comedi_alloc_subdevices(dev, 4); ret = comedi_alloc_subdevices(dev, 4);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册