Received: by 2002:a05:7412:419a:b0:f3:1519:9f41 with SMTP id i26csp231187rdh; Thu, 23 Nov 2023 02:20:49 -0800 (PST) X-Google-Smtp-Source: AGHT+IHYx8Zc59Vf7g8PBMt5qISEimwagPwlCVGZRpJ/r0d21pXa7ubpUiSHDzid3YQ8QR4okRqb X-Received: by 2002:a05:6a21:9808:b0:18b:208b:7043 with SMTP id ue8-20020a056a21980800b0018b208b7043mr4587987pzb.49.1700734849336; Thu, 23 Nov 2023 02:20:49 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1700734849; cv=none; d=google.com; s=arc-20160816; b=zqLSj5MtoS+UPoAY39mOL5gmiLg2cIvG0FT0E3taQHUrXzL6Fy7BxCTh7Iloroq53f 5ORH+C81pGz+7IEEuSVPXIJue2mCvBiTuIIyklv5Ptpy53LP25ClC3xYfjKutI3KEw+z TeUCZ5A0JvdH6H4SPQVcbm6cy+kkuYgN4PNGuQ2UStRno0eQnqSougMuDk9rXmWOqTaO vnhqSw4dkiP2+D7Sq5uNo2cXtfACFZ3S+dLU6lOt2zLkQmU44/9DlUhk/z/D6mBgiklU uL8xyB6C4a4F8IXjNdzBFBlIxByyZBYR02C+2aORCJnj6KWY3j8g3KMql5Cks6b/vI52 B1Xg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:cc:to:subject:message-id:date:from:in-reply-to :references:mime-version:dkim-signature; bh=zgcr8M5KTV6qx0UUZCDKtyAM/hgXXxP53Wb8ozrdskQ=; fh=2n1mh+8TlAzVQYh9Ymvj+R/yzCoxzYkkZ/8qpchvbfw=; b=pxiVuQ0MiZEpkjhGAjvNeILuDwOPbnDIPkr4Od105nCZMewSTaVq2Vt/7adR1JeDMO 5Vzisxodh7Sj2Y59C8/+5UL472V2QVQ4MX4QDxq67YLj0+Lq/bWCdXZSiOM55ygMmk5l PgUlUB2ufeTPpk0GoGhtywvvVFlYRTFy5AVWHubU9VCmF+tynoyMBBlYU2AkRW3cXnlN xvQzn1UPbEyQDtS+uJ6Dskh+TwnAFOE0ZBdHRzEdNb0M1muu6sHDOnfPf47k9iE2Ktv8 EiLLxyksD79mwgwOz1SjsmRJWI7OhogNEjIDiqkwwoAXetEtuZALjXFmVXCJa6+fXQ4s A14Q== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@melexis.com header.s=google header.b=S88a+TLT; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::3:4 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=melexis.com Return-Path: Received: from howler.vger.email (howler.vger.email. [2620:137:e000::3:4]) by mx.google.com with ESMTPS id d11-20020a631d4b000000b005bdbf2ac2d2si1019286pgm.85.2023.11.23.02.20.48 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Thu, 23 Nov 2023 02:20:49 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::3:4 as permitted sender) client-ip=2620:137:e000::3:4; Authentication-Results: mx.google.com; dkim=pass header.i=@melexis.com header.s=google header.b=S88a+TLT; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::3:4 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=melexis.com Received: from out1.vger.email (depot.vger.email [IPv6:2620:137:e000::3:0]) by howler.vger.email (Postfix) with ESMTP id 7B7148072A09; Thu, 23 Nov 2023 02:20:45 -0800 (PST) X-Virus-Status: Clean X-Virus-Scanned: clamav-milter 0.103.11 at howler.vger.email Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S234620AbjKWKUZ (ORCPT + 99 others); Thu, 23 Nov 2023 05:20:25 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:58286 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229844AbjKWKUW (ORCPT ); Thu, 23 Nov 2023 05:20:22 -0500 Received: from mail-oa1-x36.google.com (mail-oa1-x36.google.com [IPv6:2001:4860:4864:20::36]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id DDF111BD for ; Thu, 23 Nov 2023 02:20:27 -0800 (PST) Received: by mail-oa1-x36.google.com with SMTP id 586e51a60fabf-1f066fc2a28so462900fac.0 for ; Thu, 23 Nov 2023 02:20:27 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=melexis.com; s=google; t=1700734827; x=1701339627; darn=vger.kernel.org; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:from:to:cc:subject:date:message-id:reply-to; bh=zgcr8M5KTV6qx0UUZCDKtyAM/hgXXxP53Wb8ozrdskQ=; b=S88a+TLTEAN3+uVnj+ORn8bossWXwevsFZPIgqMiHjVA6Xi3vQGY1qIoFHeu7B/b/7 okYiYedwRnV5BThEQ4vDkZ4uCcaKf0yzCZ/x5CCNUi7unMJBBbCBIpanBq9s0WeHYswz d9jlWTggp+cwFEMsle7Up3mAL1qEs5qfmPyJzg6eFV1kGF8DTeYDgv/lOXFCjmnHxNKO kHl11F0LQb6wdMbOmipXzCy/BfihSu1oqawTzbNZGFEu1JwrQwP8dDNb8VedM/gAS1Ua CFqnzXedVfZxKQAvSBBl5JgZcl+1RIO3uXi4n5qeouidHpPyk06zOmUklD3IAlBfTSE8 xq7w== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1700734827; x=1701339627; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:x-gm-message-state:from:to:cc:subject:date:message-id :reply-to; bh=zgcr8M5KTV6qx0UUZCDKtyAM/hgXXxP53Wb8ozrdskQ=; b=CxxhzRnQ2wbm2kEuv11H3KHQSXoHneSlcl1blNzQ12jpVTmEW8dEuzRnbr8D0xsV5Y /MuVCOITnntt4tR77xuumq2NruG4bP5D3lTDIsB+vH7jhPazZhwEXq0yK+MZlc0zwfMP HgGYF0MLiANhjJFHcD4w6nJDHCj8YjvdD/8/dsiGmzAztLDK/humD/8ExpmRPk8+sX+p 7B5kuD32ml3Cb25r1m8gd9YN+RGE22UWj3Trz4qy0Q//fDSsCNgBqkP3vi9a8TMj4+Vm K6UaPTzZKwFJasoZiSmsWz4amihsdjepgulA+cDP3RG+7k2dJsKaPMxKHhopZ2tYOrXq LP4A== X-Gm-Message-State: AOJu0Ywecrwl1L74lcLAslqMxB9AWj7J2YoBcbAomEb6XFVBx/QJoN7c 672LqBDQyKldfpXx+Ghj2JlJl2AD3HSy5U8PmY60ax3tJpHTukwllIZ1VA== X-Received: by 2002:a05:6870:7903:b0:1f0:36b6:ef25 with SMTP id hg3-20020a056870790300b001f036b6ef25mr7026131oab.23.1700734827146; Thu, 23 Nov 2023 02:20:27 -0800 (PST) MIME-Version: 1.0 References: <20231123094357.222371-1-kimseer.paller@analog.com> <20231123094357.222371-2-kimseer.paller@analog.com> In-Reply-To: <20231123094357.222371-2-kimseer.paller@analog.com> From: Crt Mori Date: Thu, 23 Nov 2023 11:19:51 +0100 Message-ID: Subject: Re: [PATCH v4 2/2] iio: frequency: admfm2000: New driver To: Kim Seer Paller Cc: Jonathan Cameron , Lars-Peter Clausen , Michael Hennerich , Rob Herring , Krzysztof Kozlowski , Conor Dooley , linux-iio@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Content-Type: text/plain; charset="UTF-8" X-Spam-Status: No, score=-0.9 required=5.0 tests=DKIM_SIGNED,DKIM_VALID, DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS,MAILING_LIST_MULTI, SPF_HELO_NONE,SPF_PASS,T_SCC_BODY_TEXT_LINE autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on howler.vger.email Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org X-Greylist: Sender passed SPF test, not delayed by milter-greylist-4.6.4 (howler.vger.email [0.0.0.0]); Thu, 23 Nov 2023 02:20:45 -0800 (PST) Hi, Just minor remark inline. Best regards, Crt On Thu, 23 Nov 2023 at 10:44, Kim Seer Paller wrote: > > Dual microwave down converter module with input RF and LO frequency > ranges from 0.5 to 32 GHz and an output IF frequency range from 0.1 to > 8 GHz. It consists of a LNA, mixer, IF filter, DSA, and IF amplifier > for each down conversion path. > > Signed-off-by: Kim Seer Paller > --- > V1 -> V4: No changes. > > MAINTAINERS | 1 + > drivers/iio/frequency/Kconfig | 10 + > drivers/iio/frequency/Makefile | 1 + > drivers/iio/frequency/admfm2000.c | 309 ++++++++++++++++++++++++++++++ > 4 files changed, 321 insertions(+) > create mode 100644 drivers/iio/frequency/admfm2000.c > > diff --git a/MAINTAINERS b/MAINTAINERS > index f1692ec68..d8630e490 100644 > --- a/MAINTAINERS > +++ b/MAINTAINERS > @@ -1253,6 +1253,7 @@ L: linux-iio@vger.kernel.org > S: Supported > W: https://ez.analog.com/linux-software-drivers > F: Documentation/devicetree/bindings/iio/frequency/adi,admfm2000.yaml > +F: drivers/iio/frequency/admfm2000.c > > ANALOG DEVICES INC ADMV1013 DRIVER > M: Antoniu Miclaus > diff --git a/drivers/iio/frequency/Kconfig b/drivers/iio/frequency/Kconfig > index 9e85dfa58..c455be7d4 100644 > --- a/drivers/iio/frequency/Kconfig > +++ b/drivers/iio/frequency/Kconfig > @@ -60,6 +60,16 @@ config ADF4377 > To compile this driver as a module, choose M here: the > module will be called adf4377. > > +config ADMFM2000 > + tristate "Analog Devices ADMFM2000 Dual Microwave Down Converter" > + depends on GPIOLIB > + help > + Say yes here to build support for Analog Devices ADMFM2000 Dual > + Microwave Down Converter. > + > + To compile this driver as a module, choose M here: the > + module will be called admfm2000. > + > config ADMV1013 > tristate "Analog Devices ADMV1013 Microwave Upconverter" > depends on SPI && COMMON_CLK > diff --git a/drivers/iio/frequency/Makefile b/drivers/iio/frequency/Makefile > index b616c29b4..70d0e0b70 100644 > --- a/drivers/iio/frequency/Makefile > +++ b/drivers/iio/frequency/Makefile > @@ -8,6 +8,7 @@ obj-$(CONFIG_AD9523) += ad9523.o > obj-$(CONFIG_ADF4350) += adf4350.o > obj-$(CONFIG_ADF4371) += adf4371.o > obj-$(CONFIG_ADF4377) += adf4377.o > +obj-$(CONFIG_ADMFM2000) += admfm2000.o > obj-$(CONFIG_ADMV1013) += admv1013.o > obj-$(CONFIG_ADMV1014) += admv1014.o > obj-$(CONFIG_ADMV4420) += admv4420.o > diff --git a/drivers/iio/frequency/admfm2000.c b/drivers/iio/frequency/admfm2000.c > new file mode 100644 > index 000000000..e0b5edce7 > --- /dev/null > +++ b/drivers/iio/frequency/admfm2000.c > @@ -0,0 +1,309 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * ADMFM2000 Dual Microwave Down Converter > + * > + * Copyright 2023 Analog Devices Inc. > + */ > + > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +#define ADMFM2000_MIXER_MODE 0 > +#define ADMFM2000_DIRECT_IF_MODE 1 > +#define ADMF20000_DSA_GPIOS 5 > +#define ADMF20000_MODE_GPIOS 2 > +#define ADMF20000_MAX_GAIN 0 > +#define ADMF20000_MIN_GAIN -31000 > +#define ADMF20000_DEFAULT_GAIN -0x20 > + > +struct admfm2000_state { > + struct mutex lock; /* protect sensor state */ > + struct gpio_descs *sw_ch[2]; > + struct gpio_descs *dsa_gpios[2]; > + u32 gain[2]; > +}; > + > +static int admfm2000_mode(struct iio_dev *indio_dev, u32 reg, u32 mode) > +{ > + struct admfm2000_state *st = iio_priv(indio_dev); > + DECLARE_BITMAP(values, 2); > + > + switch (mode) { > + case ADMFM2000_MIXER_MODE: > + values[0] = (reg == 0) ? 1 : 2; > + gpiod_set_array_value_cansleep(st->sw_ch[reg]->ndescs, > + st->sw_ch[reg]->desc, > + NULL, values); > + break; > + case ADMFM2000_DIRECT_IF_MODE: > + values[0] = (reg == 0) ? 2 : 1; > + gpiod_set_array_value_cansleep(st->sw_ch[reg]->ndescs, > + st->sw_ch[reg]->desc, > + NULL, values); > + break; > + default: > + return -EINVAL; > + } > + > + return 0; > +} > + > +static int admfm2000_attenuation(struct iio_dev *indio_dev, u32 chan, > + u32 value) > +{ > + struct admfm2000_state *st = iio_priv(indio_dev); > + DECLARE_BITMAP(values, BITS_PER_TYPE(value)); > + > + values[0] = value; > + > + gpiod_set_array_value_cansleep(st->dsa_gpios[chan]->ndescs, > + st->dsa_gpios[chan]->desc, > + NULL, values); > + return 0; > +} > + > +static int admfm2000_read_raw(struct iio_dev *indio_dev, > + struct iio_chan_spec const *chan, int *val, > + int *val2, long mask) > +{ > + struct admfm2000_state *st = iio_priv(indio_dev); > + int gain; > + > + switch (mask) { > + case IIO_CHAN_INFO_HARDWAREGAIN: > + mutex_lock(&st->lock); > + gain = ~(st->gain[chan->channel]) * -1000; > + *val = gain / 1000; > + *val2 = (gain % 1000) * 1000; > + mutex_unlock(&st->lock); > + > + return IIO_VAL_INT_PLUS_MICRO_DB; > + default: > + return -EINVAL; > + } > +} > + > +static int admfm2000_write_raw(struct iio_dev *indio_dev, > + struct iio_chan_spec const *chan, int val, > + int val2, long mask) > +{ > + struct admfm2000_state *st = iio_priv(indio_dev); > + int gain, ret; > + > + if (val < 0) > + gain = (val * 1000) - (val2 / 1000); > + else > + gain = (val * 1000) + (val2 / 1000); > + > + if (gain > ADMF20000_MAX_GAIN || gain < ADMF20000_MIN_GAIN) > + return -EINVAL; > + > + switch (mask) { > + case IIO_CHAN_INFO_HARDWAREGAIN: > + mutex_lock(&st->lock); > + st->gain[chan->channel] = ~((abs(gain) / 1000) & 0x1F); > + > + ret = admfm2000_attenuation(indio_dev, chan->channel, > + st->gain[chan->channel]); > + > + mutex_unlock(&st->lock); > + if (ret) > + return ret; > + break; > + default: > + return -EINVAL; > + } > + > + return 0; > +} > + > +static int admfm2000_write_raw_get_fmt(struct iio_dev *indio_dev, > + struct iio_chan_spec const *chan, > + long mask) > +{ > + switch (mask) { > + case IIO_CHAN_INFO_HARDWAREGAIN: > + return IIO_VAL_INT_PLUS_MICRO_DB; > + default: > + return -EINVAL; > + } > +} > + > +static const struct iio_info admfm2000_info = { > + .read_raw = &admfm2000_read_raw, > + .write_raw = &admfm2000_write_raw, > + .write_raw_get_fmt = &admfm2000_write_raw_get_fmt, > +}; > + > +#define ADMFM2000_CHAN(_channel) { \ > + .type = IIO_VOLTAGE, \ > + .output = 1, \ > + .indexed = 1, \ > + .channel = _channel, \ > + .info_mask_separate = BIT(IIO_CHAN_INFO_HARDWAREGAIN), \ > +} > + > +static const struct iio_chan_spec admfm2000_channels[] = { > + ADMFM2000_CHAN(0), > + ADMFM2000_CHAN(1), > +}; > + > +static int admfm2000_channel_config(struct admfm2000_state *st, > + struct iio_dev *indio_dev) > +{ > + struct platform_device *pdev = to_platform_device(indio_dev->dev.parent); > + struct device *dev = &pdev->dev; > + struct fwnode_handle *child; > + u32 reg, mode; > + int ret; > + > + device_for_each_child_node(dev, child) { > + ret = fwnode_property_read_u32(child, "reg", ®); > + if (ret) { > + fwnode_handle_put(child); > + return dev_err_probe(dev, ret, > + "Failed to get reg property\n"); > + } > + > + if (reg >= indio_dev->num_channels) { > + fwnode_handle_put(child); > + return dev_err_probe(dev, -EINVAL, "reg bigger than: %d\n", > + indio_dev->num_channels); > + } > + > + ret = fwnode_property_read_u32(child, "adi,mode", &mode); > + if (ret) { > + fwnode_handle_put(child); > + return dev_err_probe(dev, ret, > + "Failed to get mode property\n"); > + } > + > + if (mode >= 2) { > + fwnode_handle_put(child); > + return dev_err_probe(dev, -EINVAL, "mode bigger than: 1\n"); > + } > + > + ret = admfm2000_mode(indio_dev, reg, mode); > + if (ret) { > + fwnode_handle_put(child); > + return ret; > + } > + } > + > + return 0; > +} > + > +static int admfm2000_setup(struct admfm2000_state *st, > + struct iio_dev *indio_dev) > +{ > + struct platform_device *pdev = to_platform_device(indio_dev->dev.parent); > + struct device *dev = &pdev->dev; > + > + st->sw_ch[0] = devm_gpiod_get_array(dev, "switch1", GPIOD_OUT_LOW); > + if (IS_ERR(st->sw_ch[0])) > + return dev_err_probe(dev, PTR_ERR(st->sw_ch[0]), > + "Failed to get gpios\n"); > + > + if (st->sw_ch[0]->ndescs != ADMF20000_MODE_GPIOS) { > + dev_err_probe(dev, -ENODEV, "%d GPIOs needed to operate\n", > + ADMF20000_MODE_GPIOS); > + return -ENODEV; > + } > + > + st->sw_ch[1] = devm_gpiod_get_array(dev, "switch2", GPIOD_OUT_LOW); > + if (IS_ERR(st->sw_ch[1])) > + return dev_err_probe(dev, PTR_ERR(st->sw_ch[1]), > + "Failed to get gpios\n"); > + > + if (st->sw_ch[1]->ndescs != ADMF20000_MODE_GPIOS) { > + dev_err_probe(dev, -ENODEV, "%d GPIOs needed to operate\n", > + ADMF20000_MODE_GPIOS); > + return -ENODEV; > + } > + > + st->dsa_gpios[0] = devm_gpiod_get_array(dev, "attenuation1", > + GPIOD_OUT_LOW); > + if (IS_ERR(st->dsa_gpios[0])) > + return dev_err_probe(dev, PTR_ERR(st->dsa_gpios[0]), > + "Failed to get gpios\n"); > + > + if (st->dsa_gpios[0]->ndescs != ADMF20000_DSA_GPIOS) { > + dev_err_probe(dev, -ENODEV, "%d GPIOs needed to operate\n", > + ADMF20000_DSA_GPIOS); > + return -ENODEV; > + } > + > + st->dsa_gpios[1] = devm_gpiod_get_array(dev, "attenuation2", > + GPIOD_OUT_LOW); > + if (IS_ERR(st->dsa_gpios[1])) > + return dev_err_probe(dev, PTR_ERR(st->dsa_gpios[1]), > + "Failed to get gpios\n"); > + > + if (st->dsa_gpios[1]->ndescs != ADMF20000_DSA_GPIOS) { > + dev_err_probe(dev, -ENODEV, "%d GPIOs needed to operate\n", > + ADMF20000_DSA_GPIOS); no return -ENODEV here? > + } > + > + return 0; > +} > + > +static int admfm2000_probe(struct platform_device *pdev) > +{ > + struct device *dev = &pdev->dev; > + struct iio_dev *indio_dev; > + struct admfm2000_state *st; > + int ret; Order these in reverse christmass tree like you did above. > + > + indio_dev = devm_iio_device_alloc(dev, sizeof(*st)); > + if (!indio_dev) > + return -ENOMEM; > + > + st = iio_priv(indio_dev); > + > + indio_dev->name = "admfm2000"; > + indio_dev->num_channels = ARRAY_SIZE(admfm2000_channels); > + indio_dev->channels = admfm2000_channels; > + indio_dev->info = &admfm2000_info; > + indio_dev->modes = INDIO_DIRECT_MODE; > + > + st->gain[0] = ADMF20000_DEFAULT_GAIN; > + st->gain[1] = ADMF20000_DEFAULT_GAIN; > + > + mutex_init(&st->lock); > + > + ret = admfm2000_setup(st, indio_dev); > + if (ret) > + return ret; > + > + ret = admfm2000_channel_config(st, indio_dev); > + if (ret) > + return ret; > + > + return devm_iio_device_register(dev, indio_dev); > +} > + > +static const struct of_device_id admfm2000_of_match[] = { > + { .compatible = "adi,admfm2000" }, > + { } > +}; > +MODULE_DEVICE_TABLE(of, admfm2000_of_match); > + > +static struct platform_driver admfm2000_driver = { > + .driver = { > + .name = "admfm2000", > + .of_match_table = admfm2000_of_match, > + }, > + .probe = admfm2000_probe, > +}; > +module_platform_driver(admfm2000_driver); > + > +MODULE_AUTHOR("Kim Seer Paller "); > +MODULE_DESCRIPTION("ADMFM2000 Dual Microwave Down Converter"); > +MODULE_LICENSE("GPL"); > -- > 2.34.1 > >