Inter-revision diff: patch 7

Comparing v6 (message) to v2 (message)

--- v6
+++ v2
@@ -1,213 +1,551 @@
-PECI devices may not be discoverable at the time when PECI controller is
-being added (e.g. BMC can boot up when the Host system is still in S5).
-Since we currently don't have the capabilities to figure out the Host
-system state inside the PECI subsystem itself, we have to rely on
-userspace to do it for us.
+From: Jae Hyun Yoo <jae.hyun.yoo@linux.intel.com>
 
-In the future, PECI subsystem may be expanded with mechanisms that allow
-us to avoid depending on userspace interaction (e.g. CPU presence could
-be detected using GPIO, and the information on whether it's discoverable
-could be obtained over IPMI).
-Unfortunately, those methods may ultimately not be available (support
-will vary from platform to platform), which means that we still need
-platform independent method triggered by userspace.
+ASPEED AST24xx/AST25xx/AST26xx SoCs supports the PECI electrical
+interface (a.k.a PECI wire).
 
+Signed-off-by: Jae Hyun Yoo <jae.hyun.yoo@linux.intel.com>
+Co-developed-by: Iwona Winiarska <iwona.winiarska@intel.com>
 Signed-off-by: Iwona Winiarska <iwona.winiarska@intel.com>
+Reviewed-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
 ---
- Documentation/ABI/testing/sysfs-bus-peci | 16 +++++
- drivers/peci/Makefile                    |  2 +-
- drivers/peci/core.c                      |  3 +-
- drivers/peci/device.c                    |  1 +
- drivers/peci/internal.h                  |  5 ++
- drivers/peci/sysfs.c                     | 82 ++++++++++++++++++++++++
- 6 files changed, 107 insertions(+), 2 deletions(-)
- create mode 100644 Documentation/ABI/testing/sysfs-bus-peci
- create mode 100644 drivers/peci/sysfs.c
+ MAINTAINERS                           |   9 +
+ drivers/peci/Kconfig                  |   6 +
+ drivers/peci/Makefile                 |   3 +
+ drivers/peci/controller/Kconfig       |  16 +
+ drivers/peci/controller/Makefile      |   3 +
+ drivers/peci/controller/peci-aspeed.c | 445 ++++++++++++++++++++++++++
+ 6 files changed, 482 insertions(+)
+ create mode 100644 drivers/peci/controller/Kconfig
+ create mode 100644 drivers/peci/controller/Makefile
+ create mode 100644 drivers/peci/controller/peci-aspeed.c
 
-diff --git a/Documentation/ABI/testing/sysfs-bus-peci b/Documentation/ABI/testing/sysfs-bus-peci
-new file mode 100644
-index 000000000000..56c2b2216bbd
---- /dev/null
-+++ b/Documentation/ABI/testing/sysfs-bus-peci
-@@ -0,0 +1,16 @@
-+What:		/sys/bus/peci/rescan
-+Date:		July 2021
-+KernelVersion:	5.15
-+Contact:	Iwona Winiarska <iwona.winiarska@intel.com>
-+Description:
-+		Writing a non-zero value to this attribute will
-+		initiate scan for PECI devices on all PECI controllers
-+		in the system.
-+
-+What:		/sys/bus/peci/devices/<controller_id>-<device_addr>/remove
-+Date:		July 2021
-+KernelVersion:	5.15
-+Contact:	Iwona Winiarska <iwona.winiarska@intel.com>
-+Description:
-+		Writing a non-zero value to this attribute will
-+		remove the PECI device and any of its children.
+diff --git a/MAINTAINERS b/MAINTAINERS
+index d411974aaa5e..6e9d53ff68ab 100644
+--- a/MAINTAINERS
++++ b/MAINTAINERS
+@@ -2866,6 +2866,15 @@ S:	Maintained
+ F:	Documentation/hwmon/asc7621.rst
+ F:	drivers/hwmon/asc7621.c
+ 
++ASPEED PECI CONTROLLER
++M:	Iwona Winiarska <iwona.winiarska@intel.com>
++M:	Jae Hyun Yoo <jae.hyun.yoo@linux.intel.com>
++L:	linux-aspeed at lists.ozlabs.org (moderated for non-subscribers)
++L:	openbmc at lists.ozlabs.org (moderated for non-subscribers)
++S:	Supported
++F:	Documentation/devicetree/bindings/peci/peci-aspeed.yaml
++F:	drivers/peci/controller/peci-aspeed.c
++
+ ASPEED PINCTRL DRIVERS
+ M:	Andrew Jeffery <andrew@aj.id.au>
+ L:	linux-aspeed at lists.ozlabs.org (moderated for non-subscribers)
+diff --git a/drivers/peci/Kconfig b/drivers/peci/Kconfig
+index 71a4ad81225a..99279df97a78 100644
+--- a/drivers/peci/Kconfig
++++ b/drivers/peci/Kconfig
+@@ -13,3 +13,9 @@ menuconfig PECI
+ 
+ 	  This support is also available as a module. If so, the module
+ 	  will be called peci.
++
++if PECI
++
++source "drivers/peci/controller/Kconfig"
++
++endif # PECI
 diff --git a/drivers/peci/Makefile b/drivers/peci/Makefile
