From patchwork Wed Aug 14 18:47:31 2024 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Mary Strodl X-Patchwork-Id: 13763812 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from kanga.kvack.org (kanga.kvack.org [205.233.56.17]) by smtp.lore.kernel.org (Postfix) with ESMTP id 20AECC3DA4A for ; Wed, 14 Aug 2024 18:47:51 +0000 (UTC) Received: by kanga.kvack.org (Postfix) id 4152E6B0083; Wed, 14 Aug 2024 14:47:48 -0400 (EDT) Received: by kanga.kvack.org (Postfix, from userid 40) id 3C3DB6B0085; Wed, 14 Aug 2024 14:47:48 -0400 (EDT) X-Delivered-To: int-list-linux-mm@kvack.org Received: by kanga.kvack.org (Postfix, from userid 63042) id 1F15B6B0088; Wed, 14 Aug 2024 14:47:48 -0400 (EDT) X-Delivered-To: linux-mm@kvack.org Received: from relay.hostedemail.com (smtprelay0013.hostedemail.com [216.40.44.13]) by kanga.kvack.org (Postfix) with ESMTP id E23E36B0085 for ; Wed, 14 Aug 2024 14:47:47 -0400 (EDT) Received: from smtpin10.hostedemail.com (a10.router.float.18 [10.200.18.1]) by unirelay01.hostedemail.com (Postfix) with ESMTP id 96B0A1C4C6F for ; Wed, 14 Aug 2024 18:47:47 +0000 (UTC) X-FDA: 82451734974.10.0593E36 Received: from greygoose-centos7.csh.rit.edu (greygoose-centos7.csh.rit.edu [129.21.49.170]) by imf14.hostedemail.com (Postfix) with ESMTP id C344E10001E for ; Wed, 14 Aug 2024 18:47:44 +0000 (UTC) Authentication-Results: imf14.hostedemail.com; dkim=pass header.d=csh.rit.edu header.s=mail header.b=q2jUGwrT; spf=pass (imf14.hostedemail.com: domain of mstrodl@csh.rit.edu designates 129.21.49.170 as permitted sender) smtp.mailfrom=mstrodl@csh.rit.edu; dmarc=pass (policy=none) header.from=csh.rit.edu ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=hostedemail.com; s=arc-20220608; t=1723661192; h=from:from:sender:reply-to:subject:subject:date:date: message-id:message-id:to:to:cc:cc:mime-version:mime-version: content-type:content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references:dkim-signature; bh=t6hu+6hPMo2OVR8UFbh36mJfNfH8Y1pRqLPKk3qvXkw=; b=u+8+mglWF8MYUbD5t5Egg8agmD1BazZZtLawuRDkId2ri97TVqSHARrYlCGsmE/dS1gIIz GI0PMFd0VmTLWqKLxATUY30YQl9FHvfSKA+nZEkFbw9NVU2jAHSFmLnmPZk4NAsFb9wdpK UnzeeeDW8OqeceBDscWRwxXpKtryN+Y= ARC-Seal: i=1; s=arc-20220608; d=hostedemail.com; t=1723661192; a=rsa-sha256; cv=none; b=UrpNhyEz1y69afNGy/OdcQo0UsbBF4wabFhMqckN37i/iDGuI+jhRpHs2zyH24tv4ZfYSh GT8FcGutudqKblTeKXNoUuHPfdIcbpwOUPv6apfDvZQjOqLrnu3rorEjxX4RI3J8YKvtsv gFbPSxEczQY9Yg4iSKJkySdTKlBEHJc= ARC-Authentication-Results: i=1; imf14.hostedemail.com; dkim=pass header.d=csh.rit.edu header.s=mail header.b=q2jUGwrT; spf=pass (imf14.hostedemail.com: domain of mstrodl@csh.rit.edu designates 129.21.49.170 as permitted sender) smtp.mailfrom=mstrodl@csh.rit.edu; dmarc=pass (policy=none) header.from=csh.rit.edu Received: from localhost (localhost [127.0.0.1]) by greygoose-centos7.csh.rit.edu (Postfix) with ESMTP id D8CDD414609E; Wed, 14 Aug 2024 14:47:43 -0400 (EDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=csh.rit.edu; h= content-transfer-encoding:mime-version:references:in-reply-to :x-mailer:message-id:date:date:subject:subject:from:from :received:received; s=mail; t=1723661263; x=1725475664; bh=mCBxd ka291k0ThxNZp/dZ4uorDpdfjJsGKKtbiX+TI8=; b=q2jUGwrTVtrTRnCasfTBd VTryCyRl25NkomL6RmKCZPYIlMQ/4ZXEAfGnqu79FA4DiLNho+zFFN6Z7EPzaukN iFAyRkfw03f2qeLPedGfE+IPizI/8DQmKsrUaRL7XWOWC4zJBarACkwfW8FIELou PqWgktRFhmyLqNNRCeL6aU= X-Virus-Scanned: amavisd-new at csh.rit.edu Received: from greygoose-centos7.csh.rit.edu ([127.0.0.1]) by localhost (mail.csh.rit.edu [127.0.0.1]) (amavisd-new, port 10026) with ESMTP id PL3JFN303fnb; Wed, 14 Aug 2024 14:47:43 -0400 (EDT) Received: from freedom.csh.rit.edu (freedom.csh.rit.edu [129.21.49.182]) by greygoose-centos7.csh.rit.edu (Postfix) with ESMTP id C920F45735E9; Wed, 14 Aug 2024 14:47:42 -0400 (EDT) From: Mary Strodl To: linux-kernel@vger.kernel.org Cc: akpm@linux-foundation.org, urezki@gmail.com, hch@infradead.org, linux-mm@kvack.org, lee@kernel.org, andi.shyti@kernel.org, linux-i2c@vger.kernel.org, s.hauer@pengutronix.de, christian.gmeiner@gmail.com, Mary Strodl Subject: [PATCH v4 2/2] i2c: Add Congatec CGEB I2C driver Date: Wed, 14 Aug 2024 14:47:31 -0400 Message-ID: <20240814184731.1310988-3-mstrodl@csh.rit.edu> X-Mailer: git-send-email 2.45.2 In-Reply-To: <20240814184731.1310988-1-mstrodl@csh.rit.edu> References: <20240814184731.1310988-1-mstrodl@csh.rit.edu> MIME-Version: 1.0 X-Stat-Signature: b7p1ewroh1ikqed6hhkonyy5s3t9akxf X-Rspamd-Queue-Id: C344E10001E X-Rspam-User: X-Rspamd-Server: rspam08 X-HE-Tag: 1723661264-165407 X-HE-Meta: U2FsdGVkX1+V+T9EnJHFEY8JVD8uIGpKYel7ddUlaS/cTGKHs8PNxRq7VzMDfFmpoLaCeixIk33hpZT332ORGuTOSOuNmh34ANxcDkUM/lHTNlKhnhv1oTwyxV8PVNpzQHycXetyCEA4ODHXA9OWxz+NR6/QgLD3azD2DBBAfLxFkB5WDzBrRwee4CBbv3P4uuItJWjWPwWbERFt7YJnwaF+KmZORiomyoDJe1MAthSTV2GMwNG9tf+DH02KwWnSduygEm6cmc0AEEN129KNEOzY/dpwjV2RgWsgtPL7QN0olCLtd9jDxB/dPY+SbZ0EdlWAwmcnDei1W1e4q6yxZ6vl/k8x82/jc6JgLV3U63wiqrdFEu2dgMECXe/KaIn4uKGWIejk/qfr5Hxx9Fxd951giAcvXWEyQ9GkoC93eAfO2JKD9NWIZlhaJ69wH/zfTu32VecyOQlgwqGit4uiZ9/Pc8hfyjPpUy/oJzKrsf0qJt3vXR6vl+tL9PhWMTvoHZ6Bm3b38pd9qxfl8/5SAAfDaEwV5iLfXULTx7Xs7LVPB4ANZjhYi52ptOR3OuYDiUXViNOE4eMbBQ/JSxtHlITQ23xa+OUBGlRb72AMK/cBkNssOU89bgu4nUB4tXJcPfJIhiedYZp7dwD1RPCCR859zxKavdReMxe2h6E7L7/Ml8y6WecYOkJafGw038frTtn8zOI858y+0g/k97raA1WFfYuBkX0ZmSlTbLBuBWNvMYP0gYLW3AWmUlQDNIcLuuG/QztThYAwVGGmeeyecD9fV1LJkF7kF3hUmsy3cPQ4h7JT6QtkuQApakBK2bmA9uF7V81Yc0fzgwpZ4CEub9jApy35K12LRu2bHt85asDIS9Ar0QQNZMJeYP2eygMAUlXMmNE+TGHtsj+FMwlFOiox4c2jXTL09LTlWjjyOXIoFiiBJdPkqrQQYs9D4fbTndOMyNmS9fusTgxTsFx 2pJwgRL7 nJx1vHh4Z6aTAIPKI1/3wQmbCbZerInEaHNmc1R4Pf19B7SxobD5qYUgTRY2mUbz+QTDvxCnkIReT/PRKC2/gcK1MXVpfjS/9BWKxn3nxN0n4qI3VKaxrrKsWUjjwtziqqhOdFrhKajl/EuctxubtCW7IFa4hWeCsh/pbqFnPKibLc2TxhtrOOzugm/+9YXk/12Xda+i2x0wA6cu6k7RUNFUZVg== X-Bogosity: Ham, tests=bogofilter, spamicity=0.000000, version=1.2.4 Sender: owner-linux-mm@kvack.org Precedence: bulk X-Loop: owner-majordomo@kvack.org List-ID: List-Subscribe: List-Unsubscribe: From: Sascha Hauer This driver provides a I2C bus driver for the CGEB interface found on some Congatec x86 modules. No devices are registered on the bus, the user has to do this via the i2c device /sys interface. Signed-off-by: Sascha Hauer Signed-off-by: Christian Gmeiner Signed-off-by: Mary Strodl --- drivers/i2c/busses/Kconfig | 10 ++ drivers/i2c/busses/Makefile | 1 + drivers/i2c/busses/i2c-congatec-cgeb.c | 190 +++++++++++++++++++++++++ 3 files changed, 201 insertions(+) create mode 100644 drivers/i2c/busses/i2c-congatec-cgeb.c diff --git a/drivers/i2c/busses/Kconfig b/drivers/i2c/busses/Kconfig index fe6e8a1bb607..0a5d348c4664 100644 --- a/drivers/i2c/busses/Kconfig +++ b/drivers/i2c/busses/Kconfig @@ -1261,6 +1261,16 @@ config I2C_RCAR This driver can also be built as a module. If so, the module will be called i2c-rcar. +config I2C_CONGATEC_CGEB + tristate "Congatec CGEB I2C driver" + depends on MFD_CONGATEC_CGEB + help + If you say yes to this option, support will be included for the + Congatec CGEB I2C controller. + + This driver can also be built as a module. If so, the module + will be called i2c-congatec-cgeb. + comment "External I2C/SMBus adapter drivers" config I2C_DIOLAN_U2C diff --git a/drivers/i2c/busses/Makefile b/drivers/i2c/busses/Makefile index 78d0561339e5..f4e9fa7542be 100644 --- a/drivers/i2c/busses/Makefile +++ b/drivers/i2c/busses/Makefile @@ -128,6 +128,7 @@ obj-$(CONFIG_I2C_XILINX) += i2c-xiic.o obj-$(CONFIG_I2C_XLP9XX) += i2c-xlp9xx.o obj-$(CONFIG_I2C_RCAR) += i2c-rcar.o obj-$(CONFIG_I2C_GXP) += i2c-gxp.o +obj-$(CONFIG_I2C_CONGATEC_CGEB) += i2c-congatec-cgeb.o # External I2C/SMBus adapter drivers obj-$(CONFIG_I2C_DIOLAN_U2C) += i2c-diolan-u2c.o diff --git a/drivers/i2c/busses/i2c-congatec-cgeb.c b/drivers/i2c/busses/i2c-congatec-cgeb.c new file mode 100644 index 000000000000..3adeb7bf3534 --- /dev/null +++ b/drivers/i2c/busses/i2c-congatec-cgeb.c @@ -0,0 +1,190 @@ +// SPDX-License-Identifier: GPL-2.0-only +/* + * CGEB i2c driver + * + * (c) 2011 Sascha Hauer, Pengutronix + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; version 2 of the License. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ +#include +#include +#include +#include +#include + +#define CG_I2C_FLAG_START 0x00080 /* send START condition */ +#define CG_I2C_FLAG_STOP 0x00040 /* send STOP condition */ +#define CG_I2C_FLAG_ALL_ACK 0x08000 /* send ACK on all read bytes */ +#define CG_I2C_FLAG_ALL_NAK 0x04000 /* send NAK on all read bytes */ + +struct cgeb_i2c_priv { + struct cgeb_board_data *board; + struct i2c_adapter adapter; + int unit; +}; + +static u32 cgeb_i2c_func(struct i2c_adapter *adapter) +{ + return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL; +} + +static int cgeb_i2c_set_speed(struct cgeb_i2c_priv *priv, int speed) +{ + struct cgeb_function_parameters fps; + + memset(&fps, 0, sizeof(fps)); + + fps.unit = priv->unit; + fps.pars[0] = speed; + + return cgeb_call(priv->board, &fps, CgebI2CSetFrequency); +} + +static int cgeb_i2c_xfer(struct i2c_adapter *adapter, + struct i2c_msg *msgs, int num) +{ + struct cgeb_function_parameters fps; + int i, ret; + unsigned long flags = CG_I2C_FLAG_START; + struct cgeb_i2c_priv *priv = i2c_get_adapdata(adapter); + unsigned long rdlen, wrlen; + unsigned char *rdbuf, *wrbuf, *raw_wrbuf; + unsigned short lmax = 0; + + /* + * With cgeb the I2C address is part of the write data + * buffer, so allocate a buffer with the length of the + * longest write buffer + 1 + */ + for (i = 0; i < num; i++) + if (!(msgs[i].flags & I2C_M_RD)) + lmax = max(lmax, msgs[i].len); + + raw_wrbuf = kmalloc(lmax + 1, GFP_KERNEL); + if (!raw_wrbuf) + return -ENOMEM; + + for (i = 0; i < num; i++) { + if (msgs[i].flags & I2C_M_RD) { + rdbuf = msgs[i].buf; + rdlen = msgs[i].len; + wrbuf = NULL; + wrlen = 0; + } else { + rdbuf = NULL; + rdlen = 0; + wrbuf = msgs[i].buf; + wrlen = msgs[i].len; + } + + raw_wrbuf[0] = msgs[i].addr << 1; + if (wrlen) + memcpy(&raw_wrbuf[1], wrbuf, wrlen); + + if (msgs[i].flags & I2C_M_RD) + raw_wrbuf[0] |= 1; + + if (i == num - 1) + flags |= CG_I2C_FLAG_STOP; + + dev_dbg(&adapter->dev, + "%s: rd: %p/%ld wr: %p/%ld flags: 0x%08lx %s\n", + __func__, rdbuf, rdlen, raw_wrbuf, wrlen + 1, + flags, + msgs[i].flags & I2C_M_RD ? "READ" : "WRITE"); + + memset(&fps, 0, sizeof(fps)); + + fps.unit = priv->unit; + fps.pars[0] = wrlen + 1; + fps.pars[1] = rdlen; + fps.pars[2] = flags; + fps.iptr = raw_wrbuf; + fps.optr = rdbuf; + fps.optr_size = sizeof(*rdbuf) * rdlen; + fps.iptr_size = (wrlen + 1) * sizeof(*raw_wrbuf); + + ret = cgeb_call(priv->board, &fps, CgebI2CTransfer); + if (ret) { + ret = -EREMOTEIO; + goto out; + } + } + + ret = num; + +out: + kfree(raw_wrbuf); + + return ret; +} + +static struct i2c_algorithm cgeb_i2c_algo = { + .master_xfer = cgeb_i2c_xfer, + .functionality = cgeb_i2c_func, +}; + +static int cgeb_i2c_probe(struct platform_device *pdev) +{ + struct cgeb_i2c_priv *priv; + struct cgeb_pdata *pdata = pdev->dev.platform_data; + int ret; + + priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL); + if (!priv) + return -ENOMEM; + + strscpy(priv->adapter.name, pdev->name); + priv->adapter.owner = THIS_MODULE; + priv->adapter.algo = &cgeb_i2c_algo; + priv->adapter.dev.parent = &pdev->dev; + priv->unit = pdata->unit; + priv->board = pdata->board; + i2c_set_adapdata(&priv->adapter, priv); + + platform_set_drvdata(pdev, priv); + + ret = cgeb_i2c_set_speed(priv, 400000); + if (ret) + /* not a critical error, we can continue with the default speed. */ + dev_warn(&pdev->dev, "Could not set speed to 400KHz\n"); + + ret = i2c_add_adapter(&priv->adapter); + if (ret < 0) { + dev_err_probe(&pdev->dev, ret, "registration failed\n"); + return ret; + } + + dev_info(&pdev->dev, "registered\n"); + + return 0; +}; + +static int cgeb_i2c_remove(struct platform_device *pdev) +{ + struct cgeb_i2c_priv *priv = platform_get_drvdata(pdev); + + i2c_del_adapter(&priv->adapter); + + return 0; +} + +static struct platform_driver cgeb_i2c_driver = { + .probe = cgeb_i2c_probe, + .remove = cgeb_i2c_remove, + .driver = { + .name = "cgeb-i2c", + }, +}; +module_platform_driver(cgeb_i2c_driver); + +MODULE_AUTHOR("Sascha Hauer "); +MODULE_DESCRIPTION("cgeb i2c driver"); +MODULE_LICENSE("GPL");