Lines Matching +full:adc +full:- +full:channels +full:- +full:used
1 // SPDX-License-Identifier: GPL-2.0-only
6 * Copyright (c) 2012-2020, The Linux Foundation. All rights reserved.
12 #include <linux/iio/adc/qcom-vadc-common.h>
21 #include <asm-generic/unaligned.h>
26 * Thermal monitoring block consists of 8 (ADC_TM5_NUM_CHANNELS) channels. Each
27 * channel is programmed to use one of ADC channels for voltage comparison.
28 * Voltages are programmed using ADC codes, so we have to convert temp to
29 * voltage and then to ADC code value.
31 * Configuration of TM channels must match configuration of corresponding ADC
32 * channels.
178 * struct adc_tm5_channel - ADC Thermal Monitoring channel data.
180 * @adc_channel: corresponding ADC channel number.
186 * @avg_samples: ability to provide single result from the ADC
191 * @iio: IIO channel instance used by this channel.
192 * @chip: ADC TM chip instance.
193 * @tzd: thermal zone device used by this channel.
212 * struct adc_tm5_chip - ADC Thermal Monitoring properties
216 * @channels: array of ADC TM channel data.
217 * @nchannels: amount of channels defined/allocated
219 * Applies to all channels, used only on Gen1 ADC_TM.
220 * @avg_samples: ability to provide single result from the ADC
222 * channels, used only on Gen1 ADC_TM.
224 * @adc_mutex_lock: ADC_TM mutex lock, used only on Gen2 ADC_TM.
225 * It is used to ensure only one ADC channel configuration
233 struct adc_tm5_channel *channels; member
243 return regmap_bulk_read(adc_tm->regmap, adc_tm->base + offset, data, len); in adc_tm5_read()
248 return regmap_bulk_write(adc_tm->regmap, adc_tm->base + offset, data, len); in adc_tm5_write()
253 return regmap_write_bits(adc_tm->regmap, adc_tm->base + offset, mask, val); in adc_tm5_reg_update()
264 dev_err(chip->dev, "read status low failed: %d\n", ret); in adc_tm5_isr()
270 dev_err(chip->dev, "read status high failed: %d\n", ret); in adc_tm5_isr()
274 for (i = 0; i < chip->nchannels; i++) { in adc_tm5_isr()
276 unsigned int ch = chip->channels[i].channel; in adc_tm5_isr()
279 if (!chip->channels[i].tzd) in adc_tm5_isr()
284 dev_err(chip->dev, "ctl read failed: %d, channel %d\n", ret, i); in adc_tm5_isr()
298 thermal_zone_device_update(chip->channels[i].tzd, in adc_tm5_isr()
313 dev_err(chip->dev, "read status_low failed: %d\n", ret); in adc_tm5_gen2_isr()
319 dev_err(chip->dev, "read status_high failed: %d\n", ret); in adc_tm5_gen2_isr()
325 dev_err(chip->dev, "clear status low failed with %d\n", ret); in adc_tm5_gen2_isr()
331 dev_err(chip->dev, "clear status high failed with %d\n", ret); in adc_tm5_gen2_isr()
335 for (i = 0; i < chip->nchannels; i++) { in adc_tm5_gen2_isr()
337 unsigned int ch = chip->channels[i].channel; in adc_tm5_gen2_isr()
340 if (!chip->channels[i].tzd) in adc_tm5_gen2_isr()
343 if (!chip->channels[i].meas_en) in adc_tm5_gen2_isr()
347 (chip->channels[i].low_thr_en); in adc_tm5_gen2_isr()
350 (chip->channels[i].high_thr_en); in adc_tm5_gen2_isr()
353 thermal_zone_device_update(chip->channels[i].tzd, in adc_tm5_gen2_isr()
362 struct adc_tm5_channel *channel = tz->devdata; in adc_tm5_get_temp()
365 if (!channel || !channel->iio) in adc_tm5_get_temp()
366 return -EINVAL; in adc_tm5_get_temp()
368 ret = iio_read_channel_processed(channel->iio, temp); in adc_tm5_get_temp()
373 return -EINVAL; in adc_tm5_get_temp()
380 struct adc_tm5_chip *chip = channel->chip; in adc_tm5_disable_channel()
381 unsigned int reg = ADC_TM5_M_EN(channel->channel); in adc_tm5_disable_channel()
403 dev_err(chip->dev, "adc-tm enable failed with %d\n", ret); in adc_tm5_gen2_conv_req()
410 dev_err(chip->dev, "adc-tm handshake failed with %d\n", ret); in adc_tm5_gen2_conv_req()
417 dev_err(chip->dev, "adc-tm request conversion failed with %d\n", ret); in adc_tm5_gen2_conv_req()
429 dev_err(chip->dev, "adc-tm read failed with %d\n", ret); in adc_tm5_gen2_conv_req()
439 dev_err(chip->dev, "adc-tm conversion request handshake timed out\n"); in adc_tm5_gen2_conv_req()
441 return -ETIMEDOUT; in adc_tm5_gen2_conv_req()
446 struct adc_tm5_chip *chip = channel->chip; in adc_tm5_gen2_disable_channel()
450 mutex_lock(&chip->adc_mutex_lock); in adc_tm5_gen2_disable_channel()
452 channel->meas_en = false; in adc_tm5_gen2_disable_channel()
453 channel->high_thr_en = false; in adc_tm5_gen2_disable_channel()
454 channel->low_thr_en = false; in adc_tm5_gen2_disable_channel()
458 dev_err(chip->dev, "adc-tm block read failed with %d\n", ret); in adc_tm5_gen2_disable_channel()
463 val |= FIELD_PREP(ADC_TM_GEN2_TM_CH_SEL, channel->channel); in adc_tm5_gen2_disable_channel()
467 dev_err(chip->dev, "adc-tm channel disable failed with %d\n", ret); in adc_tm5_gen2_disable_channel()
474 dev_err(chip->dev, "adc-tm interrupt disable failed with %d\n", ret); in adc_tm5_gen2_disable_channel()
479 ret = adc_tm5_gen2_conv_req(channel->chip); in adc_tm5_gen2_disable_channel()
481 dev_err(chip->dev, "adc-tm channel configure failed with %d\n", ret); in adc_tm5_gen2_disable_channel()
484 mutex_unlock(&chip->adc_mutex_lock); in adc_tm5_gen2_disable_channel()
496 dev_err(chip->dev, "adc-tm enable failed\n"); in adc_tm5_enable()
503 dev_err(chip->dev, "adc-tm request conversion failed\n"); in adc_tm5_enable()
512 struct adc_tm5_chip *chip = channel->chip; in adc_tm5_configure()
514 u16 reg = ADC_TM5_M_ADC_CH_SEL_CTL(channel->channel); in adc_tm5_configure()
519 dev_err(chip->dev, "channel %d params read failed: %d\n", channel->channel, ret); in adc_tm5_configure()
523 buf[0] = channel->adc_channel; in adc_tm5_configure()
527 u16 adc_code = qcom_adc_tm5_temp_volt_scale(channel->prescale, in adc_tm5_configure()
528 chip->data->full_scale_code_volt, high); in adc_tm5_configure()
537 if (low != -INT_MAX) { in adc_tm5_configure()
538 u16 adc_code = qcom_adc_tm5_temp_volt_scale(channel->prescale, in adc_tm5_configure()
539 chip->data->full_scale_code_volt, low); in adc_tm5_configure()
551 buf[6] |= FIELD_PREP(ADC_TM5_M_CTL_HW_SETTLE_DELAY_MASK, channel->hw_settle_time); in adc_tm5_configure()
553 buf[6] |= FIELD_PREP(ADC_TM5_M_CTL_CAL_SEL_MASK, channel->cal_method); in adc_tm5_configure()
559 dev_err(chip->dev, "channel %d params write failed: %d\n", channel->channel, ret); in adc_tm5_configure()
568 struct adc_tm5_chip *chip = channel->chip; in adc_tm5_gen2_configure()
573 mutex_lock(&chip->adc_mutex_lock); in adc_tm5_gen2_configure()
575 channel->meas_en = true; in adc_tm5_gen2_configure()
579 dev_err(chip->dev, "adc-tm block read failed with %d\n", ret); in adc_tm5_gen2_configure()
584 buf[0] = channel->adc_channel >> 8; in adc_tm5_gen2_configure()
586 /* Set TM channel number used and measurement interval */ in adc_tm5_gen2_configure()
588 buf[1] |= FIELD_PREP(ADC_TM_GEN2_TM_CH_SEL, channel->channel); in adc_tm5_gen2_configure()
593 buf[2] |= FIELD_PREP(ADC_TM_GEN2_CTL_DEC_RATIO_MASK, channel->decimation); in adc_tm5_gen2_configure()
595 buf[2] |= FIELD_PREP(ADC_TM_GEN2_CTL_CAL_SEL, channel->cal_method); in adc_tm5_gen2_configure()
597 buf[3] = channel->avg_samples | ADC_TM_GEN2_FAST_AVG_EN; in adc_tm5_gen2_configure()
599 buf[4] = channel->adc_channel & 0xff; in adc_tm5_gen2_configure()
601 buf[5] = channel->hw_settle_time & ADC_TM_GEN2_HW_SETTLE_DELAY; in adc_tm5_gen2_configure()
605 channel->low_thr_en = true; in adc_tm5_gen2_configure()
609 channel->low_thr_en = false; in adc_tm5_gen2_configure()
613 if (low != -INT_MAX) { in adc_tm5_gen2_configure()
614 channel->high_thr_en = true; in adc_tm5_gen2_configure()
618 channel->high_thr_en = false; in adc_tm5_gen2_configure()
622 if (channel->high_thr_en) in adc_tm5_gen2_configure()
624 if (channel->low_thr_en) in adc_tm5_gen2_configure()
629 dev_err(chip->dev, "channel %d params write failed: %d\n", channel->channel, ret); in adc_tm5_gen2_configure()
633 ret = adc_tm5_gen2_conv_req(channel->chip); in adc_tm5_gen2_configure()
635 dev_err(chip->dev, "adc-tm channel configure failed with %d\n", ret); in adc_tm5_gen2_configure()
638 mutex_unlock(&chip->adc_mutex_lock); in adc_tm5_gen2_configure()
644 struct adc_tm5_channel *channel = tz->devdata; in adc_tm5_set_trips()
649 return -EINVAL; in adc_tm5_set_trips()
651 chip = channel->chip; in adc_tm5_set_trips()
652 dev_dbg(chip->dev, "%d:low(mdegC):%d, high(mdegC):%d\n", in adc_tm5_set_trips()
653 channel->channel, low, high); in adc_tm5_set_trips()
655 if (high == INT_MAX && low <= -INT_MAX) in adc_tm5_set_trips()
656 ret = chip->data->disable_channel(channel); in adc_tm5_set_trips()
658 ret = chip->data->configure(channel, low, high); in adc_tm5_set_trips()
673 for (i = 0; i < adc_tm->nchannels; i++) { in adc_tm5_register_tzd()
674 adc_tm->channels[i].chip = adc_tm; in adc_tm5_register_tzd()
675 tzd = devm_thermal_of_zone_register(adc_tm->dev, in adc_tm5_register_tzd()
676 adc_tm->channels[i].channel, in adc_tm5_register_tzd()
677 &adc_tm->channels[i], in adc_tm5_register_tzd()
680 if (PTR_ERR(tzd) == -ENODEV) { in adc_tm5_register_tzd()
681 dev_warn(adc_tm->dev, "thermal sensor on channel %d is not used\n", in adc_tm5_register_tzd()
682 adc_tm->channels[i].channel); in adc_tm5_register_tzd()
686 dev_err(adc_tm->dev, "Error registering TZ zone for channel %d: %ld\n", in adc_tm5_register_tzd()
687 adc_tm->channels[i].channel, PTR_ERR(tzd)); in adc_tm5_register_tzd()
690 adc_tm->channels[i].tzd = tzd; in adc_tm5_register_tzd()
692 dev_warn(adc_tm->dev, in adc_tm5_register_tzd()
705 for (i = 0; i < chip->nchannels; i++) { in adc_tm_hc_init()
706 if (chip->channels[i].channel >= ADC_TM5_NUM_CHANNELS) { in adc_tm_hc_init()
707 dev_err(chip->dev, "Invalid channel %d\n", chip->channels[i].channel); in adc_tm_hc_init()
708 return -EINVAL; in adc_tm_hc_init()
712 buf[0] = chip->decimation; in adc_tm_hc_init()
713 buf[1] = chip->avg_samples | ADC_TM5_FAST_AVG_EN; in adc_tm_hc_init()
717 dev_err(chip->dev, "block write failed: %d\n", ret); in adc_tm_hc_init()
731 dev_err(chip->dev, "read failed for BTM channels\n"); in adc_tm5_init()
735 for (i = 0; i < chip->nchannels; i++) { in adc_tm5_init()
736 if (chip->channels[i].channel >= channels_available) { in adc_tm5_init()
737 dev_err(chip->dev, "Invalid channel %d\n", chip->channels[i].channel); in adc_tm5_init()
738 return -EINVAL; in adc_tm5_init()
742 buf[0] = chip->decimation; in adc_tm5_init()
743 buf[1] = chip->avg_samples | ADC_TM5_FAST_AVG_EN; in adc_tm5_init()
750 dev_err(chip->dev, "block write failed: %d\n", ret); in adc_tm5_init()
766 dev_err(chip->dev, "read failed for BTM channels\n"); in adc_tm5_gen2_init()
770 for (i = 0; i < chip->nchannels; i++) { in adc_tm5_gen2_init()
771 if (chip->channels[i].channel >= channels_available) { in adc_tm5_gen2_init()
772 dev_err(chip->dev, "Invalid channel %d\n", chip->channels[i].channel); in adc_tm5_gen2_init()
773 return -EINVAL; in adc_tm5_gen2_init()
777 mutex_init(&chip->adc_mutex_lock); in adc_tm5_gen2_init()
786 const char *name = node->name; in adc_tm5_get_dt_channel_data()
789 struct device *dev = adc_tm->dev; in adc_tm5_get_dt_channel_data()
800 return -EINVAL; in adc_tm5_get_dt_channel_data()
803 channel->channel = chan; in adc_tm5_get_dt_channel_data()
806 * We are tied to PMIC's ADC controller, which always use single in adc_tm5_get_dt_channel_data()
808 * #io-channel-cells, just enforce cell_count = 1. in adc_tm5_get_dt_channel_data()
810 ret = of_parse_phandle_with_fixed_args(node, "io-channels", 1, 0, &args); in adc_tm5_get_dt_channel_data()
812 dev_err(dev, "%s: error parsing ADC channel number %d: %d\n", name, chan, ret); in adc_tm5_get_dt_channel_data()
818 dev_err(dev, "%s: invalid args count for ADC channel %d\n", name, chan); in adc_tm5_get_dt_channel_data()
819 return -EINVAL; in adc_tm5_get_dt_channel_data()
823 if (adc_tm->data->gen == ADC_TM5_GEN2) in adc_tm5_get_dt_channel_data()
827 dev_err(dev, "%s: invalid ADC channel number %d\n", name, chan); in adc_tm5_get_dt_channel_data()
828 return -EINVAL; in adc_tm5_get_dt_channel_data()
830 channel->adc_channel = args.args[0]; in adc_tm5_get_dt_channel_data()
832 channel->iio = devm_fwnode_iio_channel_get_by_name(adc_tm->dev, in adc_tm5_get_dt_channel_data()
834 if (IS_ERR(channel->iio)) { in adc_tm5_get_dt_channel_data()
835 ret = PTR_ERR(channel->iio); in adc_tm5_get_dt_channel_data()
836 if (ret != -EPROBE_DEFER) in adc_tm5_get_dt_channel_data()
841 ret = of_property_read_u32_array(node, "qcom,pre-scaling", varr, 2); in adc_tm5_get_dt_channel_data()
845 dev_err(dev, "%s: invalid pre-scaling <%d %d>\n", in adc_tm5_get_dt_channel_data()
849 channel->prescale = ret; in adc_tm5_get_dt_channel_data()
852 channel->prescale = 0; in adc_tm5_get_dt_channel_data()
855 ret = of_property_read_u32(node, "qcom,hw-settle-time-us", &value); in adc_tm5_get_dt_channel_data()
857 ret = qcom_adc5_hw_settle_time_from_dt(value, adc_tm->data->hw_settle); in adc_tm5_get_dt_channel_data()
859 dev_err(dev, "%s invalid hw-settle-time-us %d us\n", in adc_tm5_get_dt_channel_data()
863 channel->hw_settle_time = ret; in adc_tm5_get_dt_channel_data()
865 channel->hw_settle_time = VADC_DEF_HW_SETTLE_TIME; in adc_tm5_get_dt_channel_data()
869 channel->cal_method = ADC_TM5_RATIOMETRIC_CAL; in adc_tm5_get_dt_channel_data()
871 channel->cal_method = ADC_TM5_ABSOLUTE_CAL; in adc_tm5_get_dt_channel_data()
873 if (adc_tm->data->gen == ADC_TM5_GEN2) { in adc_tm5_get_dt_channel_data()
876 ret = qcom_adc5_decimation_from_dt(value, adc_tm->data->decimation); in adc_tm5_get_dt_channel_data()
881 channel->decimation = ret; in adc_tm5_get_dt_channel_data()
883 channel->decimation = ADC5_DECIMATION_DEFAULT; in adc_tm5_get_dt_channel_data()
886 ret = of_property_read_u32(node, "qcom,avg-samples", &value); in adc_tm5_get_dt_channel_data()
890 dev_err(dev, "invalid avg-samples %d\n", value); in adc_tm5_get_dt_channel_data()
893 channel->avg_samples = ret; in adc_tm5_get_dt_channel_data()
895 channel->avg_samples = VADC_DEF_AVG_SAMPLES; in adc_tm5_get_dt_channel_data()
912 .irq_name = "pm-adc-tm5",
925 .irq_name = "pm-adc-tm5",
939 .irq_name = "pm-adc-tm5-gen2",
945 struct adc_tm5_channel *channels; in adc_tm5_get_dt_data() local
949 struct device *dev = adc_tm->dev; in adc_tm5_get_dt_data()
951 adc_tm->nchannels = of_get_available_child_count(node); in adc_tm5_get_dt_data()
952 if (!adc_tm->nchannels) in adc_tm5_get_dt_data()
953 return -EINVAL; in adc_tm5_get_dt_data()
955 adc_tm->channels = devm_kcalloc(dev, adc_tm->nchannels, in adc_tm5_get_dt_data()
956 sizeof(*adc_tm->channels), GFP_KERNEL); in adc_tm5_get_dt_data()
957 if (!adc_tm->channels) in adc_tm5_get_dt_data()
958 return -ENOMEM; in adc_tm5_get_dt_data()
960 channels = adc_tm->channels; in adc_tm5_get_dt_data()
962 adc_tm->data = of_device_get_match_data(dev); in adc_tm5_get_dt_data()
963 if (!adc_tm->data) in adc_tm5_get_dt_data()
964 adc_tm->data = &adc_tm5_data_pmic; in adc_tm5_get_dt_data()
968 ret = qcom_adc5_decimation_from_dt(value, adc_tm->data->decimation); in adc_tm5_get_dt_data()
973 adc_tm->decimation = ret; in adc_tm5_get_dt_data()
975 adc_tm->decimation = ADC5_DECIMATION_DEFAULT; in adc_tm5_get_dt_data()
978 ret = of_property_read_u32(node, "qcom,avg-samples", &value); in adc_tm5_get_dt_data()
982 dev_err(dev, "invalid avg-samples %d\n", value); in adc_tm5_get_dt_data()
985 adc_tm->avg_samples = ret; in adc_tm5_get_dt_data()
987 adc_tm->avg_samples = VADC_DEF_AVG_SAMPLES; in adc_tm5_get_dt_data()
991 ret = adc_tm5_get_dt_channel_data(adc_tm, channels, child); in adc_tm5_get_dt_data()
997 channels++; in adc_tm5_get_dt_data()
1005 struct device_node *node = pdev->dev.of_node; in adc_tm5_probe()
1006 struct device *dev = &pdev->dev; in adc_tm5_probe()
1012 regmap = dev_get_regmap(dev->parent, NULL); in adc_tm5_probe()
1014 return -ENODEV; in adc_tm5_probe()
1020 adc_tm = devm_kzalloc(&pdev->dev, sizeof(*adc_tm), GFP_KERNEL); in adc_tm5_probe()
1022 return -ENOMEM; in adc_tm5_probe()
1024 adc_tm->regmap = regmap; in adc_tm5_probe()
1025 adc_tm->dev = dev; in adc_tm5_probe()
1026 adc_tm->base = reg; in adc_tm5_probe()
1038 ret = adc_tm->data->init(adc_tm); in adc_tm5_probe()
1040 dev_err(dev, "adc-tm init failed\n"); in adc_tm5_probe()
1050 return devm_request_threaded_irq(dev, irq, NULL, adc_tm->data->isr, in adc_tm5_probe()
1051 IRQF_ONESHOT, adc_tm->data->irq_name, adc_tm); in adc_tm5_probe()
1056 .compatible = "qcom,spmi-adc-tm5",
1060 .compatible = "qcom,spmi-adc-tm-hc",
1064 .compatible = "qcom,spmi-adc-tm5-gen2",
1073 .name = "qcom-spmi-adc-tm5",
1080 MODULE_DESCRIPTION("SPMI PMIC Thermal Monitor ADC driver");