-index c5f9d3fe21bb..917f689e147a 100644
+index e789a354e842..926d8df15cbd 100644
 --- a/drivers/peci/Makefile
 +++ b/drivers/peci/Makefile
-@@ -1,7 +1,7 @@
- # SPDX-License-Identifier: GPL-2.0-only
- 
+@@ -3,3 +3,6 @@
  # Core functionality
--peci-y := core.o request.o device.o
-+peci-y := core.o request.o device.o sysfs.o
+ peci-y := core.o
  obj-$(CONFIG_PECI) += peci.o
- 
- # Hardware specific bus drivers
-diff --git a/drivers/peci/core.c b/drivers/peci/core.c
-index c3361e6e043a..e993615cf521 100644
---- a/drivers/peci/core.c
-+++ b/drivers/peci/core.c
-@@ -29,7 +29,7 @@ struct device_type peci_controller_type = {
- 	.release	= peci_controller_dev_release,
- };
- 
--static int peci_controller_scan_devices(struct peci_controller *controller)
-+int peci_controller_scan_devices(struct peci_controller *controller)
- {
- 	int ret;
- 	u8 addr;
-@@ -162,6 +162,7 @@ EXPORT_SYMBOL_NS_GPL(devm_peci_controller_add, PECI);
- 
- struct bus_type peci_bus_type = {
- 	.name		= "peci",
-+	.bus_groups	= peci_bus_groups,
- };
- 
- static int __init peci_init(void)
-diff --git a/drivers/peci/device.c b/drivers/peci/device.c
-index 2b3a2d893aaf..d10ed1cfcd48 100644
---- a/drivers/peci/device.c
-+++ b/drivers/peci/device.c
-@@ -116,5 +116,6 @@ static void peci_device_release(struct device *dev)
- }
- 
- struct device_type peci_device_type = {
-+	.groups		= peci_device_groups,
- 	.release	= peci_device_release,
- };
-diff --git a/drivers/peci/internal.h b/drivers/peci/internal.h
-index 57d11a902c5d..978e12c8e1d3 100644
---- a/drivers/peci/internal.h
-+++ b/drivers/peci/internal.h
-@@ -8,6 +8,7 @@
- #include <linux/types.h>
- 
- struct peci_controller;
-+struct attribute_group;
- struct peci_device;
- struct peci_request;
- 
-@@ -19,12 +20,16 @@ struct peci_request *peci_request_alloc(struct peci_device *device, u8 tx_len, u
- void peci_request_free(struct peci_request *req);
- 
- extern struct device_type peci_device_type;
-+extern const struct attribute_group *peci_device_groups[];
- 
- int peci_device_create(struct peci_controller *controller, u8 addr);
- void peci_device_destroy(struct peci_device *device);
- 
- extern struct bus_type peci_bus_type;
-+extern const struct attribute_group *peci_bus_groups[];
- 
- extern struct device_type peci_controller_type;
- 
-+int peci_controller_scan_devices(struct peci_controller *controller);
-+
- #endif /* __PECI_INTERNAL_H */
-diff --git a/drivers/peci/sysfs.c b/drivers/peci/sysfs.c
++
++# Hardware specific bus drivers
++obj-y += controller/
+diff --git a/drivers/peci/controller/Kconfig b/drivers/peci/controller/Kconfig
 new file mode 100644
-index 000000000000..db9ef05776e3
+index 000000000000..6d48df08db1c
 --- /dev/null
-+++ b/drivers/peci/sysfs.c
-@@ -0,0 +1,82 @@
++++ b/drivers/peci/controller/Kconfig
+@@ -0,0 +1,16 @@
++# SPDX-License-Identifier: GPL-2.0-only
++
++config PECI_ASPEED
++	tristate "ASPEED PECI support"
++	depends on ARCH_ASPEED || COMPILE_TEST
++	depends on OF
++	depends on HAS_IOMEM
++	help
++	  This option enables PECI controller driver for ASPEED AST2400,
++	  AST2500 and AST2600 SoCs.
++
++	  Say Y here if your system runs on ASPEED SoC and you are using it
++	  as BMC for Intel platform.
++
++	  This driver can also be built as a module. If so, the module will
++	  be called peci-aspeed.
+diff --git a/drivers/peci/controller/Makefile b/drivers/peci/controller/Makefile
+new file mode 100644
+index 000000000000..022c28ef1bf0
+--- /dev/null
++++ b/drivers/peci/controller/Makefile
+@@ -0,0 +1,3 @@
++# SPDX-License-Identifier: GPL-2.0-only
++
++obj-$(CONFIG_PECI_ASPEED)	+= peci-aspeed.o
+diff --git a/drivers/peci/controller/peci-aspeed.c b/drivers/peci/controller/peci-aspeed.c
+new file mode 100644
+index 000000000000..1d708c983749
+--- /dev/null
++++ b/drivers/peci/controller/peci-aspeed.c
+@@ -0,0 +1,445 @@
 +// SPDX-License-Identifier: GPL-2.0-only
-+// Copyright (c) 2021 Intel Corporation
-+
-+#include <linux/device.h>
-+#include <linux/kernel.h>
++// Copyright (C) 2012-2017 ASPEED Technology Inc.
++// Copyright (c) 2018-2021 Intel Corporation
++
++#include <linux/bitfield.h>
++#include <linux/clk.h>
++#include <linux/delay.h>
++#include <linux/interrupt.h>
++#include <linux/io.h>
++#include <linux/iopoll.h>
++#include <linux/jiffies.h>
++#include <linux/module.h>
++#include <linux/of.h>
 +#include <linux/peci.h>
-+
-+#include "internal.h"
-+
-+static int rescan_controller(struct device *dev, void *data)
-+{
-+	if (dev->type != &peci_controller_type)
-+		return 0;
-+
-+	return peci_controller_scan_devices(to_peci_controller(dev));
-+}
-+
-+static ssize_t rescan_store(struct bus_type *bus, const char *buf, size_t count)
-+{
-+	bool res;
++#include <linux/platform_device.h>
++#include <linux/reset.h>
++
++#include <asm/unaligned.h>
++
++/* ASPEED PECI Registers */
++/* Control Register */
++#define ASPEED_PECI_CTRL			0x00
++#define   ASPEED_PECI_CTRL_SAMPLING_MASK	GENMASK(19, 16)
++#define   ASPEED_PECI_CTRL_RD_MODE_MASK		GENMASK(13, 12)
++#define     ASPEED_PECI_CTRL_RD_MODE_DBG	BIT(13)
++#define     ASPEED_PECI_CTRL_RD_MODE_COUNT	BIT(12)
++#define   ASPEED_PECI_CTRL_CLK_SOURCE		BIT(11)
++#define   ASPEED_PECI_CTRL_CLK_DIV_MASK		GENMASK(10, 8)
++#define   ASPEED_PECI_CTRL_INVERT_OUT		BIT(7)
++#define   ASPEED_PECI_CTRL_INVERT_IN		BIT(6)
++#define   ASPEED_PECI_CTRL_BUS_CONTENTION_EN	BIT(5)
++#define   ASPEED_PECI_CTRL_PECI_EN		BIT(4)
++#define   ASPEED_PECI_CTRL_PECI_CLK_EN		BIT(0)
++
++/* Timing Negotiation Register */
++#define ASPEED_PECI_TIMING_NEGOTIATION		0x04
++#define   ASPEED_PECI_T_NEGO_MSG_MASK		GENMASK(15, 8)
++#define   ASPEED_PECI_T_NEGO_ADDR_MASK		GENMASK(7, 0)
++
++/* Command Register */
++#define ASPEED_PECI_CMD				0x08
++#define   ASPEED_PECI_CMD_PIN_MONITORING	BIT(31)
++#define   ASPEED_PECI_CMD_STS_MASK		GENMASK(27, 24)
++#define     ASPEED_PECI_CMD_STS_ADDR_T_NEGO	0x3
++#define   ASPEED_PECI_CMD_IDLE_MASK		\
++	  (ASPEED_PECI_CMD_STS_MASK | ASPEED_PECI_CMD_PIN_MONITORING)
++#define   ASPEED_PECI_CMD_FIRE			BIT(0)
++
++/* Read/Write Length Register */
++#define ASPEED_PECI_RW_LENGTH			0x0c
++#define   ASPEED_PECI_AW_FCS_EN			BIT(31)
++#define   ASPEED_PECI_RD_LEN_MASK		GENMASK(23, 16)
++#define   ASPEED_PECI_WR_LEN_MASK		GENMASK(15, 8)
++#define   ASPEED_PECI_TARGET_ADDR_MASK		GENMASK(7, 0)
++
++/* Expected FCS Data Register */
++#define ASPEED_PECI_EXPECTED_FCS		0x10
++#define   ASPEED_PECI_EXPECTED_RD_FCS_MASK	GENMASK(23, 16)
++#define   ASPEED_PECI_EXPECTED_AW_FCS_AUTO_MASK	GENMASK(15, 8)
++#define   ASPEED_PECI_EXPECTED_WR_FCS_MASK	GENMASK(7, 0)
++
++/* Captured FCS Data Register */
++#define ASPEED_PECI_CAPTURED_FCS		0x14
++#define   ASPEED_PECI_CAPTURED_RD_FCS_MASK	GENMASK(23, 16)
++#define   ASPEED_PECI_CAPTURED_WR_FCS_MASK	GENMASK(7, 0)
++
++/* Interrupt Register */
++#define ASPEED_PECI_INT_CTRL			0x18
++#define   ASPEED_PECI_TIMING_NEGO_SEL_MASK	GENMASK(31, 30)
++#define     ASPEED_PECI_1ST_BIT_OF_ADDR_NEGO	0
++#define     ASPEED_PECI_2ND_BIT_OF_ADDR_NEGO	1
++#define     ASPEED_PECI_MESSAGE_NEGO		2
++#define   ASPEED_PECI_INT_MASK			GENMASK(4, 0)
++#define     ASPEED_PECI_INT_BUS_TIMEOUT		BIT(4)
++#define     ASPEED_PECI_INT_BUS_CONTENTION	BIT(3)
++#define     ASPEED_PECI_INT_WR_FCS_BAD		BIT(2)
++#define     ASPEED_PECI_INT_WR_FCS_ABORT	BIT(1)
++#define     ASPEED_PECI_INT_CMD_DONE		BIT(0)
++
++/* Interrupt Status Register */
++#define ASPEED_PECI_INT_STS			0x1c
++#define   ASPEED_PECI_INT_TIMING_RESULT_MASK	GENMASK(29, 16)
++	  /* bits[4..0]: Same bit fields in the 'Interrupt Register' */
++
++/* Rx/Tx Data Buffer Registers */
++#define ASPEED_PECI_WR_DATA0			0x20
++#define ASPEED_PECI_WR_DATA1			0x24
++#define ASPEED_PECI_WR_DATA2			0x28
++#define ASPEED_PECI_WR_DATA3			0x2c
++#define ASPEED_PECI_RD_DATA0			0x30
++#define ASPEED_PECI_RD_DATA1			0x34
++#define ASPEED_PECI_RD_DATA2			0x38
++#define ASPEED_PECI_RD_DATA3			0x3c
++#define ASPEED_PECI_WR_DATA4			0x40
++#define ASPEED_PECI_WR_DATA5			0x44
++#define ASPEED_PECI_WR_DATA6			0x48
++#define ASPEED_PECI_WR_DATA7			0x4c
++#define ASPEED_PECI_RD_DATA4			0x50
++#define ASPEED_PECI_RD_DATA5			0x54
++#define ASPEED_PECI_RD_DATA6			0x58
++#define ASPEED_PECI_RD_DATA7			0x5c
++#define   ASPEED_PECI_DATA_BUF_SIZE_MAX		32
++
++/* Timing Negotiation */
++#define ASPEED_PECI_RD_SAMPLING_POINT_DEFAULT	8
++#define ASPEED_PECI_RD_SAMPLING_POINT_MAX	(BIT(4) - 1)
++#define ASPEED_PECI_CLK_DIV_DEFAULT		0
++#define ASPEED_PECI_CLK_DIV_MAX			(BIT(3) - 1)
++#define ASPEED_PECI_MSG_TIMING_DEFAULT		1
++#define ASPEED_PECI_MSG_TIMING_MAX		(BIT(8) - 1)
++#define ASPEED_PECI_ADDR_TIMING_DEFAULT		1
++#define ASPEED_PECI_ADDR_TIMING_MAX		(BIT(8) - 1)
++
++/* Timeout */
++#define ASPEED_PECI_IDLE_CHECK_TIMEOUT_US	(50 * USEC_PER_MSEC)
++#define ASPEED_PECI_IDLE_CHECK_INTERVAL_US	(10 * USEC_PER_MSEC)
++#define ASPEED_PECI_CMD_TIMEOUT_MS_DEFAULT	(1000)
++#define ASPEED_PECI_CMD_TIMEOUT_MS_MAX		(1000)
++
++struct aspeed_peci {
++	struct peci_controller *controller;
++	struct device *dev;
++	void __iomem *base;
++	struct clk *clk;
++	struct reset_control *rst;
++	int irq;
++	spinlock_t lock; /* to sync completion status handling */
++	struct completion xfer_complete;
++	u32 status;
++	u32 cmd_timeout_ms;
++	u32 msg_timing;
++	u32 addr_timing;
++	u32 rd_sampling_point;
++	u32 clk_div;
++};
++
++static void aspeed_peci_init_regs(struct aspeed_peci *priv)
++{
++	u32 val;
++
++	val = FIELD_PREP(ASPEED_PECI_CTRL_CLK_DIV_MASK, ASPEED_PECI_CLK_DIV_DEFAULT);
++	val |= ASPEED_PECI_CTRL_PECI_CLK_EN;
++	writel(val, priv->base + ASPEED_PECI_CTRL);
++	/*
++	 * Timing negotiation period setting.
++	 * The unit of the programmed value is 4 times of PECI clock period.
++	 */
++	val = FIELD_PREP(ASPEED_PECI_T_NEGO_MSG_MASK, priv->msg_timing);
++	val |= FIELD_PREP(ASPEED_PECI_T_NEGO_ADDR_MASK, priv->addr_timing);
++	writel(val, priv->base + ASPEED_PECI_TIMING_NEGOTIATION);
++
++	/* Clear interrupts */
++	val = readl(priv->base + ASPEED_PECI_INT_STS) | ASPEED_PECI_INT_MASK;
++	writel(val, priv->base + ASPEED_PECI_INT_STS);
++
++	/* Set timing negotiation mode and enable interrupts */
++	val = FIELD_PREP(ASPEED_PECI_TIMING_NEGO_SEL_MASK, ASPEED_PECI_1ST_BIT_OF_ADDR_NEGO);
++	val |= ASPEED_PECI_INT_MASK;
++	writel(val, priv->base + ASPEED_PECI_INT_CTRL);
++
++	val = FIELD_PREP(ASPEED_PECI_CTRL_SAMPLING_MASK, priv->rd_sampling_point);
++	val |= FIELD_PREP(ASPEED_PECI_CTRL_CLK_DIV_MASK, priv->clk_div);
++	val |= ASPEED_PECI_CTRL_PECI_EN;
++	val |= ASPEED_PECI_CTRL_PECI_CLK_EN;
++	writel(val, priv->base + ASPEED_PECI_CTRL);
++}
++
++static inline int aspeed_peci_check_idle(struct aspeed_peci *priv)
++{
++	u32 cmd_sts = readl(priv->base + ASPEED_PECI_CMD);
++
++	if (FIELD_GET(ASPEED_PECI_CMD_STS_MASK, cmd_sts) == ASPEED_PECI_CMD_STS_ADDR_T_NEGO)
++		aspeed_peci_init_regs(priv);
++
++	return readl_poll_timeout(priv->base + ASPEED_PECI_CMD,
++				  cmd_sts,
++				  !(cmd_sts & ASPEED_PECI_CMD_IDLE_MASK),
++				  ASPEED_PECI_IDLE_CHECK_INTERVAL_US,
++				  ASPEED_PECI_IDLE_CHECK_TIMEOUT_US);
++}
++
++static int aspeed_peci_xfer(struct peci_controller *controller,
++			    u8 addr, struct peci_request *req)
++{
++	struct aspeed_peci *priv = dev_get_drvdata(controller->dev.parent);
++	unsigned long flags, timeout = msecs_to_jiffies(priv->cmd_timeout_ms);
++	u32 peci_head;
 +	int ret;
 +
-+	ret = kstrtobool(buf, &res);
++	if (req->tx.len > ASPEED_PECI_DATA_BUF_SIZE_MAX ||
++	    req->rx.len > ASPEED_PECI_DATA_BUF_SIZE_MAX)
++		return -EINVAL;
++
++	/* Check command sts and bus idle state */
++	ret = aspeed_peci_check_idle(priv);
++	if (ret)
++		return ret; /* -ETIMEDOUT */
++
++	spin_lock_irqsave(&priv->lock, flags);
++	reinit_completion(&priv->xfer_complete);
++
++	peci_head = FIELD_PREP(ASPEED_PECI_TARGET_ADDR_MASK, addr) |
++		    FIELD_PREP(ASPEED_PECI_WR_LEN_MASK, req->tx.len) |
++		    FIELD_PREP(ASPEED_PECI_RD_LEN_MASK, req->rx.len);
++
++	writel(peci_head, priv->base + ASPEED_PECI_RW_LENGTH);
++
++	memcpy_toio(priv->base + ASPEED_PECI_WR_DATA0, req->tx.buf, min_t(u8, req->tx.len, 16));
++	if (req->tx.len > 16)
++		memcpy_toio(priv->base + ASPEED_PECI_WR_DATA4, req->tx.buf + 16,
++			    req->tx.len - 16);
++
++	dev_dbg(priv->dev, "HEAD : 0x%08x\n", peci_head);
++	print_hex_dump_bytes("TX : ", DUMP_PREFIX_NONE, req->tx.buf, req->tx.len);
++
++	priv->status = 0;
++	writel(ASPEED_PECI_CMD_FIRE, priv->base + ASPEED_PECI_CMD);
++	spin_unlock_irqrestore(&priv->lock, flags);
++
++	ret = wait_for_completion_interruptible_timeout(&priv->xfer_complete, timeout);
++	if (ret < 0)
++		return ret;
++
++	if (ret == 0) {
++		dev_dbg(priv->dev, "Timeout waiting for a response!\n");
++		return -ETIMEDOUT;
++	}
++
++	spin_lock_irqsave(&priv->lock, flags);
++
++	writel(0, priv->base + ASPEED_PECI_CMD);
++
++	if (priv->status != ASPEED_PECI_INT_CMD_DONE) {
++		spin_unlock_irqrestore(&priv->lock, flags);
++		dev_dbg(priv->dev, "No valid response!\n");
++		return -EIO;
++	}
++
++	spin_unlock_irqrestore(&priv->lock, flags);
++
++	memcpy_fromio(req->rx.buf, priv->base + ASPEED_PECI_RD_DATA0, min_t(u8, req->rx.len, 16));
++	if (req->rx.len > 16)
++		memcpy_fromio(req->rx.buf + 16, priv->base + ASPEED_PECI_RD_DATA4,
++			      req->rx.len - 16);
++
++	print_hex_dump_bytes("RX : ", DUMP_PREFIX_NONE, req->rx.buf, req->rx.len);
++
++	return 0;
++}
++
++static irqreturn_t aspeed_peci_irq_handler(int irq, void *arg)
++{
++	struct aspeed_peci *priv = arg;
++	u32 status;
++
++	spin_lock(&priv->lock);
++	status = readl(priv->base + ASPEED_PECI_INT_STS);
++	writel(status, priv->base + ASPEED_PECI_INT_STS);
++	priv->status |= (status & ASPEED_PECI_INT_MASK);
++
++	/*
++	 * In most cases, interrupt bits will be set one by one but also note
++	 * that multiple interrupt bits could be set at the same time.
++	 */
++	if (status & ASPEED_PECI_INT_BUS_TIMEOUT)
++		dev_dbg_ratelimited(priv->dev, "ASPEED_PECI_INT_BUS_TIMEOUT\n");
++
++	if (status & ASPEED_PECI_INT_BUS_CONTENTION)
++		dev_dbg_ratelimited(priv->dev, "ASPEED_PECI_INT_BUS_CONTENTION\n");
++
++	if (status & ASPEED_PECI_INT_WR_FCS_BAD)
++		dev_dbg_ratelimited(priv->dev, "ASPEED_PECI_INT_WR_FCS_BAD\n");
++
++	if (status & ASPEED_PECI_INT_WR_FCS_ABORT)
++		dev_dbg_ratelimited(priv->dev, "ASPEED_PECI_INT_WR_FCS_ABORT\n");
++
++	/*
++	 * All commands should be ended up with a ASPEED_PECI_INT_CMD_DONE bit
++	 * set even in an error case.
++	 */
++	if (status & ASPEED_PECI_INT_CMD_DONE)
++		complete(&priv->xfer_complete);
++
++	spin_unlock(&priv->lock);
++
++	return IRQ_HANDLED;
++}
++
++static void aspeed_peci_property_sanitize(struct device *dev, const char *propname,
++					  u32 min, u32 max, u32 default_val, u32 *propval)
++{
++	u32 val;
++	int ret;
++
++	ret = device_property_read_u32(dev, propname, &val);
++	if (ret) {
++		val = default_val;
++	} else if (val > max || val < min) {
++		dev_warn(dev, "Invalid %s: %u, falling back to: %u\n",
++			 propname, val, default_val);
++
++		val = default_val;
++	}
++
++	*propval = val;
++}
++
++static void aspeed_peci_property_setup(struct aspeed_peci *priv)
++{
++	aspeed_peci_property_sanitize(priv->dev, "aspeed,clock-divider",
++				      0, ASPEED_PECI_CLK_DIV_MAX,
++				      ASPEED_PECI_CLK_DIV_DEFAULT, &priv->clk_div);
++	aspeed_peci_property_sanitize(priv->dev, "aspeed,msg-timing",
++				      0, ASPEED_PECI_MSG_TIMING_MAX,
++				      ASPEED_PECI_MSG_TIMING_DEFAULT, &priv->msg_timing);
++	aspeed_peci_property_sanitize(priv->dev, "aspeed,addr-timing",
++				      0, ASPEED_PECI_ADDR_TIMING_MAX,
++				      ASPEED_PECI_ADDR_TIMING_DEFAULT, &priv->addr_timing);
++	aspeed_peci_property_sanitize(priv->dev, "aspeed,rd-sampling-point",
++				      0, ASPEED_PECI_RD_SAMPLING_POINT_MAX,
++				      ASPEED_PECI_RD_SAMPLING_POINT_DEFAULT,
++				      &priv->rd_sampling_point);
++	aspeed_peci_property_sanitize(priv->dev, "cmd-timeout-ms",
++				      1, ASPEED_PECI_CMD_TIMEOUT_MS_MAX,
++				      ASPEED_PECI_CMD_TIMEOUT_MS_DEFAULT, &priv->cmd_timeout_ms);
++}
++
++static struct peci_controller_ops aspeed_ops = {
++	.xfer = aspeed_peci_xfer,
++};
++
++static void aspeed_peci_reset_control_release(void *data)
++{
++	reset_control_assert(data);
++}
++
++int aspeed_peci_reset_control_deassert(struct device *dev, struct reset_control *rst)
++{
++	int ret;
++
++	ret = reset_control_deassert(rst);
 +	if (ret)
 +		return ret;
 +
-+	if (!res)
-+		return count;
-+
-+	ret = bus_for_each_dev(&peci_bus_type, NULL, NULL, rescan_controller);
++	return devm_add_action_or_reset(dev, aspeed_peci_reset_control_release, rst);
++}
++
++static void aspeed_peci_clk_release(void *data)
++{
++	clk_disable_unprepare(data);
++}
++
++static int aspeed_peci_clk_enable(struct device *dev, struct clk *clk)
++{
++	int ret;
++
++	ret = clk_prepare_enable(clk);
 +	if (ret)
 +		return ret;
 +
-+	return count;
-+}
-+static BUS_ATTR_WO(rescan);
-+
-+static struct attribute *peci_bus_attrs[] = {
-+	&bus_attr_rescan.attr,
-+	NULL
-+};
-+
-+static const struct attribute_group peci_bus_group = {
-+	.attrs = peci_bus_attrs,
-+};
-+
-+const struct attribute_group *peci_bus_groups[] = {
-+	&peci_bus_group,
-+	NULL
-+};
-+
-+static ssize_t remove_store(struct device *dev, struct device_attribute *attr,
-+			    const char *buf, size_t count)
-+{
-+	struct peci_device *device = to_peci_device(dev);
-+	bool res;
++	return devm_add_action_or_reset(dev, aspeed_peci_clk_release, clk);
++}
++
++static int aspeed_peci_probe(struct platform_device *pdev)
++{
++	struct peci_controller *controller;
++	struct aspeed_peci *priv;
 +	int ret;
 +
-+	ret = kstrtobool(buf, &res);
++	priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
++	if (!priv)
++		return -ENOMEM;
++
++	priv->dev = &pdev->dev;
++	dev_set_drvdata(priv->dev, priv);
++
++	priv->base = devm_platform_ioremap_resource(pdev, 0);
++	if (IS_ERR(priv->base))
++		return PTR_ERR(priv->base);
++
++	priv->irq = platform_get_irq(pdev, 0);
++	if (!priv->irq)
++		return priv->irq;
++
++	ret = devm_request_irq(&pdev->dev, priv->irq, aspeed_peci_irq_handler,
++			       0, "peci-aspeed", priv);
 +	if (ret)
 +		return ret;
 +
-+	if (res && device_remove_file_self(dev, attr))
-+		peci_device_destroy(device);
-+
-+	return count;
-+}
-+static DEVICE_ATTR_IGNORE_LOCKDEP(remove, 0200, NULL, remove_store);
-+
-+static struct attribute *peci_device_attrs[] = {
-+	&dev_attr_remove.attr,
-+	NULL
++	init_completion(&priv->xfer_complete);
++	spin_lock_init(&priv->lock);
++
++	priv->rst = devm_reset_control_get(&pdev->dev, NULL);
++	if (IS_ERR(priv->rst))
++		return dev_err_probe(priv->dev, PTR_ERR(priv->rst),
++				     "failed to get reset control\n");
++
++	ret = aspeed_peci_reset_control_deassert(priv->dev, priv->rst);
++	if (ret)
++		return dev_err_probe(priv->dev, ret, "cannot deassert reset control\n");
++
++	priv->clk = devm_clk_get(priv->dev, NULL);
++	if (IS_ERR(priv->clk))
++		return dev_err_probe(priv->dev, PTR_ERR(priv->clk), "failed to get clk\n");
++
++	ret = aspeed_peci_clk_enable(priv->dev, priv->clk);
++	if (ret)
++		return dev_err_probe(priv->dev, ret, "failed to enable clock\n");
++
++	aspeed_peci_property_setup(priv);
++
++	aspeed_peci_init_regs(priv);
++
++	controller = devm_peci_controller_add(priv->dev, &aspeed_ops);
++	if (IS_ERR(controller))
++		return dev_err_probe(priv->dev, PTR_ERR(controller),
++				     "failed to add aspeed peci controller\n");
++
++	priv->controller = controller;
++
++	return 0;
++}
++
++static const struct of_device_id aspeed_peci_of_table[] = {
++	{ .compatible = "aspeed,ast2400-peci", },
++	{ .compatible = "aspeed,ast2500-peci", },
++	{ .compatible = "aspeed,ast2600-peci", },
++	{ }
 +};
-+
-+static const struct attribute_group peci_device_group = {
-+	.attrs = peci_device_attrs,
++MODULE_DEVICE_TABLE(of, aspeed_peci_of_table);
++
++static struct platform_driver aspeed_peci_driver = {
++	.probe  = aspeed_peci_probe,
++	.driver = {
++		.name           = "peci-aspeed",
++		.of_match_table = aspeed_peci_of_table,
++	},
 +};
-+
-+const struct attribute_group *peci_device_groups[] = {
-+	&peci_device_group,
-+	NULL
-+};
++module_platform_driver(aspeed_peci_driver);
++
++MODULE_AUTHOR("Ryan Chen <ryan_chen@aspeedtech.com>");
++MODULE_AUTHOR("Jae Hyun Yoo <jae.hyun.yoo@linux.intel.com>");
++MODULE_DESCRIPTION("ASPEED PECI driver");
++MODULE_LICENSE("GPL");
++MODULE_IMPORT_NS(PECI);
 -- 
 2.31.1
 
Keyboard shortcuts
hback out one level
jnext message in thread
kprevious message in thread
ldrill in
Escclose help / fold thread tree
?toggle this help