Received: by 2002:ad5:474a:0:0:0:0:0 with SMTP id i10csp1491069imu; Wed, 28 Nov 2018 10:17:27 -0800 (PST) X-Google-Smtp-Source: AFSGD/VMD1BS2h2tXfLzqHEwkvWjSlUrMj0odtRM6D0P9kQ8L1gpE2HFBZzZSA+5nw/KtVYQq/qE X-Received: by 2002:a17:902:5588:: with SMTP id g8mr22301233pli.22.1543429047194; Wed, 28 Nov 2018 10:17:27 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1543429047; cv=none; d=google.com; s=arc-20160816; b=j9bV6Pr9lPjfZnbUOAp6f5Sd69aP3P4cIiD9XJS71WKyaq9N6Xhfv0O2AJmDxOYsiC kjsWKMVsoz4cBw0L241AGBAs1mmKHGfaiKnZqTMaJ8S6OP1snQOl59WdAelpMV8DIE4r kc/VFU94tKmbskGpaIzk8aC2JBPknOwdIT6F/BLRXsESJIKJBpDqonrV/NKjteohPhMC p/lsPYu9I1RnXTv+ftMO7paJ7aCLms5/l7jv45PlJ4pOnik30QB8HnTP9BNH3mItJbhm kHiRsVWDopG72Vf7hHzBvaOjEP7hCvb9KU9drF+mWtFK9w5W9kt/lApdNMWfVuDjlnAG 2pdg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:user-agent:in-reply-to :content-disposition:mime-version:references:message-id:subject:cc :to:date:from:dkim-signature; bh=iP89O5yt8cPK3Ietczi17UITaQMICxETiZ1FFCX7LK4=; b=XrDsMcK7BrpHrQUjcVSAm1oPmdOnu457hGH8qb2y9Oo7G+QsSZMSN5S2kvPMrsfAve NkqRIuEsQv0tnm4wcxIB42z7NB5BkYlatPSRa4vG0IkVSMIFSU8gk3w+0wqZLNQllnh7 LmulmaykAAA/GljBN5ADRrK/gG3BmPhs2RT1pKfszXJepBe4f/T/VNptZiGt7MmbAybY +C9e6bJ8z4xZu2+0PDLq1jrLKxTivM0pk33cttDXBD4a6IZlxr2o/lg580N2BdHY7IZx PD2fhXBfwhdkELcfgDNNybh99DG8KiCPCA+Z5CU+tpvDn6WJFfsx6eNzl89ISZAqG5L4 uC9A== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=mFQy2hpk; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id x74si8996926pfe.23.2018.11.28.10.17.04; Wed, 28 Nov 2018 10:17:27 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=mFQy2hpk; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728498AbeK2FSh (ORCPT + 99 others); Thu, 29 Nov 2018 00:18:37 -0500 Received: from mail-qt1-f193.google.com ([209.85.160.193]:38734 "EHLO mail-qt1-f193.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1725266AbeK2FSh (ORCPT ); Thu, 29 Nov 2018 00:18:37 -0500 Received: by mail-qt1-f193.google.com with SMTP id p17so26919777qtl.5; Wed, 28 Nov 2018 10:16:05 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=from:date:to:cc:subject:message-id:references:mime-version :content-disposition:in-reply-to:user-agent; bh=iP89O5yt8cPK3Ietczi17UITaQMICxETiZ1FFCX7LK4=; b=mFQy2hpk1BwpKhVnSVD0ZRV2/kKz4U/lxtD1dpIyAmGI3HeZ3jmFxB8QauGEDhNlpZ PggN+ejiBId02wwSNC2He+568sk/40WB4aMZlXgn/w56DdK9jbHns/q04FOamo5RBe9P LzkR3xixdPRXnctjrVRkVlBNOSGSp1qxT98EQOp2/L1g4kl2SfWyzzaE8+rIJsHZigPM 7ar0ijxmOwaLiF0MwVRtRylTdEpNgmHR5ht6SAN1MwzaUBkIZXKbO0bwPpwoIpPLhMHo SjE3bp9321rJB02FVwkNMcsN2Dn7+lDg41k+/KbKO1f1PcecME9hEMUvKp+mj9ruL4RK NabQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:date:to:cc:subject:message-id:references :mime-version:content-disposition:in-reply-to:user-agent; bh=iP89O5yt8cPK3Ietczi17UITaQMICxETiZ1FFCX7LK4=; b=iK9Bf6MSrTy4vwOYHXmOrxjPhW9v9sHON7FvBxPETb+pEvbNDtC/za5et1fgJ8hW4c b5DnXx9GTwNMrRVOy6kTx+M4HamMGiGsM6NwnIrmGtFXGGxTmvHqPg+IZcO8fRR0zNgi 2etJU4cM1hDOhJ6YfW1LOgpzhlbE5l91ScRjztwP5ZQdcc/XTfkzIQXRQTIBnev4XXpC /ZDaz/Hp16MYaYVKP086S8x2UaoTBawLnfS+rGlyFZfR6GF7O93vFJQQqkmW3zH5OMz2 iUkaomsGov3Qj2kd/3sDKrZIhTCGem473eaLS6eoWrv1dOid39yAK9glOynhnMoQ1/mI KAfg== X-Gm-Message-State: AGRZ1gLD9/JaewalPSAJKDXfuKqKtbx5ikVz1MjBsCgn+ArYGF78sQey K1YBviOXILmz7Q0So7utpmA= X-Received: by 2002:ac8:3518:: with SMTP id y24mr36002794qtb.241.1543428965393; Wed, 28 Nov 2018 10:16:05 -0800 (PST) Received: from smtp.gmail.com ([143.107.45.1]) by smtp.gmail.com with ESMTPSA id k185sm3613156qkd.62.2018.11.28.10.16.01 (version=TLS1_2 cipher=ECDHE-RSA-CHACHA20-POLY1305 bits=256/256); Wed, 28 Nov 2018 10:16:04 -0800 (PST) From: Giuliano Belinassi X-Google-Original-From: Giuliano Belinassi Date: Wed, 28 Nov 2018 16:15:59 -0200 To: lars@metafoo.de, Michael.Hennerich@analog.com, jic23@kernel.org, knaack.h@gmx.de, pmeerw@pmeerw.net, gregkh@linuxfoundation.org, stefan.popa@analog.com, alexandru.Ardelean@analog.com, renatogeh@gmail.com Cc: linux-iio@vger.kernel.org, devel@driverdev.osuosl.org, linux-kernel@vger.kernel.org, kernel-usp@googlegroups.com Subject: [PATCH 1/2] staging: iio: ad7780: Add gain & filter gpio support Message-ID: <231cbc909b892a0ebe172460d34e9eeb9cfa3003.1543428366.git.giuliano.belinassi@usp.br> References: MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: User-Agent: NeoMutt/20180716 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Previously, the AD7780 driver only supported gpio for the 'powerdown' pin. This commit adds suppport for the 'gain' and 'filter' pin. Signed-off-by: Giuliano Belinassi --- Changes in v2: - Now this patch is part of the patchset that aims to remove ad7780 out of staging. https://marc.info/?l=linux-iio&m=154282349808890&w=2 - Also, now it reads voltage and filter values from the userspace instead of gpio pin states. drivers/staging/iio/adc/ad7780.c | 78 ++++++++++++++++++++++++-- include/linux/iio/adc/ad_sigma_delta.h | 5 ++ 2 files changed, 79 insertions(+), 4 deletions(-) diff --git a/drivers/staging/iio/adc/ad7780.c b/drivers/staging/iio/adc/ad7780.c index c4a85789c2db..05979a79fda3 100644 --- a/drivers/staging/iio/adc/ad7780.c +++ b/drivers/staging/iio/adc/ad7780.c @@ -39,6 +39,12 @@ #define AD7170_PATTERN (AD7780_PAT0 | AD7170_PAT2) #define AD7170_PATTERN_MASK (AD7780_PAT0 | AD7780_PAT1 | AD7170_PAT2) +#define AD7780_GAIN_GPIO 0 +#define AD7780_FILTER_GPIO 1 + +#define AD7780_GAIN_MIDPOINT 64 +#define AD7780_FILTER_MIDPOINT 13350 + struct ad7780_chip_info { struct iio_chan_spec channel; unsigned int pattern_mask; @@ -50,6 +56,8 @@ struct ad7780_state { const struct ad7780_chip_info *chip_info; struct regulator *reg; struct gpio_desc *powerdown_gpio; + struct gpio_desc *gain_gpio; + struct gpio_desc *filter_gpio; unsigned int gain; struct ad_sigma_delta sd; @@ -115,18 +123,65 @@ static int ad7780_read_raw(struct iio_dev *indio_dev, return -EINVAL; } +static int ad7780_write_raw(struct iio_dev *indio_dev, + struct iio_chan_spec const *chan, + int val, + int val2, + long m) +{ + struct ad7780_state *st = iio_priv(indio_dev); + const struct ad7780_chip_info *chip_info = st->chip_info; + int uvref, gain; + unsigned int full_scale; + + if (!chip_info->is_ad778x) + return 0; + + switch (m) { + case IIO_CHAN_INFO_SCALE: + if (val != 0) + return -EINVAL; + + uvref = regulator_get_voltage(st->reg); + + if (uvref < 0) + return uvref; + + full_scale = 1 << (chip_info->channel.scan_type.realbits - 1); + gain = DIV_ROUND_CLOSEST(uvref, full_scale); + gain = DIV_ROUND_CLOSEST(gain, val2); + + gpiod_set_value(st->gain_gpio, gain < AD7780_GAIN_MIDPOINT ? 0 : 1); + break; + case IIO_CHAN_INFO_SAMP_FREQ: + if (val2 != 0) + return -EINVAL; + + gpiod_set_value(st->filter_gpio, val < AD7780_FILTER_MIDPOINT ? 0 : 1); + break; + } + + return 0; +} + static int ad7780_postprocess_sample(struct ad_sigma_delta *sigma_delta, unsigned int raw_sample) { struct ad7780_state *st = ad_sigma_delta_to_ad7780(sigma_delta); const struct ad7780_chip_info *chip_info = st->chip_info; + int val; if ((raw_sample & AD7780_ERR) || ((raw_sample & chip_info->pattern_mask) != chip_info->pattern)) return -EIO; if (chip_info->is_ad778x) { - if (raw_sample & AD7780_GAIN) + val = raw_sample & AD7780_GAIN; + + if (val != gpiod_get_value(st->gain_gpio)) + return -EIO; + + if (val) st->gain = 1; else st->gain = 128; @@ -141,18 +196,20 @@ static const struct ad_sigma_delta_info ad7780_sigma_delta_info = { .has_registers = false, }; -#define AD7780_CHANNEL(bits, wordsize) \ +#define AD7170_CHANNEL(bits, wordsize) \ AD_SD_CHANNEL_NO_SAMP_FREQ(1, 0, 0, bits, 32, wordsize - bits) +#define AD7780_CHANNEL(bits, wordsize) \ + AD_SD_CHANNEL_GAIN_FILTER(1, 0, 0, bits, 32, wordsize - bits) static const struct ad7780_chip_info ad7780_chip_info_tbl[] = { [ID_AD7170] = { - .channel = AD7780_CHANNEL(12, 24), + .channel = AD7170_CHANNEL(12, 24), .pattern = AD7170_PATTERN, .pattern_mask = AD7170_PATTERN_MASK, .is_ad778x = false, }, [ID_AD7171] = { - .channel = AD7780_CHANNEL(16, 24), + .channel = AD7170_CHANNEL(16, 24), .pattern = AD7170_PATTERN, .pattern_mask = AD7170_PATTERN_MASK, .is_ad778x = false, @@ -173,6 +230,7 @@ static const struct ad7780_chip_info ad7780_chip_info_tbl[] = { static const struct iio_info ad7780_info = { .read_raw = ad7780_read_raw, + .write_raw = ad7780_write_raw, }; static int ad7780_probe(struct spi_device *spi) @@ -222,6 +280,18 @@ static int ad7780_probe(struct spi_device *spi) goto error_disable_reg; } + if (st->chip_info->is_ad778x) { + st->gain_gpio = devm_gpiod_get_optional(&spi->dev, + "gain", + GPIOD_OUT_HIGH); + if (IS_ERR(st->gain_gpio)) { + ret = PTR_ERR(st->gain_gpio); + dev_err(&spi->dev, "Failed to request gain GPIO: %d\n", + ret); + goto error_disable_reg; + } + } + ret = ad_sd_setup_buffer_and_trigger(indio_dev); if (ret) goto error_disable_reg; diff --git a/include/linux/iio/adc/ad_sigma_delta.h b/include/linux/iio/adc/ad_sigma_delta.h index 730ead1a46df..6cadab6fd5fd 100644 --- a/include/linux/iio/adc/ad_sigma_delta.h +++ b/include/linux/iio/adc/ad_sigma_delta.h @@ -173,6 +173,11 @@ int ad_sd_validate_trigger(struct iio_dev *indio_dev, struct iio_trigger *trig); __AD_SD_CHANNEL(_si, _channel, -1, _address, _bits, \ _storagebits, _shift, NULL, IIO_VOLTAGE, 0) +#define AD_SD_CHANNEL_GAIN_FILTER(_si, _channel, _address, _bits, \ + _storagebits, _shift) \ + __AD_SD_CHANNEL(_si, _channel, -1, _address, _bits, \ + _storagebits, _shift, NULL, IIO_VOLTAGE, BIT(IIO_CHAN_INFO_RAW)) + #define AD_SD_TEMP_CHANNEL(_si, _address, _bits, _storagebits, _shift) \ __AD_SD_CHANNEL(_si, 0, -1, _address, _bits, \ _storagebits, _shift, NULL, IIO_TEMP, \ -- 2.19.1