// SPDX-License-Identifier: GPL-2.0+ /* * comedi/drivers/amplc_pci224.c * Driver for Amplicon PCI224 and PCI234 AO boards. * * Copyright (C) 2005 MEV Ltd. * * COMEDI - Linux Control and Measurement Device Interface * Copyright (C) 1998,2000 David A. Schleef */ /* * Driver: amplc_pci224 * Description: Amplicon PCI224, PCI234 * Author: Ian Abbott * Devices: [Amplicon] PCI224 (amplc_pci224), PCI234 * Updated: Thu, 31 Jul 2014 11:08:03 +0000 * Status: works, but see caveats * * Supports: * * - ao_insn read/write * - ao_do_cmd mode with the following sources: * * - start_src TRIG_INT TRIG_EXT * - scan_begin_src TRIG_TIMER TRIG_EXT * - convert_src TRIG_NOW * - scan_end_src TRIG_COUNT * - stop_src TRIG_COUNT TRIG_EXT TRIG_NONE * * The channel list must contain at least one channel with no repeated * channels. The scan end count must equal the number of channels in * the channel list. * * There is only one external trigger source so only one of start_src, * scan_begin_src or stop_src may use TRIG_EXT. * * Configuration options: * none * * Manual configuration of PCI cards is not supported; they are configured * automatically. * * Output range selection - PCI224: * * Output ranges on PCI224 are partly software-selectable and partly * hardware-selectable according to jumper LK1. All channels are set * to the same range: * * - LK1 position 1-2 (factory default) corresponds to the following * comedi ranges: * * 0: [-10V,+10V]; 1: [-5V,+5V]; 2: [-2.5V,+2.5V], 3: [-1.25V,+1.25V], * 4: [0,+10V], 5: [0,+5V], 6: [0,+2.5V], 7: [0,+1.25V] * * - LK1 position 2-3 corresponds to the following Comedi ranges, using * an external voltage reference: * * 0: [-Vext,+Vext], * 1: [0,+Vext] * * Output range selection - PCI234: * * Output ranges on PCI234 are hardware-selectable according to jumper * LK1 which affects all channels, and jumpers LK2, LK3, LK4 and LK5 * which affect channels 0, 1, 2 and 3 individually. LK1 chooses between * an internal 5V reference and an external voltage reference (Vext). * LK2/3/4/5 choose (per channel) to double the reference or not according * to the following table: * * LK1 position LK2/3/4/5 pos Comedi range * ------------- ------------- -------------- * 2-3 (factory) 1-2 (factory) 0: [-10V,+10V] * 2-3 (factory) 2-3 1: [-5V,+5V] * 1-2 1-2 (factory) 2: [-2*Vext,+2*Vext] * 1-2 2-3 3: [-Vext,+Vext] * * Caveats: * * 1) All channels on the PCI224 share the same range. Any change to the * range as a result of insn_write or a streaming command will affect * the output voltages of all channels, including those not specified * by the instruction or command. * * 2) For the analog output command, the first scan may be triggered * falsely at the start of acquisition. This occurs when the DAC scan * trigger source is switched from 'none' to 'timer' (scan_begin_src = * TRIG_TIMER) or 'external' (scan_begin_src == TRIG_EXT) at the start * of acquisition and the trigger source is at logic level 1 at the * time of the switch. This is very likely for TRIG_TIMER. For * TRIG_EXT, it depends on the state of the external line and whether * the CR_INVERT flag has been set. The remaining scans are triggered * correctly. */ #include #include #include #include "../comedi_pci.h" #include "comedi_8254.h" /* * PCI224/234 i/o space 1 (PCIBAR2) registers. */ #define PCI224_Z2_BASE 0x14 /* 82C54 counter/timer */ #define PCI224_ZCLK_SCE 0x1A /* Group Z Clock Configuration Register */ #define PCI224_ZGAT_SCE 0x1D /* Group Z Gate Configuration Register */ #define PCI224_INT_SCE 0x1E /* ISR Interrupt source mask register */ /* /Interrupt status */ /* * PCI224/234 i/o space 2 (PCIBAR3) 16-bit registers. */ #define PCI224_DACDATA 0x00 /* (w-o) DAC FIFO data. */ #define PCI224_SOFTTRIG 0x00 /* (r-o) DAC software scan trigger. */ #define PCI224_DACCON 0x02 /* (r/w) DAC status/configuration. */ #define PCI224_FIFOSIZ 0x04 /* (w-o) FIFO size for wraparound mode. */ #define PCI224_DACCEN 0x06 /* (w-o) DAC channel enable register. */ /* * DACCON values. */ /* (r/w) Scan trigger. */ #define PCI224_DACCON_TRIG(x) (((x) & 0x7) << 0) #define PCI224_DACCON_TRIG_MASK PCI224_DACCON_TRIG(7) #define PCI224_DACCON_TRIG_NONE PCI224_DACCON_TRIG(0) /* none */ #define PCI224_DACCON_TRIG_SW PCI224_DACCON_TRIG(1) /* soft trig */ #define PCI224_DACCON_TRIG_EXTP PCI224_DACCON_TRIG(2) /* ext + edge */ #define PCI224_DACCON_TRIG_EXTN PCI224_DACCON_TRIG(3) /* ext - edge */ #define PCI224_DACCON_TRIG_Z2CT0 PCI224_DACCON_TRIG(4) /* Z2 CT0 out */ #define PCI224_DACCON_TRIG_Z2CT1 PCI224_DACCON_TRIG(5) /* Z2 CT1 out */ #define PCI224_DACCON_TRIG_Z2CT2 PCI224_DACCON_TRIG(6) /* Z2 CT2 out */ /* (r/w) Polarity (PCI224 only, PCI234 always bipolar!). */ #define PCI224_DACCON_POLAR(x) (((x) & 0x1) << 3) #define PCI224_DACCON_POLAR_MASK PCI224_DACCON_POLAR(1) #define PCI224_DACCON_POLAR_UNI PCI224_DACCON_POLAR(0) /* [0,+V] */ #define PCI224_DACCON_POLAR_BI PCI224_DACCON_POLAR(1) /* [-V,+V] */ /* (r/w) Internal Vref (PCI224 only, when LK1 in position 1-2). */ #define PCI224_DACCON_VREF(x) (((x) & 0x3) << 4) #define PCI224_DACCON_VREF_MASK PCI224_DACCON_VREF(3) #define PCI224_DACCON_VREF_1_25 PCI224_DACCON_VREF(0) /* 1.25V */ #define PCI224_DACCON_VREF_2_5 PCI224_DACCON_VREF(1) /* 2.5V */ #define PCI224_DACCON_VREF_5 PCI224_DACCON_VREF(2) /* 5V */ #define PCI224_DACCON_VREF_10 PCI224_DACCON_VREF(3) /* 10V */ /* (r/w) Wraparound mode enable (to play back stored waveform). */ #define PCI224_DACCON_FIFOWRAP BIT(7) /* (r/w) FIFO enable. It MUST be set! */ #define PCI224_DACCON_FIFOENAB BIT(8) /* (r/w) FIFO interrupt trigger level (most values are not very useful). */ #define PCI224_DACCON_FIFOINTR(x) (((x) & 0x7) << 9) #define PCI224_DACCON_FIFOINTR_MASK PCI224_DACCON_FIFOINTR(7) #define PCI224_DACCON_FIFOINTR_EMPTY PCI224_DACCON_FIFOINTR(0) /* empty */ #define PCI224_DACCON_FIFOINTR_NEMPTY PCI224_DACCON_FIFOINTR(1) /* !empty */ #define PCI224_DACCON_FIFOINTR_NHALF PCI224_DACCON_FIFOINTR(2) /* !half */ #define PCI224_DACCON_FIFOINTR_HALF PCI224_DACCON_FIFOINTR(3) /* half */ #define PCI224_DACCON_FIFOINTR_NFULL PCI224_DACCON_FIFOINTR(4) /* !full */ #define PCI224_DACCON_FIFOINTR_FULL PCI224_DACCON_FIFOINTR(5) /* full */ /* (r-o) FIFO fill level. */ #define PCI224_DACCON_FIFOFL(x) (((x) & 0x7) << 12) #define PCI224_DACCON_FIFOFL_MASK PCI224_DACCON_FIFOFL(7) #define PCI224_DACCON_FIFOFL_EMPTY PCI224_DACCON_FIFOFL(1) /* 0 */ #define PCI224_DACCON_FIFOFL_ONETOHALF PCI224_DACCON_FIFOFL(0) /* 1-2048 */ #define PCI224_DACCON_FIFOFL_HALFTOFULL PCI224_DACCON_FIFOFL(4) /* 2049-4095 */ #define PCI224_DACCON_FIFOFL_FULL PCI224_DACCON_FIFOFL(6) /* 4096 */ /* (r-o) DAC busy flag. */ #define PCI224_DACCON_BUSY BIT(15) /* (w-o) FIFO reset. */ #define PCI224_DACCON_FIFORESET BIT(12) /* (w-o) Global reset (not sure what it does). */ #define PCI224_DACCON_GLOBALRESET BIT(13) /* * DAC FIFO size. */ #define PCI224_FIFO_SIZE 4096 /* * DAC FIFO guaranteed minimum room available, depending on reported fill level. * The maximum room available depends on the reported fill level and how much * has been written! */ #define PCI224_FIFO_ROOM_EMPTY PCI224_FIFO_SIZE #define PCI224_FIFO_ROOM_ONETOHALF (PCI224_FIFO_SIZE / 2) #define PCI224_FIFO_ROOM_HALFTOFULL 1 #define PCI224_FIFO_ROOM_FULL 0 /* * Counter/timer clock input configuration sources. */ #define CLK_CLK 0 /* reserved (channel-specific clock) */ #define CLK_10MHZ 1 /* internal 10 MHz clock */ #define CLK_1MHZ 2 /* internal 1 MHz clock */ #define CLK_100KHZ 3 /* internal 100 kHz clock */ #define CLK_10KHZ 4 /* internal 10 kHz clock */ #define CLK_1KHZ 5 /* internal 1 kHz clock */ #define CLK_OUTNM1 6 /* output of channel-1 modulo total */ #define CLK_EXT 7 /* external clock */ static unsigned int pci224_clk_config(unsigned int chan, unsigned int src) { return ((chan & 3) << 3) | (src & 7); } /* * Counter/timer gate input configuration sources. */ #define GAT_VCC 0 /* VCC (i.e. enabled) */ #define GAT_GND 1 /* GND (i.e. disabled) */ #define GAT_EXT 2 /* reserved (external gate input) */ #define GAT_NOUTNM2 3 /* inverted output of channel-2 modulo total */ static unsigned int pci224_gat_config(unsigned int chan, unsigned int src) { return ((chan & 3) << 3) | (src & 7); } /* * Summary of CLK_OUTNM1 and GAT_NOUTNM2 connections for PCI224 and PCI234: * * Channel's Channel's * clock input gate input * Channel CLK_OUTNM1 GAT_NOUTNM2 * ------- ---------- ----------- * Z2-CT0 Z2-CT2-OUT /Z2-CT1-OUT * Z2-CT1 Z2-CT0-OUT /Z2-CT2-OUT * Z2-CT2 Z2-CT1-OUT /Z2-CT0-OUT */ /* * Interrupt enable/status bits */ #define PCI224_INTR_EXT 0x01 /* rising edge on external input */ #define PCI224_INTR_DAC 0x04 /* DAC (FIFO) interrupt */ #define PCI224_INTR_Z2CT1 0x20 /* rising edge on Z2-CT1 output */ #define PCI224_INTR_EDGE_BITS (PCI224_INTR_EXT | PCI224_INTR_Z2CT1) #define PCI224_INTR_LEVEL_BITS PCI224_INTR_DACFIFO /* * Handy macros. */ /* Combine old and new bits. */ #define COMBINE(old, new, mask) (((old) & ~(mask)) | ((new) & (mask))) /* Current CPU. XXX should this be hard_smp_processor_id()? */ #define THISCPU smp_processor_id() /* State bits for use with atomic bit operations. */ #define AO_CMD_STARTED 0 /* * Range tables. */ /* * The ranges for PCI224. * * These are partly hardware-selectable by jumper LK1 and partly * software-selectable. * * All channels share the same hardware range. */ static const struct comedi_lrange range_pci224 = { 10, { /* jumper LK1 in position 1-2 (factory default) */ BIP_RANGE(10), BIP_RANGE(5), BIP_RANGE(2.5), BIP_RANGE(1.25), UNI_RANGE(10), UNI_RANGE(5), UNI_RANGE(2.5), UNI_RANGE(1.25), /* jumper LK1 in position 2-3 */ RANGE_ext(-1, 1), /* bipolar [-Vext,+Vext] */ RANGE_ext(0, 1), /* unipolar [0,+Vext] */ } }; static const unsigned short hwrange_pci224[10] = { /* jumper LK1 in position 1-2 (factory default) */ PCI224_DACCON_POLAR_BI | PCI224_DACCON_VREF_10, PCI224_DACCON_POLAR_BI | PCI224_DACCON_VREF_5, PCI224_DACCON_POLAR_BI | PCI224_DACCON_VREF_2_5, PCI224_DACCON_POLAR_BI | PCI224_DACCON_VREF_1_25, PCI224_DACCON_POLAR_UNI | PCI224_DACCON_VREF_10, PCI224_DACCON_POLAR_UNI | PCI224_DACCON_VREF_5, PCI224_DACCON_POLAR_UNI | PCI224_DACCON_VREF_2_5, PCI224_DACCON_POLAR_UNI | PCI224_DACCON_VREF_1_25, /* jumper LK1 in position 2-3 */ PCI224_DACCON_POLAR_BI, PCI224_DACCON_POLAR_UNI, }; /* Used to check all channels set to the same range on PCI224. */ static const unsigned char range_check_pci224[10] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, }; /* * The ranges for PCI234. * * These are all hardware-selectable by jumper LK1 affecting all channels, * and jumpers LK2, LK3, LK4 and LK5 affecting channels 0, 1, 2 and 3 * individually. */ static const struct comedi_lrange range_pci234 = { 4, { /* LK1: 1-2 (fact def), LK2/3/4/5: 2-3 (fac def) */ BIP_RANGE(10), /* LK1: 1-2 (fact def), LK2/3/4/5: 1-2 */ BIP_RANGE(5), /* LK1: 2-3, LK2/3/4/5: 2-3 (fac def) */ RANGE_ext(-2, 2), /* bipolar [-2*Vext,+2*Vext] */ /* LK1: 2-3, LK2/3/4/5: 1-2 */ RANGE_ext(-1, 1), /* bipolar [-Vext,+Vext] */ } }; /* N.B. PCI234 ignores the polarity bit, but software uses it. */ static const unsigned short hwrange_pci234[4] = { PCI224_DACCON_POLAR_BI, PCI224_DACCON_POLAR_BI, PCI224_DACCON_POLAR_BI, PCI224_DACCON_POLAR_BI, }; /* Used to check all channels use same LK1 setting on PCI234. */ static const unsigned char range_check_pci234[4] = { 0, 0, 1, 1, }; /* * Board descriptions. */ enum pci224_model { pci224_model, pci234_model }; struct pci224_board { const char *name; unsigned int ao_chans; unsigned int ao_bits; const struct comedi_lrange *ao_range; const unsigned short *ao_hwrange; const unsigned char *ao_range_check; }; static const struct pci224_board pci224_boards[] = { [pci224_model] = { .name = "pci224", .ao_chans = 16, .ao_bits = 12, .ao_range = &range_pci224, .ao_hwrange = &hwrange_pci224[0], .ao_range_check = &range_check_pci224[0], }, [pci234_model] = { .name = "pci234", .ao_chans = 4, .ao_bits = 16, .ao_range = &range_pci234, .ao_hwrange = &hwrange_pci234[0], .ao_range_check = &range_check_pci234[0], }, }; struct pci224_private { unsigned long iobase1; unsigned long state; spinlock_t ao_spinlock; /* spinlock for AO command handling */ unsigned short *ao_scan_vals; unsigned char *ao_scan_order; int intr_cpuid; short intr_running; unsigned short daccon; unsigned short ao_enab; /* max 16 channels so 'short' will do */ unsigned char intsce; }; /* * Called from the 'insn_write' function to perform a single write. */ static void pci224_ao_set_data(struct comedi_device *dev, int chan, int range, unsigned int data) { const struct pci224_board *board = dev->board_ptr; struct pci224_private *devpriv = dev->private; unsigned short mangled; /* Enable the channel. */ outw(1 << chan, dev->iobase + PCI224_DACCEN); /* Set range and reset FIFO. */ devpriv->daccon = COMBINE(devpriv->daccon, board->ao_hwrange[range], PCI224_DACCON_POLAR_MASK | PCI224_DACCON_VREF_MASK); outw(devpriv->daccon | PCI224_DACCON_FIFORESET, dev->iobase + PCI224_DACCON); /* * Mangle the data. The hardware expects: * - bipolar: 16-bit 2's complement * - unipolar: 16-bit unsigned */ mangled = (unsigned short)data << (16 - board->ao_bits); if ((devpriv->daccon & PCI224_DACCON_POLAR_MASK) == PCI224_DACCON_POLAR_BI) { mangled ^= 0x8000; } /* Write mangled data to the FIFO. */ outw(mangled, dev->iobase + PCI224_DACDATA); /* Trigger the conversion. */ inw(dev->iobase + PCI224_SOFTTRIG); } static int pci224_ao_insn_write(struct comedi_device *dev, struct comedi_subdevice *s, struct comedi_insn *insn, unsigned int *data) { unsigned int chan = CR_CHAN(insn->chanspec); unsigned int range = CR_RANGE(insn->chanspec); unsigned int val = s->readback[chan]; int i; for (i = 0; i < insn->n; i++) { val = data[i]; pci224_ao_set_data(dev, chan, range, val); } s->readback[chan] = val; return insn->n; } /* * Kills a command running on the AO subdevice. */ static void pci224_ao_stop(struct comedi_device *dev, struct comedi_subdevice *s) { struct pci224_private *devpriv = dev->private; unsigned long flags; if (!test_and_clear_bit(AO_CMD_STARTED, &devpriv->state)) return; spin_lock_irqsave(&devpriv->ao_spinlock, flags); /* Kill the interrupts. */ devpriv->intsce = 0; outb(0, devpriv->iobase1 + PCI224_INT_SCE); /* * Interrupt routine may or may not be running. We may or may not * have been called from the interrupt routine (directly or * indirectly via a comedi_events() callback routine). It's highly * unlikely that we've been called from some other interrupt routine * but who knows what strange things coders get up to! * * If the interrupt routine is currently running, wait for it to * finish, unless we appear to have been called via the interrupt * routine. */ while (devpriv->intr_running && devpriv->intr_cpuid != THISCPU) { spin_unlock_irqrestore(&devpriv->ao_spinlock, flags); spin_lock_irqsave(&devpriv->ao_spinlock, flags); } spin_unlock_irqrestore(&devpriv->ao_spinlock, flags); /* Reconfigure DAC for insn_write usage. */ outw(0, dev->iobase + PCI224_DACCEN); /* Disable channels. */ devpriv->daccon = COMBINE(devpriv->daccon, PCI224_DACCON_TRIG_SW | PCI224_DACCON_FIFOINTR_EMPTY, PCI224_DACCON_TRIG_MASK | PCI224_DACCON_FIFOINTR_MASK); outw(devpriv->daccon | PCI224_DACCON_FIFORESET, dev->iobase + PCI224_DACCON); } /* * Handles start of acquisition for the AO subdevice. */ static void pci224_ao_start(struct comedi_device *dev, struct comedi_subdevice *s) { struct pci224_private *devpriv = dev->private; struct comedi_cmd *cmd = &s->async->cmd; unsigned long flags; set_bit(AO_CMD_STARTED, &devpriv->state); /* Enable interrupts. */ spin_lock_irqsave(&devpriv->ao_spinlock, flags); if (cmd->stop_src == TRIG_EXT) devpriv->intsce = PCI224_INTR_EXT | PCI224_INTR_DAC; else devpriv->intsce = PCI224_INTR_DAC; outb(devpriv->intsce, devpriv->iobase1 + PCI224_INT_SCE); spin_unlock_irqrestore(&devpriv->ao_spinlock, flags); } /* * Handles interrupts from the DAC FIFO. */ static void pci224_ao_handle_fifo(struct comedi_device *dev, struct comedi_subdevice *s) { struct pci224_private *devpriv = dev->private; struct comedi_cmd *cmd = &s->async->cmd; unsigned int num_scans = comedi_nscans_left(s, 0); unsigned int room; unsigned short dacstat; unsigned int i, n; /* Determine how much room is in the FIFO (in samples). */ dacstat = inw(dev->iobase + PCI224_DACCON); switch (dacstat & PCI224_DACCON_FIFOFL_MASK) { case PCI224_DACCON_FIFOFL_EMPTY: room = PCI224_FIFO_ROOM_EMPTY; if (cmd->stop_src == TRIG_COUNT && s->async->scans_done >= cmd->stop_arg) { /* FIFO empty at end of counted acquisition. */ s->async->events |= COMEDI_CB_EOA; comedi_handle_events(dev, s); return; } break; case PCI224_DACCON_FIFOFL_ONETOHALF: room = PCI224_FIFO_ROOM_ONETOHALF; break; case PCI224_DACCON_FIFOFL_HALFTOFULL: room = PCI224_FIFO_ROOM_HALFTOFULL; break; default: room = PCI224_FIFO_ROOM_FULL; break; } if (room >= PCI224_FIFO_ROOM_ONETOHALF) { /* FIFO is less than half-full. */ if (num_scans == 0) { /* Nothing left to put in the FIFO. */ dev_err(dev->class_dev, "AO buffer underrun\n"); s->async->events |= COMEDI_CB_OVERFLOW; } } /* Determine how many new scans can be put in the FIFO. */ room /= cmd->chanlist_len; /* Determine how many scans to process. */ if (num_scans > room) num_scans = room; /* Process scans. */ for (n = 0; n < num_scans; n++) { comedi_buf_read_samples(s, &devpriv->ao_scan_vals[0], cmd->chanlist_len); for (i = 0; i < cmd->chanlist_len; i++) { outw(devpriv->ao_scan_vals[devpriv->ao_scan_order[i]], dev->iobase + PCI224_DACDATA); } } if (cmd->stop_src == TRIG_COUNT && s->async->scans_done >= cmd->stop_arg) { /* * Change FIFO interrupt trigger level to wait * until FIFO is empty. */ devpriv->daccon = COMBINE(devpriv->daccon, PCI224_DACCON_FIFOINTR_EMPTY, PCI224_DACCON_FIFOINTR_MASK); outw(devpriv->daccon, dev->iobase + PCI224_DACCON); } if ((devpriv->daccon & PCI224_DACCON_TRIG_MASK) == PCI224_DACCON_TRIG_NONE) { unsigned short trig; /* * This is the initial DAC FIFO interrupt at the * start of the acquisition. The DAC's scan trigger * has been set to 'none' up until now. * * Now that data has been written to the FIFO, the * DAC's scan trigger source can be set to the * correct value. * * BUG: The first scan will be triggered immediately * if the scan trigger source is at logic level 1. */ if (cmd->scan_begin_src == TRIG_TIMER) { trig = PCI224_DACCON_TRIG_Z2CT0; } else { /* cmd->scan_begin_src == TRIG_EXT */ if (cmd->scan_begin_arg & CR_INVERT) trig = PCI224_DACCON_TRIG_EXTN; else trig = PCI224_DACCON_TRIG_EXTP; } devpriv->daccon = COMBINE(devpriv->daccon, trig, PCI224_DACCON_TRIG_MASK); outw(devpriv->daccon, dev->iobase + PCI224_DACCON); } comedi_handle_events(dev, s); } static int pci224_ao_inttrig_start(struct comedi_device *dev, struct comedi_subdevice *s, unsigned int trig_num) { struct comedi_cmd *cmd = &s->async->cmd; if (trig_num != cmd->start_arg) return -EINVAL; s->async->inttrig = NULL; pci224_ao_start(dev, s); return 1; } static int pci224_ao_check_chanlist(struct comedi_device *dev, struct comedi_subdevice *s, struct comedi_cmd *cmd) { const struct pci224_board *board = dev->board_ptr; unsigned int range_check_0; unsigned int chan_mask = 0; int i; range_check_0 = board->ao_range_check[CR_RANGE(cmd->chanlist[0])]; for (i = 0; i < cmd->chanlist_len; i++) { unsigned int chan = CR_CHAN(cmd->chanlist[i]); if (chan_mask & (1 << chan)) { dev_dbg(dev->class_dev, "%s: entries in chanlist must contain no duplicate channels\n", __func__); return -EINVAL; } chan_mask |= 1 << chan; if (board->ao_range_check[CR_RANGE(cmd->chanlist[i])] != range_check_0) { dev_dbg(dev->class_dev, "%s: entries in chanlist have incompatible ranges\n", __func__); return -EINVAL; } } return 0; } #define MAX_SCAN_PERIOD 0xFFFFFFFFU #define MIN_SCAN_PERIOD 2500 #define CONVERT_PERIOD 625 /* * 'do_cmdtest' function for AO subdevice. */ static int pci224_ao_cmdtest(struct comedi_device *dev, struct comedi_subdevice *s, struct comedi_cmd *cmd) { int err = 0; unsigned int arg; /* Step 1 : check if triggers are trivially valid */ err |= comedi_check_trigger_src(&cmd->start_src, TRIG_INT | TRIG_EXT); err |= comedi_check_trigger_src(&cmd->scan_begin_src, TRIG_EXT | TRIG_TIMER); err |= comedi_check_trigger_src(&cmd->convert_src, TRIG_NOW); err |= comedi_check_trigger_src(&cmd->scan_end_src, TRIG_COUNT); err |= comedi_check_trigger_src(&cmd->stop_src, TRIG_COUNT | TRIG_EXT | TRIG_NONE); if (err) return 1; /* Step 2a : make sure trigger sources are unique */ err |= comedi_check_trigger_is_unique(cmd->start_src); err |= comedi_check_trigger_is_unique(cmd->scan_begin_src); err |= comedi_check_trigger_is_unique(cmd->stop_src); /* Step 2b : and mutually compatible */ /* * There's only one external trigger signal (which makes these * tests easier). Only one thing can use it. */ arg = 0; if (cmd->start_src & TRIG_EXT) arg++; if (cmd->scan_begin_src & TRIG_EXT) arg++; if (cmd->stop_src & TRIG_EXT) arg++; if (arg > 1) err |= -EINVAL; if (err) return 2; /* Step 3: check if arguments are trivially valid */ switch (cmd->start_src) { case TRIG_INT: err |= comedi_check_trigger_arg_is(&cmd->start_arg, 0); break; case TRIG_EXT: /* Force to external trigger 0. */ if (cmd->start_arg & ~CR_FLAGS_MASK) { cmd->start_arg = COMBINE(cmd->start_arg, 0, ~CR_FLAGS_MASK); err |= -EINVAL; } /* The only flag allowed is CR_EDGE, which is ignored. */ if (cmd->start_arg & CR_FLAGS_MASK & ~CR_EDGE) { cmd->start_arg = COMBINE(cmd->start_arg, 0, CR_FLAGS_MASK & ~CR_EDGE); err |= -EINVAL; } break; } switch (cmd->scan_begin_src) { case TRIG_TIMER: err |= comedi_check_trigger_arg_max(&cmd->scan_begin_arg, MAX_SCAN_PERIOD); arg = cmd->chanlist_len * CONVERT_PERIOD; if (arg < MIN_SCAN_PERIOD) arg = MIN_SCAN_PERIOD; err |= comedi_check_trigger_arg_min(&cmd->scan_begin_arg, arg); break; case TRIG_EXT: /* Force to external trigger 0. */ if (cmd->scan_begin_arg & ~CR_FLAGS_MASK) { cmd->scan_begin_arg = COMBINE(cmd->scan_begin_arg, 0, ~CR_FLAGS_MASK); err |= -EINVAL; } /* Only allow flags CR_EDGE and CR_INVERT. Ignore CR_EDGE. */ if (cmd->scan_begin_arg & CR_FLAGS_MASK & ~(CR_EDGE | CR_INVERT)) { cmd->scan_begin_arg = COMBINE(cmd->scan_begin_arg, 0, CR_FLAGS_MASK & ~(CR_EDGE | CR_INVERT)); err |= -EINVAL; } break; } err |= comedi_check_trigger_arg_is(&cmd->convert_arg, 0); err |= comedi_check_trigger_arg_is(&cmd->scan_end_arg, cmd->chanlist_len); switch (cmd->stop_src) { case TRIG_COUNT: err |= comedi_check_trigger_arg_min(&cmd->stop_arg, 1); break; case TRIG_EXT: /* Force to external trigger 0. */ if (cmd->stop_arg & ~CR_FLAGS_MASK) { cmd->stop_arg = COMBINE(cmd->stop_arg, 0, ~CR_FLAGS_MASK); err |= -EINVAL; } /* The only flag allowed is CR_EDGE, which is ignored. */ if (cmd->stop_arg & CR_FLAGS_MASK & ~CR_EDGE) { cmd->stop_arg = COMBINE(cmd->stop_arg, 0, CR_FLAGS_MASK & ~CR_EDGE); } break; case TRIG_NONE: err |= comedi_check_trigger_arg_is(&cmd->stop_arg, 0); break; } if (err) return 3; /* Step 4: fix up any arguments. */ if (cmd->scan_begin_src == TRIG_TIMER) { arg = cmd->scan_begin_arg; /* Use two timers. */ comedi_8254_cascade_ns_to_timer(dev->pacer, &arg, cmd->flags); err |= comedi_check_trigger_arg_is(&cmd->scan_begin_arg, arg); } if (err) return 4; /* Step 5: check channel list if it exists */ if (cmd->chanlist && cmd->chanlist_len > 0) err |= pci224_ao_check_chanlist(dev, s, cmd); if (err) return 5; return 0; } static void pci224_ao_start_pacer(struct comedi_device *dev, struct comedi_subdevice *s) { struct pci224_private *devpriv = dev->private; /* * The output of timer Z2-0 will be used as the scan trigger * source. */ /* Make sure Z2-0 is gated on. */ outb(pci224_gat_config(0, GAT_VCC), devpriv->iobase1 + PCI224_ZGAT_SCE); /* Cascading with Z2-2. */ /* Make sure Z2-2 is gated on. */ outb(pci224_gat_config(2, GAT_VCC), devpriv->iobase1 + PCI224_ZGAT_SCE); /* Z2-2 needs 10 MHz clock. */ outb(pci224_clk_config(2, CLK_10MHZ), devpriv->iobase1 + PCI224_ZCLK_SCE); /* Z2-0 is clocked from Z2-2's output. */ outb(pci224_clk_config(0, CLK_OUTNM1), devpriv->iobase1 + PCI224_ZCLK_SCE); comedi_8254_pacer_enable(dev->pacer, 2, 0, false); } static int pci224_ao_cmd(struct comedi_device *dev, struct comedi_subdevice *s) { const struct pci224_board *board = dev->board_ptr; struct pci224_private *devpriv = dev->private; struct comedi_cmd *cmd = &s->async->cmd; int range; unsigned int i, j; unsigned int ch; unsigned int rank; unsigned long flags; /* Cannot handle null/empty chanlist. */ if (!cmd->chanlist || cmd->chanlist_len == 0) return -EINVAL; /* Determine which channels are enabled and their load order. */ devpriv->ao_enab = 0; for (i = 0; i < cmd->chanlist_len; i++) { ch = CR_CHAN(cmd->chanlist[i]); devpriv->ao_enab |= 1U << ch; rank = 0; for (j = 0; j < cmd->chanlist_len; j++) { if (CR_CHAN(cmd->chanlist[j]) < ch) rank++; } devpriv->ao_scan_order[rank] = i; } /* Set enabled channels. */ outw(devpriv->ao_enab, dev->iobase + PCI224_DACCEN); /* Determine range and polarity. All channels the same. */ range = CR_RANGE(cmd->chanlist[0]); /* * Set DAC range and polarity. * Set DAC scan trigger source to 'none'. * Set DAC FIFO interrupt trigger level to 'not half full'. * Reset DAC FIFO. * * N.B. DAC FIFO interrupts are currently disabled. */ devpriv->daccon = COMBINE(devpriv->daccon, board->ao_hwrange[range] | PCI224_DACCON_TRIG_NONE | PCI224_DACCON_FIFOINTR_NHALF, PCI224_DACCON_POLAR_MASK | PCI224_DACCON_VREF_MASK | PCI224_DACCON_TRIG_MASK | PCI224_DACCON_FIFOINTR_MASK); outw(devpriv->daccon | PCI224_DACCON_FIFORESET, dev->iobase + PCI224_DACCON); if (cmd->scan_begin_src == TRIG_TIMER) { comedi_8254_update_divisors(dev->pacer); pci224_ao_start_pacer(dev, s); } spin_lock_irqsave(&devpriv->ao_spinlock, flags); if (cmd->start_src == TRIG_INT) { s->async->inttrig = pci224_ao_inttrig_start; } else { /* TRIG_EXT */ /* Enable external interrupt trigger to start acquisition. */ devpriv->intsce |= PCI224_INTR_EXT; outb(devpriv->intsce, devpriv->iobase1 + PCI224_INT_SCE); } spin_unlock_irqrestore(&devpriv->ao_spinlock, flags); return 0; } /* * 'cancel' function for AO subdevice. */ static int pci224_ao_cancel(struct comedi_device *dev, struct comedi_subdevice *s) { pci224_ao_stop(dev, s); return 0; } /* * 'munge' data for AO command. */ static void pci224_ao_munge(struct comedi_device *dev, struct comedi_subdevice *s, void *data, unsigned int num_bytes, unsigned int chan_index) { const struct pci224_board *board = dev->board_ptr; struct comedi_cmd *cmd = &s->async->cmd; unsigned short *array = data; unsigned int length = num_bytes / sizeof(*array); unsigned int offset; unsigned int shift; unsigned int i; /* The hardware expects 16-bit numbers. */ shift = 16 - board->ao_bits; /* Channels will be all bipolar or all unipolar. */ if ((board->ao_hwrange[CR_RANGE(cmd->chanlist[0])] & PCI224_DACCON_POLAR_MASK) == PCI224_DACCON_POLAR_UNI) { /* Unipolar */ offset = 0; } else { /* Bipolar */ offset = 32768; } /* Munge the data. */ for (i = 0; i < length; i++) array[i] = (array[i] << shift) - offset; } /* * Interrupt handler. */ static irqreturn_t pci224_interrupt(int irq, void *d) { struct comedi_device *dev = d; struct pci224_private *devpriv = dev->private; struct comedi_subdevice *s = dev->write_subdev; struct comedi_cmd *cmd; unsigned char intstat, valid_intstat; unsigned char curenab; int retval = 0; unsigned long flags; intstat = inb(devpriv->iobase1 + PCI224_INT_SCE) & 0x3F; if (intstat) { retval = 1; spin_lock_irqsave(&devpriv->ao_spinlock, flags); valid_intstat = devpriv->intsce & intstat; /* Temporarily disable interrupt sources. */ curenab = devpriv->intsce & ~intstat; outb(curenab, devpriv->iobase1 + PCI224_INT_SCE); devpriv->intr_running = 1; devpriv->intr_cpuid = THISCPU; spin_unlock_irqrestore(&devpriv->ao_spinlock, flags); if (valid_intstat) { cmd = &s->async->cmd; if (valid_intstat & PCI224_INTR_EXT) { devpriv->intsce &= ~PCI224_INTR_EXT; if (cmd->start_src == TRIG_EXT) pci224_ao_start(dev, s); else if (cmd->stop_src == TRIG_EXT) pci224_ao_stop(dev, s); } if (valid_intstat & PCI224_INTR_DAC) pci224_ao_handle_fifo(dev, s); } /* Reenable interrupt sources. */ spin_lock_irqsave(&devpriv->ao_spinlock, flags); if (curenab != devpriv->intsce) { outb(devpriv->intsce, devpriv->iobase1 + PCI224_INT_SCE); } devpriv->intr_running = 0; spin_unlock_irqrestore(&devpriv->ao_spinlock, flags); } return IRQ_RETVAL(retval); } static int pci224_auto_attach(struct comedi_device *dev, unsigned long context_model) { struct pci_dev *pci_dev = comedi_to_pci_dev(dev); const struct pci224_board *board = NULL; struct pci224_private *devpriv; struct comedi_subdevice *s; unsigned int irq; int ret; if (context_model < ARRAY_SIZE(pci224_boards)) board = &pci224_boards[context_model]; if (!board || !board->name) { dev_err(dev->class_dev, "amplc_pci224: BUG! cannot determine board type!\n"); return -EINVAL; } dev->board_ptr = board; dev->board_name = board->name; dev_info(dev->class_dev, "amplc_pci224: attach pci %s - %s\n", pci_name(pci_dev), dev->board_name); devpriv = comedi_alloc_devpriv(dev, sizeof(*devpriv)); if (!devpriv) return -ENOMEM; ret = comedi_pci_enable(dev); if (ret) return ret; spin_lock_init(&devpriv->ao_spinlock); devpriv->iobase1 = pci_resource_start(pci_dev, 2); dev->iobase = pci_resource_start(pci_dev, 3); irq = pci_dev->irq; /* Allocate buffer to hold values for AO channel scan. */ devpriv->ao_scan_vals = kmalloc_array(board->ao_chans, sizeof(devpriv->ao_scan_vals[0]), GFP_KERNEL); if (!devpriv->ao_scan_vals) return -ENOMEM; /* Allocate buffer to hold AO channel scan order. */ devpriv->ao_scan_order = kmalloc_array(board->ao_chans, sizeof(devpriv->ao_scan_order[0]), GFP_KERNEL); if (!devpriv->ao_scan_order) return -ENOMEM; /* Disable interrupt sources. */ devpriv->intsce = 0; outb(0, devpriv->iobase1 + PCI224_INT_SCE); /* Initialize the DAC hardware. */ outw(PCI224_DACCON_GLOBALRESET, dev->iobase + PCI224_DACCON); outw(0, dev->iobase + PCI224_DACCEN); outw(0, dev->iobase + PCI224_FIFOSIZ); devpriv->daccon = PCI224_DACCON_TRIG_SW | PCI224_DACCON_POLAR_BI | PCI224_DACCON_FIFOENAB | PCI224_DACCON_FIFOINTR_EMPTY; outw(devpriv->daccon | PCI224_DACCON_FIFORESET, dev->iobase + PCI224_DACCON); dev->pacer = comedi_8254_init(devpriv->iobase1 + PCI224_Z2_BASE, I8254_OSC_BASE_10MHZ, I8254_IO8, 0); if (!dev->pacer) return -ENOMEM; ret = comedi_alloc_subdevices(dev, 1); if (ret) return ret; s = &dev->subdevices[0]; /* Analog output subdevice. */ s->type = COMEDI_SUBD_AO; s->subdev_flags = SDF_WRITABLE | SDF_GROUND | SDF_CMD_WRITE; s->n_chan = board->ao_chans; s->maxdata = (1 << board->ao_bits) - 1; s->range_table = board->ao_range; s->insn_write = pci224_ao_insn_write; s->len_chanlist = s->n_chan; dev->write_subdev = s; s->do_cmd = pci224_ao_cmd; s->do_cmdtest = pci224_ao_cmdtest; s->cancel = pci224_ao_cancel; s->munge = pci224_ao_munge; ret = comedi_alloc_subdev_readback(s); if (ret) return ret; if (irq) { ret = request_irq(irq, pci224_interrupt, IRQF_SHARED, dev->board_name, dev); if (ret < 0) { dev_err(dev->class_dev, "error! unable to allocate irq %u\n", irq); return ret; } dev->irq = irq; } return 0; } static void pci224_detach(struct comedi_device *dev) { struct pci224_private *devpriv = dev->private; comedi_pci_detach(dev); if (devpriv) { kfree(devpriv->ao_scan_vals); kfree(devpriv->ao_scan_order); } } static struct comedi_driver amplc_pci224_driver = { .driver_name = "amplc_pci224", .module = THIS_MODULE, .detach = pci224_detach, .auto_attach = pci224_auto_attach, .board_name = &pci224_boards[0].name, .offset = sizeof(struct pci224_board), .num_names = ARRAY_SIZE(pci224_boards), }; static int amplc_pci224_pci_probe(struct pci_dev *dev, const struct pci_device_id *id) { return comedi_pci_auto_config(dev, &lc_pci224_driver, id->driver_data); } static const struct pci_device_id amplc_pci224_pci_table[] = { { PCI_VDEVICE(AMPLICON, 0x0007), pci224_model }, { PCI_VDEVICE(AMPLICON, 0x0008), pci234_model }, { 0 } }; MODULE_DEVICE_TABLE(pci, amplc_pci224_pci_table); static struct pci_driver amplc_pci224_pci_driver = { .name = "amplc_pci224", .id_table = amplc_pci224_pci_table, .probe = amplc_pci224_pci_probe, .remove = comedi_pci_auto_unconfig, }; module_comedi_pci_driver(amplc_pci224_driver, amplc_pci224_pci_driver); MODULE_AUTHOR("Comedi http://www.comedi.org"); MODULE_DESCRIPTION("Comedi driver for Amplicon PCI224 and PCI234 AO boards"); MODULE_LICENSE("GPL");