162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * AMD Platform Security Processor (PSP) Platform Access interface
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2023 Advanced Micro Devices, Inc.
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Author: Mario Limonciello <mario.limonciello@amd.com>
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci * Some of this code is adapted from drivers/i2c/busses/i2c-designware-amdpsp.c
1062306a36Sopenharmony_ci * developed by Jan Dabros <jsd@semihalf.com> and Copyright (C) 2022 Google Inc.
1162306a36Sopenharmony_ci *
1262306a36Sopenharmony_ci */
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#include <linux/bitfield.h>
1562306a36Sopenharmony_ci#include <linux/errno.h>
1662306a36Sopenharmony_ci#include <linux/iopoll.h>
1762306a36Sopenharmony_ci#include <linux/mutex.h>
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ci#include "platform-access.h"
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_ci#define PSP_CMD_TIMEOUT_US	(500 * USEC_PER_MSEC)
2262306a36Sopenharmony_ci#define DOORBELL_CMDRESP_STS	GENMASK(7, 0)
2362306a36Sopenharmony_ci
2462306a36Sopenharmony_ci/* Recovery field should be equal 0 to start sending commands */
2562306a36Sopenharmony_cistatic int check_recovery(u32 __iomem *cmd)
2662306a36Sopenharmony_ci{
2762306a36Sopenharmony_ci	return FIELD_GET(PSP_CMDRESP_RECOVERY, ioread32(cmd));
2862306a36Sopenharmony_ci}
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_cistatic int wait_cmd(u32 __iomem *cmd)
3162306a36Sopenharmony_ci{
3262306a36Sopenharmony_ci	u32 tmp, expected;
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci	/* Expect mbox_cmd to be cleared and ready bit to be set by PSP */
3562306a36Sopenharmony_ci	expected = FIELD_PREP(PSP_CMDRESP_RESP, 1);
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci	/*
3862306a36Sopenharmony_ci	 * Check for readiness of PSP mailbox in a tight loop in order to
3962306a36Sopenharmony_ci	 * process further as soon as command was consumed.
4062306a36Sopenharmony_ci	 */
4162306a36Sopenharmony_ci	return readl_poll_timeout(cmd, tmp, (tmp & expected), 0,
4262306a36Sopenharmony_ci				  PSP_CMD_TIMEOUT_US);
4362306a36Sopenharmony_ci}
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_ciint psp_check_platform_access_status(void)
4662306a36Sopenharmony_ci{
4762306a36Sopenharmony_ci	struct psp_device *psp = psp_get_master_device();
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	if (!psp || !psp->platform_access_data)
5062306a36Sopenharmony_ci		return -ENODEV;
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	return 0;
5362306a36Sopenharmony_ci}
5462306a36Sopenharmony_ciEXPORT_SYMBOL(psp_check_platform_access_status);
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ciint psp_send_platform_access_msg(enum psp_platform_access_msg msg,
5762306a36Sopenharmony_ci				 struct psp_request *req)
5862306a36Sopenharmony_ci{
5962306a36Sopenharmony_ci	struct psp_device *psp = psp_get_master_device();
6062306a36Sopenharmony_ci	u32 __iomem *cmd, *lo, *hi;
6162306a36Sopenharmony_ci	struct psp_platform_access_device *pa_dev;
6262306a36Sopenharmony_ci	phys_addr_t req_addr;
6362306a36Sopenharmony_ci	u32 cmd_reg;
6462306a36Sopenharmony_ci	int ret;
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci	if (!psp || !psp->platform_access_data)
6762306a36Sopenharmony_ci		return -ENODEV;
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_ci	pa_dev = psp->platform_access_data;
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	if (!pa_dev->vdata->cmdresp_reg || !pa_dev->vdata->cmdbuff_addr_lo_reg ||
7262306a36Sopenharmony_ci	    !pa_dev->vdata->cmdbuff_addr_hi_reg)
7362306a36Sopenharmony_ci		return -ENODEV;
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci	cmd = psp->io_regs + pa_dev->vdata->cmdresp_reg;
7662306a36Sopenharmony_ci	lo = psp->io_regs + pa_dev->vdata->cmdbuff_addr_lo_reg;
7762306a36Sopenharmony_ci	hi = psp->io_regs + pa_dev->vdata->cmdbuff_addr_hi_reg;
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci	mutex_lock(&pa_dev->mailbox_mutex);
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci	if (check_recovery(cmd)) {
8262306a36Sopenharmony_ci		dev_dbg(psp->dev, "platform mailbox is in recovery\n");
8362306a36Sopenharmony_ci		ret = -EBUSY;
8462306a36Sopenharmony_ci		goto unlock;
8562306a36Sopenharmony_ci	}
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci	if (wait_cmd(cmd)) {
8862306a36Sopenharmony_ci		dev_dbg(psp->dev, "platform mailbox is not done processing command\n");
8962306a36Sopenharmony_ci		ret = -EBUSY;
9062306a36Sopenharmony_ci		goto unlock;
9162306a36Sopenharmony_ci	}
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci	/*
9462306a36Sopenharmony_ci	 * Fill mailbox with address of command-response buffer, which will be
9562306a36Sopenharmony_ci	 * used for sending i2c requests as well as reading status returned by
9662306a36Sopenharmony_ci	 * PSP. Use physical address of buffer, since PSP will map this region.
9762306a36Sopenharmony_ci	 */
9862306a36Sopenharmony_ci	req_addr = __psp_pa(req);
9962306a36Sopenharmony_ci	iowrite32(lower_32_bits(req_addr), lo);
10062306a36Sopenharmony_ci	iowrite32(upper_32_bits(req_addr), hi);
10162306a36Sopenharmony_ci
10262306a36Sopenharmony_ci	print_hex_dump_debug("->psp ", DUMP_PREFIX_OFFSET, 16, 2, req,
10362306a36Sopenharmony_ci			     req->header.payload_size, false);
10462306a36Sopenharmony_ci
10562306a36Sopenharmony_ci	/* Write command register to trigger processing */
10662306a36Sopenharmony_ci	cmd_reg = FIELD_PREP(PSP_CMDRESP_CMD, msg);
10762306a36Sopenharmony_ci	iowrite32(cmd_reg, cmd);
10862306a36Sopenharmony_ci
10962306a36Sopenharmony_ci	if (wait_cmd(cmd)) {
11062306a36Sopenharmony_ci		ret = -ETIMEDOUT;
11162306a36Sopenharmony_ci		goto unlock;
11262306a36Sopenharmony_ci	}
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_ci	/* Ensure it was triggered by this driver */
11562306a36Sopenharmony_ci	if (ioread32(lo) != lower_32_bits(req_addr) ||
11662306a36Sopenharmony_ci	    ioread32(hi) != upper_32_bits(req_addr)) {
11762306a36Sopenharmony_ci		ret = -EBUSY;
11862306a36Sopenharmony_ci		goto unlock;
11962306a36Sopenharmony_ci	}
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	/*
12262306a36Sopenharmony_ci	 * Read status from PSP. If status is non-zero, it indicates an error
12362306a36Sopenharmony_ci	 * occurred during "processing" of the command.
12462306a36Sopenharmony_ci	 * If status is zero, it indicates the command was "processed"
12562306a36Sopenharmony_ci	 * successfully, but the result of the command is in the payload.
12662306a36Sopenharmony_ci	 * Return both cases to the caller as -EIO to investigate.
12762306a36Sopenharmony_ci	 */
12862306a36Sopenharmony_ci	cmd_reg = ioread32(cmd);
12962306a36Sopenharmony_ci	if (FIELD_GET(PSP_CMDRESP_STS, cmd_reg))
13062306a36Sopenharmony_ci		req->header.status = FIELD_GET(PSP_CMDRESP_STS, cmd_reg);
13162306a36Sopenharmony_ci	if (req->header.status) {
13262306a36Sopenharmony_ci		ret = -EIO;
13362306a36Sopenharmony_ci		goto unlock;
13462306a36Sopenharmony_ci	}
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci	print_hex_dump_debug("<-psp ", DUMP_PREFIX_OFFSET, 16, 2, req,
13762306a36Sopenharmony_ci			     req->header.payload_size, false);
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_ci	ret = 0;
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_ciunlock:
14262306a36Sopenharmony_ci	mutex_unlock(&pa_dev->mailbox_mutex);
14362306a36Sopenharmony_ci
14462306a36Sopenharmony_ci	return ret;
14562306a36Sopenharmony_ci}
14662306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(psp_send_platform_access_msg);
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_ciint psp_ring_platform_doorbell(int msg, u32 *result)
14962306a36Sopenharmony_ci{
15062306a36Sopenharmony_ci	struct psp_device *psp = psp_get_master_device();
15162306a36Sopenharmony_ci	struct psp_platform_access_device *pa_dev;
15262306a36Sopenharmony_ci	u32 __iomem *button, *cmd;
15362306a36Sopenharmony_ci	int ret, val;
15462306a36Sopenharmony_ci
15562306a36Sopenharmony_ci	if (!psp || !psp->platform_access_data)
15662306a36Sopenharmony_ci		return -ENODEV;
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_ci	pa_dev = psp->platform_access_data;
15962306a36Sopenharmony_ci	button = psp->io_regs + pa_dev->vdata->doorbell_button_reg;
16062306a36Sopenharmony_ci	cmd = psp->io_regs + pa_dev->vdata->doorbell_cmd_reg;
16162306a36Sopenharmony_ci
16262306a36Sopenharmony_ci	mutex_lock(&pa_dev->doorbell_mutex);
16362306a36Sopenharmony_ci
16462306a36Sopenharmony_ci	if (wait_cmd(cmd)) {
16562306a36Sopenharmony_ci		dev_err(psp->dev, "doorbell command not done processing\n");
16662306a36Sopenharmony_ci		ret = -EBUSY;
16762306a36Sopenharmony_ci		goto unlock;
16862306a36Sopenharmony_ci	}
16962306a36Sopenharmony_ci
17062306a36Sopenharmony_ci	iowrite32(FIELD_PREP(DOORBELL_CMDRESP_STS, msg), cmd);
17162306a36Sopenharmony_ci	iowrite32(PSP_DRBL_RING, button);
17262306a36Sopenharmony_ci
17362306a36Sopenharmony_ci	if (wait_cmd(cmd)) {
17462306a36Sopenharmony_ci		ret = -ETIMEDOUT;
17562306a36Sopenharmony_ci		goto unlock;
17662306a36Sopenharmony_ci	}
17762306a36Sopenharmony_ci
17862306a36Sopenharmony_ci	val = FIELD_GET(DOORBELL_CMDRESP_STS, ioread32(cmd));
17962306a36Sopenharmony_ci	if (val) {
18062306a36Sopenharmony_ci		if (result)
18162306a36Sopenharmony_ci			*result = val;
18262306a36Sopenharmony_ci		ret = -EIO;
18362306a36Sopenharmony_ci		goto unlock;
18462306a36Sopenharmony_ci	}
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_ci	ret = 0;
18762306a36Sopenharmony_ciunlock:
18862306a36Sopenharmony_ci	mutex_unlock(&pa_dev->doorbell_mutex);
18962306a36Sopenharmony_ci
19062306a36Sopenharmony_ci	return ret;
19162306a36Sopenharmony_ci}
19262306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(psp_ring_platform_doorbell);
19362306a36Sopenharmony_ci
19462306a36Sopenharmony_civoid platform_access_dev_destroy(struct psp_device *psp)
19562306a36Sopenharmony_ci{
19662306a36Sopenharmony_ci	struct psp_platform_access_device *pa_dev = psp->platform_access_data;
19762306a36Sopenharmony_ci
19862306a36Sopenharmony_ci	if (!pa_dev)
19962306a36Sopenharmony_ci		return;
20062306a36Sopenharmony_ci
20162306a36Sopenharmony_ci	mutex_destroy(&pa_dev->mailbox_mutex);
20262306a36Sopenharmony_ci	mutex_destroy(&pa_dev->doorbell_mutex);
20362306a36Sopenharmony_ci	psp->platform_access_data = NULL;
20462306a36Sopenharmony_ci}
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ciint platform_access_dev_init(struct psp_device *psp)
20762306a36Sopenharmony_ci{
20862306a36Sopenharmony_ci	struct device *dev = psp->dev;
20962306a36Sopenharmony_ci	struct psp_platform_access_device *pa_dev;
21062306a36Sopenharmony_ci
21162306a36Sopenharmony_ci	pa_dev = devm_kzalloc(dev, sizeof(*pa_dev), GFP_KERNEL);
21262306a36Sopenharmony_ci	if (!pa_dev)
21362306a36Sopenharmony_ci		return -ENOMEM;
21462306a36Sopenharmony_ci
21562306a36Sopenharmony_ci	psp->platform_access_data = pa_dev;
21662306a36Sopenharmony_ci	pa_dev->psp = psp;
21762306a36Sopenharmony_ci	pa_dev->dev = dev;
21862306a36Sopenharmony_ci
21962306a36Sopenharmony_ci	pa_dev->vdata = (struct platform_access_vdata *)psp->vdata->platform_access;
22062306a36Sopenharmony_ci
22162306a36Sopenharmony_ci	mutex_init(&pa_dev->mailbox_mutex);
22262306a36Sopenharmony_ci	mutex_init(&pa_dev->doorbell_mutex);
22362306a36Sopenharmony_ci
22462306a36Sopenharmony_ci	dev_dbg(dev, "platform access enabled\n");
22562306a36Sopenharmony_ci
22662306a36Sopenharmony_ci	return 0;
22762306a36Sopenharmony_ci}
228