staging: comedi: adl_pci9118: eliminate DMA buffer defragmentation step
The DMA operations used by the driver may have been set up to acquire data from unwanted channels in addition to the wanted channels. Currently, `interrupt_pci9118_ai_dma()` calls `defragment_dma_buffer()` to move all the wanted data to the start of the DMA buffer and then calls `comedi_buf_write_samples()` to copy it all to the comedi async buffer. Those two functions used to be called from `move_block_from_dma()` which was absorbed into `interrupt_pci9118_ai_dma()`. Reinstate `move_block_from_dma()` but rewrite it to copy data directly from the wanted fragments of the DMA buffer to the comedi async buffer without defragmenting the buffer first. Signed-off-by: Ian Abbott <abbotti@mev.co.uk> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
bb4a51a1b5
commit
e87f65b238
|
@ -502,29 +502,56 @@ static unsigned int valid_samples_in_act_dma_buf(struct comedi_device *dev,
|
|||
return n_samples;
|
||||
}
|
||||
|
||||
static unsigned int defragment_dma_buffer(struct comedi_device *dev,
|
||||
struct comedi_subdevice *s,
|
||||
unsigned short *dma_buffer,
|
||||
unsigned int num_samples)
|
||||
static void move_block_from_dma(struct comedi_device *dev,
|
||||
struct comedi_subdevice *s,
|
||||
unsigned short *dma_buffer,
|
||||
unsigned int n_raw_samples)
|
||||
{
|
||||
struct pci9118_private *devpriv = dev->private;
|
||||
struct comedi_cmd *cmd = &s->async->cmd;
|
||||
unsigned int i = 0, j = 0;
|
||||
unsigned int start_pos = devpriv->ai_add_front,
|
||||
stop_pos = devpriv->ai_add_front + cmd->chanlist_len;
|
||||
unsigned int raw_scanlen = devpriv->ai_add_front + cmd->chanlist_len +
|
||||
devpriv->ai_add_back;
|
||||
unsigned int start_pos = devpriv->ai_add_front;
|
||||
unsigned int stop_pos = start_pos + cmd->chanlist_len;
|
||||
unsigned int span_len = stop_pos + devpriv->ai_add_back;
|
||||
unsigned int dma_pos = devpriv->ai_act_dmapos;
|
||||
unsigned int x;
|
||||
|
||||
for (i = 0; i < num_samples; i++) {
|
||||
if (devpriv->ai_act_dmapos >= start_pos &&
|
||||
devpriv->ai_act_dmapos < stop_pos) {
|
||||
dma_buffer[j++] = dma_buffer[i];
|
||||
if (span_len == cmd->chanlist_len) {
|
||||
/* All samples are to be copied. */
|
||||
comedi_buf_write_samples(s, dma_buffer, n_raw_samples);
|
||||
dma_pos += n_raw_samples;
|
||||
} else {
|
||||
/*
|
||||
* Not all samples are to be copied. Buffer contents consist
|
||||
* of a possibly non-whole number of spans and a region of
|
||||
* each span is to be copied.
|
||||
*/
|
||||
while (n_raw_samples) {
|
||||
if (dma_pos < start_pos) {
|
||||
/* Skip samples before start position. */
|
||||
x = start_pos - dma_pos;
|
||||
if (x > n_raw_samples)
|
||||
x = n_raw_samples;
|
||||
dma_pos += x;
|
||||
n_raw_samples -= x;
|
||||
if (!n_raw_samples)
|
||||
break;
|
||||
}
|
||||
if (dma_pos < stop_pos) {
|
||||
/* Copy samples before stop position. */
|
||||
x = stop_pos - dma_pos;
|
||||
if (x > n_raw_samples)
|
||||
x = n_raw_samples;
|
||||
comedi_buf_write_samples(s, dma_buffer, x);
|
||||
dma_pos += x;
|
||||
n_raw_samples -= x;
|
||||
}
|
||||
/* Advance to next span. */
|
||||
start_pos += span_len;
|
||||
stop_pos += span_len;
|
||||
}
|
||||
devpriv->ai_act_dmapos++;
|
||||
devpriv->ai_act_dmapos %= raw_scanlen;
|
||||
}
|
||||
|
||||
return j;
|
||||
/* Update position in span for next time. */
|
||||
devpriv->ai_act_dmapos = dma_pos % span_len;
|
||||
}
|
||||
|
||||
static void pci9118_exttrg_enable(struct comedi_device *dev, bool enable)
|
||||
|
@ -681,10 +708,8 @@ static void interrupt_pci9118_ai_dma(struct comedi_device *dev,
|
|||
}
|
||||
}
|
||||
|
||||
if (n_all) {
|
||||
n_valid = defragment_dma_buffer(dev, s, dmabuf->virt, n_all);
|
||||
comedi_buf_write_samples(s, dmabuf->virt, n_valid);
|
||||
}
|
||||
if (n_all)
|
||||
move_block_from_dma(dev, s, dmabuf->virt, n_all);
|
||||
|
||||
if (!devpriv->ai_neverending) {
|
||||
if (s->async->scans_done >= cmd->stop_arg)
|
||||
|
|
Loading…
Reference in New Issue