Lines Matching +full:adc +full:- +full:chan
2 * TI ADC MFD driver
4 * Copyright (C) 2012 Texas Instruments Incorporated - https://www.ti.com/
34 #include <linux/dma-mapping.h>
40 struct dma_chan *chan; member
62 static unsigned int tiadc_readl(struct tiadc_device *adc, unsigned int reg) in tiadc_readl() argument
64 return readl(adc->mfd_tscadc->tscadc_base + reg); in tiadc_readl()
67 static void tiadc_writel(struct tiadc_device *adc, unsigned int reg, in tiadc_writel() argument
70 writel(val, adc->mfd_tscadc->tscadc_base + reg); in tiadc_writel()
77 step_en = ((1 << adc_dev->channels) - 1); in get_adc_step_mask()
78 step_en <<= TOTAL_STEPS - adc_dev->channels + 1; in get_adc_step_mask()
83 struct iio_chan_spec const *chan) in get_adc_chan_step_mask() argument
87 for (i = 0; i < ARRAY_SIZE(adc_dev->channel_step); i++) { in get_adc_chan_step_mask()
88 if (chan->channel == adc_dev->channel_line[i]) { in get_adc_chan_step_mask()
91 step = adc_dev->channel_step[i]; in get_adc_chan_step_mask()
100 static u32 get_adc_step_bit(struct tiadc_device *adc_dev, int chan) in get_adc_step_bit() argument
102 return 1 << adc_dev->channel_step[chan]; in get_adc_step_bit()
108 struct device *dev = adc_dev->mfd_tscadc->dev; in tiadc_step_config()
114 * lines available which are shared between Touchscreen and ADC. in tiadc_step_config()
116 * Steps forwards i.e. from 0 towards 16 are used by ADC in tiadc_step_config()
119 * needs to be given to ADC to digitalize data. in tiadc_step_config()
123 for (i = 0; i < adc_dev->channels; i++) { in tiadc_step_config()
124 int chan; in tiadc_step_config() local
126 chan = adc_dev->channel_line[i]; in tiadc_step_config()
128 if (adc_dev->step_avg[i] > STEPCONFIG_AVG_16) { in tiadc_step_config()
129 dev_warn(dev, "chan %d step_avg truncating to %d\n", in tiadc_step_config()
130 chan, STEPCONFIG_AVG_16); in tiadc_step_config()
131 adc_dev->step_avg[i] = STEPCONFIG_AVG_16; in tiadc_step_config()
134 if (adc_dev->step_avg[i]) in tiadc_step_config()
136 STEPCONFIG_AVG(ffs(adc_dev->step_avg[i]) - 1) | in tiadc_step_config()
145 stepconfig | STEPCONFIG_INP(chan) | in tiadc_step_config()
150 if (adc_dev->open_delay[i] > STEPDELAY_OPEN_MASK) { in tiadc_step_config()
151 dev_warn(dev, "chan %d open delay truncating to 0x3FFFF\n", in tiadc_step_config()
152 chan); in tiadc_step_config()
153 adc_dev->open_delay[i] = STEPDELAY_OPEN_MASK; in tiadc_step_config()
156 if (adc_dev->sample_delay[i] > 0xFF) { in tiadc_step_config()
157 dev_warn(dev, "chan %d sample delay truncating to 0xFF\n", in tiadc_step_config()
158 chan); in tiadc_step_config()
159 adc_dev->sample_delay[i] = 0xFF; in tiadc_step_config()
163 STEPDELAY_OPEN(adc_dev->open_delay[i]) | in tiadc_step_config()
164 STEPDELAY_SAMPLE(adc_dev->sample_delay[i])); in tiadc_step_config()
166 adc_dev->channel_step[i] = steps; in tiadc_step_config()
181 * ADC and touchscreen share the IRQ line. in tiadc_irq_h()
185 /* FIFO Overrun. Clear flag. Disable/Enable ADC to recover */ in tiadc_irq_h()
193 * ADC needs to finish the current conversion in tiadc_irq_h()
216 u16 *data = adc_dev->data; in tiadc_worker_h()
220 for (i = 0; i < (indio_dev->scan_bytes)/2; i++) { in tiadc_worker_h()
237 struct tiadc_dma *dma = &adc_dev->dma; in tiadc_dma_rx_complete()
241 data = dma->buf + dma->current_period * dma->period_size; in tiadc_dma_rx_complete()
242 dma->current_period = 1 - dma->current_period; /* swap the buffer ID */ in tiadc_dma_rx_complete()
244 for (i = 0; i < dma->period_size; i += indio_dev->scan_bytes) { in tiadc_dma_rx_complete()
246 data += indio_dev->scan_bytes; in tiadc_dma_rx_complete()
253 struct tiadc_dma *dma = &adc_dev->dma; in tiadc_start_dma()
256 dma->current_period = 0; /* We start to fill period 0 */ in tiadc_start_dma()
264 dma->fifo_thresh = rounddown(FIFO1_THRESHOLD + 1, in tiadc_start_dma()
265 adc_dev->total_ch_enabled) - 1; in tiadc_start_dma()
267 dma->period_size = rounddown(DMA_BUFFER_SIZE / 2, in tiadc_start_dma()
268 (dma->fifo_thresh + 1) * sizeof(u16)); in tiadc_start_dma()
270 dma->conf.src_maxburst = dma->fifo_thresh + 1; in tiadc_start_dma()
271 dmaengine_slave_config(dma->chan, &dma->conf); in tiadc_start_dma()
273 desc = dmaengine_prep_dma_cyclic(dma->chan, dma->addr, in tiadc_start_dma()
274 dma->period_size * 2, in tiadc_start_dma()
275 dma->period_size, DMA_DEV_TO_MEM, in tiadc_start_dma()
278 return -EBUSY; in tiadc_start_dma()
280 desc->callback = tiadc_dma_rx_complete; in tiadc_start_dma()
281 desc->callback_param = indio_dev; in tiadc_start_dma()
283 dma->cookie = dmaengine_submit(desc); in tiadc_start_dma()
285 dma_async_issue_pending(dma->chan); in tiadc_start_dma()
287 tiadc_writel(adc_dev, REG_FIFO1THR, dma->fifo_thresh); in tiadc_start_dma()
288 tiadc_writel(adc_dev, REG_DMA1REQ, dma->fifo_thresh); in tiadc_start_dma()
303 /* Flush FIFO. Needed in corner cases in simultaneous tsc/adc use */ in tiadc_buffer_preenable()
314 struct tiadc_dma *dma = &adc_dev->dma; in tiadc_buffer_postenable()
320 for_each_set_bit(bit, indio_dev->active_scan_mask, adc_dev->channels) { in tiadc_buffer_postenable()
322 adc_dev->total_ch_enabled++; in tiadc_buffer_postenable()
324 adc_dev->buffer_en_ch_steps = enb; in tiadc_buffer_postenable()
326 if (dma->chan) in tiadc_buffer_postenable()
329 am335x_tsc_se_set_cache(adc_dev->mfd_tscadc, enb); in tiadc_buffer_postenable()
335 if (!dma->chan) in tiadc_buffer_postenable()
345 struct tiadc_dma *dma = &adc_dev->dma; in tiadc_buffer_predisable()
350 am335x_tsc_se_clr(adc_dev->mfd_tscadc, adc_dev->buffer_en_ch_steps); in tiadc_buffer_predisable()
351 adc_dev->buffer_en_ch_steps = 0; in tiadc_buffer_predisable()
352 adc_dev->total_ch_enabled = 0; in tiadc_buffer_predisable()
353 if (dma->chan) { in tiadc_buffer_predisable()
355 dmaengine_terminate_async(dma->chan); in tiadc_buffer_predisable()
358 /* Flush FIFO of leftover data in the time it takes to disable adc */ in tiadc_buffer_predisable()
393 return -ENOMEM; in tiadc_iio_buffered_hardware_setup()
398 flags, indio_dev->name, indio_dev); in tiadc_iio_buffered_hardware_setup()
402 indio_dev->setup_ops = setup_ops; in tiadc_iio_buffered_hardware_setup()
403 indio_dev->modes |= INDIO_BUFFER_SOFTWARE; in tiadc_iio_buffered_hardware_setup()
408 iio_kfifo_free(indio_dev->buffer); in tiadc_iio_buffered_hardware_setup()
428 struct iio_chan_spec *chan; in tiadc_channel_init() local
431 indio_dev->num_channels = channels; in tiadc_channel_init()
435 return -ENOMEM; in tiadc_channel_init()
437 chan = chan_array; in tiadc_channel_init()
438 for (i = 0; i < channels; i++, chan++) { in tiadc_channel_init()
440 chan->type = IIO_VOLTAGE; in tiadc_channel_init()
441 chan->indexed = 1; in tiadc_channel_init()
442 chan->channel = adc_dev->channel_line[i]; in tiadc_channel_init()
443 chan->info_mask_separate = BIT(IIO_CHAN_INFO_RAW); in tiadc_channel_init()
444 chan->datasheet_name = chan_name_ain[chan->channel]; in tiadc_channel_init()
445 chan->scan_index = i; in tiadc_channel_init()
446 chan->scan_type.sign = 'u'; in tiadc_channel_init()
447 chan->scan_type.realbits = 12; in tiadc_channel_init()
448 chan->scan_type.storagebits = 16; in tiadc_channel_init()
451 indio_dev->channels = chan_array; in tiadc_channel_init()
457 struct iio_chan_spec const *chan, in tiadc_read_raw() argument
469 return -EBUSY; in tiadc_read_raw()
471 step_en = get_adc_chan_step_mask(adc_dev, chan); in tiadc_read_raw()
473 return -EINVAL; in tiadc_read_raw()
475 mutex_lock(&adc_dev->fifo1_lock); in tiadc_read_raw()
477 while (fifo1count--) in tiadc_read_raw()
480 am335x_tsc_se_set_once(adc_dev->mfd_tscadc, step_en); in tiadc_read_raw()
483 (IDLE_TIMEOUT * adc_dev->channels); in tiadc_read_raw()
491 am335x_tsc_se_adc_done(adc_dev->mfd_tscadc); in tiadc_read_raw()
492 ret = -EAGAIN; in tiadc_read_raw()
496 map_val = adc_dev->channel_step[chan->scan_index]; in tiadc_read_raw()
500 * something went wrong we left empty handed (-EAGAIN previously) and in tiadc_read_raw()
516 am335x_tsc_se_adc_done(adc_dev->mfd_tscadc); in tiadc_read_raw()
519 ret = -EBUSY; in tiadc_read_raw()
522 mutex_unlock(&adc_dev->fifo1_lock); in tiadc_read_raw()
533 struct tiadc_dma *dma = &adc_dev->dma; in tiadc_request_dma()
537 dma->conf.direction = DMA_DEV_TO_MEM; in tiadc_request_dma()
538 dma->conf.src_addr_width = DMA_SLAVE_BUSWIDTH_2_BYTES; in tiadc_request_dma()
539 dma->conf.src_addr = adc_dev->mfd_tscadc->tscadc_phys_base + REG_FIFO1; in tiadc_request_dma()
545 dma->chan = dma_request_chan(adc_dev->mfd_tscadc->dev, "fifo1"); in tiadc_request_dma()
546 if (IS_ERR(dma->chan)) { in tiadc_request_dma()
547 int ret = PTR_ERR(dma->chan); in tiadc_request_dma()
549 dma->chan = NULL; in tiadc_request_dma()
554 dma->buf = dma_alloc_coherent(dma->chan->device->dev, DMA_BUFFER_SIZE, in tiadc_request_dma()
555 &dma->addr, GFP_KERNEL); in tiadc_request_dma()
556 if (!dma->buf) in tiadc_request_dma()
561 dma_release_channel(dma->chan); in tiadc_request_dma()
562 return -ENOMEM; in tiadc_request_dma()
568 struct device_node *node = pdev->dev.of_node; in tiadc_parse_dt()
574 of_property_for_each_u32(node, "ti,adc-channels", prop, cur, val) { in tiadc_parse_dt()
575 adc_dev->channel_line[channels] = val; in tiadc_parse_dt()
578 adc_dev->open_delay[channels] = STEPCONFIG_OPENDLY; in tiadc_parse_dt()
579 adc_dev->sample_delay[channels] = STEPCONFIG_SAMPLEDLY; in tiadc_parse_dt()
580 adc_dev->step_avg[channels] = 16; in tiadc_parse_dt()
585 of_property_read_u32_array(node, "ti,chan-step-avg", in tiadc_parse_dt()
586 adc_dev->step_avg, channels); in tiadc_parse_dt()
587 of_property_read_u32_array(node, "ti,chan-step-opendelay", in tiadc_parse_dt()
588 adc_dev->open_delay, channels); in tiadc_parse_dt()
589 of_property_read_u32_array(node, "ti,chan-step-sampledelay", in tiadc_parse_dt()
590 adc_dev->sample_delay, channels); in tiadc_parse_dt()
592 adc_dev->channels = channels; in tiadc_parse_dt()
600 struct device_node *node = pdev->dev.of_node; in tiadc_probe()
604 dev_err(&pdev->dev, "Could not find valid DT data.\n"); in tiadc_probe()
605 return -EINVAL; in tiadc_probe()
608 indio_dev = devm_iio_device_alloc(&pdev->dev, sizeof(*adc_dev)); in tiadc_probe()
610 dev_err(&pdev->dev, "failed to allocate iio device\n"); in tiadc_probe()
611 return -ENOMEM; in tiadc_probe()
615 adc_dev->mfd_tscadc = ti_tscadc_dev_get(pdev); in tiadc_probe()
618 indio_dev->name = dev_name(&pdev->dev); in tiadc_probe()
619 indio_dev->modes = INDIO_DIRECT_MODE; in tiadc_probe()
620 indio_dev->info = &tiadc_info; in tiadc_probe()
624 mutex_init(&adc_dev->fifo1_lock); in tiadc_probe()
626 err = tiadc_channel_init(&pdev->dev, indio_dev, adc_dev->channels); in tiadc_probe()
630 err = tiadc_iio_buffered_hardware_setup(&pdev->dev, indio_dev, in tiadc_probe()
633 adc_dev->mfd_tscadc->irq, in tiadc_probe()
647 if (err && err == -EPROBE_DEFER) in tiadc_probe()
663 struct tiadc_dma *dma = &adc_dev->dma; in tiadc_remove()
666 if (dma->chan) { in tiadc_remove()
667 dma_free_coherent(dma->chan->device->dev, DMA_BUFFER_SIZE, in tiadc_remove()
668 dma->buf, dma->addr); in tiadc_remove()
669 dma_release_channel(dma->chan); in tiadc_remove()
674 am335x_tsc_se_clr(adc_dev->mfd_tscadc, step_en); in tiadc_remove()
699 /* Make sure ADC is powered up */ in tiadc_resume()
705 am335x_tsc_se_set_cache(adc_dev->mfd_tscadc, in tiadc_resume()
706 adc_dev->buffer_en_ch_steps); in tiadc_resume()
713 { .compatible = "ti,am3359-adc", },
720 .name = "TI-am335x-adc",
729 MODULE_DESCRIPTION("TI ADC controller driver");