Received: by 2002:a05:6a10:1d13:0:0:0:0 with SMTP id pp19csp878488pxb; Mon, 16 Aug 2021 21:14:19 -0700 (PDT) X-Google-Smtp-Source: ABdhPJz7Vd8mPH5PU+wAWtC5tUPxjrELqWbzw/R/RWS2DHWrQbprM1/6ReyX0RYqEyzT9q3tywqh X-Received: by 2002:a05:6602:48a:: with SMTP id y10mr1404443iov.130.1629173658998; Mon, 16 Aug 2021 21:14:18 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1629173658; cv=none; d=google.com; s=arc-20160816; b=I0UaFYqAR587Sye4fp9bBo5lU7+ReYO1zEgm7clCkFytxJ0jB4Iu0VrgqMmmMQwH68 Z6FHUb37H/pyAAFg6OtiiCGeE0+4kWQjGlOdgaN2mdf8bboj1+lswSIv8/OFyd2ga4uu 4N00ubDejcMeTncViSEhArVq2WgW8pHPelM0q7B6N+KV38o1ugA4TlSey7q6qlt12FC2 EM8AgaoFUT76zaOSatpaSplch3N5qvAcd0ybhESYg1HMWpyJzMl2P5vRZACzryntsxk5 pyPa/nDB0hh/l9aJscs73eMnud3gWFbsdmp8Mma+38cQjtC2/JT5yol0TqOodEuh9bz1 sc5w== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from; bh=FE0HmL26F6/a52Eje4wU2eKONjvVqvqV7OTYQbegMps=; b=RY6n4j0FaDjM8ze8HmaPKuJ9AI+rrii+naiXBf7U3K2GE1yvwGKiJs5tXP1oexDhmv xM0Gxz5SGRxuhKtv9zQshVfVCvhMs0f5mqfskTwY3lsWomisgj3tlEZqutFfEkIgw0mb Uvs8ceLh3xITsoRV7TPOByOzjZMS+g3Cn48gFKP62AOaVodOrn+49HlI5B7CQm0tULdl ks8kHHZ3X6moKq2bIMI/+5pmkTEFmcsJYZGneKhyz+SviJE1jQfWn9UBsGFYuYESx5/n HGMstRH27CcZpTg6FDx/3W7CWMbTWpiymbRFv073OzdE6xwFoEQGuUKv9w6hA0B+KsY4 +3IQ== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id e123si1249839iof.100.2021.08.16.21.14.07; Mon, 16 Aug 2021 21:14:18 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S237438AbhHQENy (ORCPT + 99 others); Tue, 17 Aug 2021 00:13:54 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:56412 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229477AbhHQENw (ORCPT ); Tue, 17 Aug 2021 00:13:52 -0400 Received: from metis.ext.pengutronix.de (metis.ext.pengutronix.de [IPv6:2001:67c:670:201:290:27ff:fe1d:cc33]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id E5D7AC0613A3 for ; Mon, 16 Aug 2021 21:13:19 -0700 (PDT) Received: from dude.hi.pengutronix.de ([2001:67c:670:100:1d::7]) by metis.ext.pengutronix.de with esmtps (TLS1.3:ECDHE_RSA_AES_256_GCM_SHA384:256) (Exim 4.92) (envelope-from ) id 1mFqTN-0003Bb-S1; Tue, 17 Aug 2021 06:13:09 +0200 Received: from ore by dude.hi.pengutronix.de with local (Exim 4.92) (envelope-from ) id 1mFqTM-0006ZY-8h; Tue, 17 Aug 2021 06:13:08 +0200 From: Oleksij Rempel To: Wolfgang Grandegger , Marc Kleine-Budde , "David S. Miller" , Jakub Kicinski , Rob Herring Cc: Oleksij Rempel , linux-can@vger.kernel.org, netdev@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, kernel@pengutronix.de, David Jander Subject: [PATCH v2 3/3] can: dev: provide optional GPIO based termination support Date: Tue, 17 Aug 2021 06:13:06 +0200 Message-Id: <20210817041306.25185-4-o.rempel@pengutronix.de> X-Mailer: git-send-email 2.30.2 In-Reply-To: <20210817041306.25185-1-o.rempel@pengutronix.de> References: <20210817041306.25185-1-o.rempel@pengutronix.de> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-SA-Exim-Connect-IP: 2001:67c:670:100:1d::7 X-SA-Exim-Mail-From: ore@pengutronix.de X-SA-Exim-Scanned: No (on metis.ext.pengutronix.de); SAEximRunCond expanded to false X-PTX-Original-Recipient: linux-kernel@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org For CAN buses to work, a termination resistor has to be present at both ends of the bus. This resistor is usually 120 Ohms, other values may be required for special bus topologies. This patch adds support for a generic GPIO based CAN termination. The resistor value has to be specified via device tree, and it can only be attached to or detached from the bus. By default the termination is not active. Signed-off-by: Oleksij Rempel --- drivers/net/can/dev/dev.c | 54 +++++++++++++++++++++++++++++++++++++++ include/linux/can/dev.h | 8 ++++++ 2 files changed, 62 insertions(+) diff --git a/drivers/net/can/dev/dev.c b/drivers/net/can/dev/dev.c index 311d8564d611..b4a6c7a6fc18 100644 --- a/drivers/net/can/dev/dev.c +++ b/drivers/net/can/dev/dev.c @@ -15,6 +15,7 @@ #include #include #include +#include #include #define MOD_DESC "CAN device driver interface" @@ -400,10 +401,57 @@ void close_candev(struct net_device *dev) } EXPORT_SYMBOL_GPL(close_candev); +static int can_set_termination(struct net_device *ndev, u16 term) +{ + struct can_priv *priv = netdev_priv(ndev); + int set; + + if (term == priv->termination_gpio_ohms[CAN_TERMINATION_GPIO_ENABLED]) + set = 1; + else + set = 0; + + gpiod_set_value(priv->termination_gpio, set); + + return 0; +} + +static int can_get_termination(struct net_device *ndev) +{ + struct can_priv *priv = netdev_priv(ndev); + struct device *dev = ndev->dev.parent; + struct gpio_desc *gpio; + u16 term; + int ret; + + /* Disabling termination by default is the safe choice: Else if many + * bus participants enable it, no communication is possible at all. + */ + gpio = devm_gpiod_get_optional(dev, "termination", GPIOD_OUT_LOW); + if (IS_ERR(gpio)) + return dev_err_probe(dev, PTR_ERR(gpio), + "Cannot get termination-gpios\n"); + + ret = device_property_read_u16(dev, "termination-ohms", &term); + if (ret) + return ret; + + priv->termination_const_cnt = ARRAY_SIZE(priv->termination_gpio_ohms); + priv->termination_const = priv->termination_gpio_ohms; + priv->termination_gpio = gpio; + priv->termination_gpio_ohms[CAN_TERMINATION_GPIO_DISABLED] = + CAN_TERMINATION_DISABLED; + priv->termination_gpio_ohms[CAN_TERMINATION_GPIO_ENABLED] = term; + priv->do_set_termination = can_set_termination; + + return 0; +} + /* Register the CAN network device */ int register_candev(struct net_device *dev) { struct can_priv *priv = netdev_priv(dev); + int err; /* Ensure termination_const, termination_const_cnt and * do_set_termination consistency. All must be either set or @@ -419,6 +467,12 @@ int register_candev(struct net_device *dev) if (!priv->data_bitrate_const != !priv->data_bitrate_const_cnt) return -EINVAL; + if (!priv->termination_const) { + err = can_get_termination(dev); + if (err) + return err; + } + dev->rtnl_link_ops = &can_link_ops; netif_carrier_off(dev); diff --git a/include/linux/can/dev.h b/include/linux/can/dev.h index 27b275e463da..2413253e54c7 100644 --- a/include/linux/can/dev.h +++ b/include/linux/can/dev.h @@ -32,6 +32,12 @@ enum can_mode { CAN_MODE_SLEEP }; +enum can_termination_gpio { + CAN_TERMINATION_GPIO_DISABLED = 0, + CAN_TERMINATION_GPIO_ENABLED, + CAN_TERMINATION_GPIO_MAX, +}; + /* * CAN common private data */ @@ -55,6 +61,8 @@ struct can_priv { unsigned int termination_const_cnt; const u16 *termination_const; u16 termination; + struct gpio_desc *termination_gpio; + u16 termination_gpio_ohms[CAN_TERMINATION_GPIO_MAX]; enum can_state state; -- 2.30.2