On 04/19/2018 11:50 AM, Ley Foon Tan wrote: > Add mailbox support for Stratix SoC > > Signed-off-by: Ley Foon Tan <ley.foon....@intel.com> > Signed-off-by: Chin Liang See <chin.liang....@intel.com> > --- > arch/arm/mach-socfpga/Makefile | 1 + > arch/arm/mach-socfpga/include/mach/mailbox_s10.h | 155 +++++++++ > arch/arm/mach-socfpga/mailbox_s10.c | 378 > ++++++++++++++++++++++ > 3 files changed, 534 insertions(+), 0 deletions(-) > create mode 100644 arch/arm/mach-socfpga/include/mach/mailbox_s10.h > create mode 100644 arch/arm/mach-socfpga/mailbox_s10.c > > diff --git a/arch/arm/mach-socfpga/Makefile b/arch/arm/mach-socfpga/Makefile > index b253914..43e18d2 100644 > --- a/arch/arm/mach-socfpga/Makefile > +++ b/arch/arm/mach-socfpga/Makefile > @@ -32,6 +32,7 @@ endif > > ifdef CONFIG_TARGET_SOCFPGA_STRATIX10 > obj-y += clock_manager_s10.o > +obj-y += mailbox_s10.o > obj-y += misc_s10.o > obj-y += reset_manager_s10.o > obj-y += system_manager_s10.o > diff --git a/arch/arm/mach-socfpga/include/mach/mailbox_s10.h > b/arch/arm/mach-socfpga/include/mach/mailbox_s10.h > new file mode 100644 > index 0000000..85e7f84 > --- /dev/null > +++ b/arch/arm/mach-socfpga/include/mach/mailbox_s10.h > @@ -0,0 +1,155 @@ > +/* SPDX-License-Identifier: GPL-2.0 > + * > + * Copyright (C) 2017-2018 Intel Corporation <www.intel.com> > + * > + */ > + > +#ifndef _MAILBOX_S10_H_ > +#define _MAILBOX_S10_H_ > + > +/* user define Uboot ID */ > +#define MBOX_CLIENT_ID_UBOOT 0xB > +#define MBOX_ID_UBOOT 0x1 > + > +#define MBOX_CMD_DIRECT 0 > +#define MBOX_CMD_INDIRECT 1 > + > +#define MBOX_MAX_CMD_INDEX 2047 > +#define MBOX_CMD_BUFFER_SIZE 32 > +#define MBOX_RESP_BUFFER_SIZE 16 > + > +#define MBOX_HDR_CMD_LSB 0 > +#define MBOX_HDR_CMD_MSK (BIT(11) - 1) > +#define MBOX_HDR_I_LSB 11 > +#define MBOX_HDR_I_MSK BIT(11) > +#define MBOX_HDR_LEN_LSB 12 > +#define MBOX_HDR_LEN_MSK 0x007FF000 > +#define MBOX_HDR_ID_LSB 24 > +#define MBOX_HDR_ID_MSK 0x0F000000 > +#define MBOX_HDR_CLIENT_LSB 28 > +#define MBOX_HDR_CLIENT_MSK 0xF0000000 > + > +/* Interrupt flags */ > +#define MBOX_FLAGS_INT_COE BIT(0) /* COUT update interrupt enable */ > +#define MBOX_FLAGS_INT_RIE BIT(1) /* RIN update interrupt enable */ > +#define MBOX_FLAGS_INT_UAE BIT(8) /* Urgent ACK interrupt enable */ > +#define MBOX_ALL_INTRS (MBOX_FLAGS_INT_COE | \ > + MBOX_FLAGS_INT_RIE | \ > + MBOX_FLAGS_INT_UAE) > + > +/* Status */ > +#define MBOX_STATUS_UA_MSK BIT(8) > + > +#define MBOX_CMD_HEADER(client, id, len, indirect, cmd) \ > + ((((cmd) << MBOX_HDR_CMD_LSB) & MBOX_HDR_CMD_MSK) | \ > + (((indirect) << MBOX_HDR_I_LSB) & MBOX_HDR_I_MSK) | \ > + (((len) << MBOX_HDR_LEN_LSB) & MBOX_HDR_LEN_MSK) | \ > + (((id) << MBOX_HDR_ID_LSB) & MBOX_HDR_ID_MSK) | \ > + (((client) << MBOX_HDR_CLIENT_LSB) & MBOX_HDR_CLIENT_MSK)) > + > +#define MBOX_RESP_ERR_GET(resp) \ > + (((resp) & MBOX_HDR_CMD_MSK) >> MBOX_HDR_CMD_LSB) > +#define MBOX_RESP_LEN_GET(resp) \ > + (((resp) & MBOX_HDR_LEN_MSK) >> MBOX_HDR_LEN_LSB) > +#define MBOX_RESP_ID_GET(resp) \ > + (((resp) & MBOX_HDR_ID_MSK) >> MBOX_HDR_ID_LSB) > +#define MBOX_RESP_CLIENT_GET(resp) \ > + (((resp) & MBOX_HDR_CLIENT_MSK) >> MBOX_HDR_CLIENT_LSB) > + > +/* Response error list */ > +enum ALT_SDM_MBOX_RESP_CODE { > + /* CMD completed successfully, but check resp ARGS for any errors */ > + MBOX_RESP_STATOK = 0, > + /* CMD is incorrectly formatted in some way */ > + MBOX_RESP_INVALID_COMMAND = 1, > + /* BootROM Command code not undesrtood */ > + MBOX_RESP_UNKNOWN_BR = 2, > + /* CMD code not recognized by firmware */ > + MBOX_RESP_UNKNOWN = 3, > + /* Indicates that the device is not configured */ > + MBOX_RESP_NOT_CONFIGURED = 256, > + /* Indicates that the device is busy */ > + MBOX_RESP_DEVICE_BUSY = 0x1FF, > + /* Indicates that there is no valid response available */ > + MBOX_RESP_NO_VALID_RESP_AVAILABLE = 0x2FF, > + /* General Error */ > + MBOX_RESP_ERROR = 0x3FF, > +}; > + > +/* Mailbox command list */ > +#define MBOX_RESTART 2 > +#define MBOX_CONFIG_STATUS 4 > +#define MBOX_RECONFIG 6 > +#define MBOX_RECONFIG_MSEL 7 > +#define MBOX_RECONFIG_DATA 8 > +#define MBOX_RECONFIG_STATUS 9 > +#define MBOX_QSPI_OPEN 50 > +#define MBOX_QSPI_CLOSE 51 > +#define MBOX_QSPI_DIRECT 59 > +#define MBOX_REBOOT_HPS 71 > + > +struct socfpga_mailbox {
We should probably just use register offset macros in new code, this struct {} stuff often doesn't work too well and the limitations are showing. > + u32 cin; /* command valid offset */ > + u32 rout; /* response output offset */ > + u32 urg; /* urgent command */ > + u32 flags; /* interrupt enables */ > + u32 pad_0x10_0x1f[4]; /* 0x10 - 0x1F reserved */ > + u32 cout; /* command free offset */ > + u32 rin; /* respond valid offset */ > + u32 pad_0x28; /* 0x28 reserved */ > + u32 status; /* mailbox status */ > + u32 pad_0x30_0x3f[4]; /* 0x30 - 0x3F reserved */ > + u32 cmd_buf[MBOX_CMD_BUFFER_SIZE]; /* 0x40 - 0xBC circular command > + * buffer to SDM > + */ > + u32 resp_buf[MBOX_RESP_BUFFER_SIZE]; /* 0xC0 - 0xFF circular > + * response buffer > + */ > +}; > + > +/* Use define other than put into struct socfpga_mailbox to save spaces */ > +#define MBOX_DOORBELL_TO_SDM_REG (SOCFPGA_MAILBOX_ADDRESS + 0x400) > +#define MBOX_DOORBELL_FROM_SDM_REG (SOCFPGA_MAILBOX_ADDRESS + 0x480) > + > +/******** Status and bit information returned by RECONFIG_STATUS ********/ > +#define RECONFIG_STATUS_RESPONSE_LEN 6 > +#define RECONFIG_STATUS_STATE 0 > +#define RECONFIG_STATUS_PIN_STATUS 2 > +#define RECONFIG_STATUS_SOFTFUNC_STATUS 3 > + > +#define MBOX_CFGSTAT_STATE_IDLE 0x00000000 > +#define MBOX_CFGSTAT_STATE_CONFIG 0x10000000 > +#define MBOX_CFGSTAT_STATE_FAILACK 0x08000000 > +#define MBOX_CFGSTAT_STATE_ERROR_INVALID 0xf0000001 > +#define MBOX_CFGSTAT_STATE_ERROR_CORRUPT 0xf0000002 > +#define MBOX_CFGSTAT_STATE_ERROR_AUTH 0xf0000003 > +#define MBOX_CFGSTAT_STATE_ERROR_CORE_IO 0xf0000004 > +#define MBOX_CFGSTAT_STATE_ERROR_HARDWARE 0xf0000005 > +#define MBOX_CFGSTAT_STATE_ERROR_FAKE 0xf0000006 > +#define MBOX_CFGSTAT_STATE_ERROR_BOOT_INFO 0xf0000007 > +#define MBOX_CFGSTAT_STATE_ERROR_QSPI_ERROR 0xf0000008 > + > +#define RCF_SOFTFUNC_STATUS_CONF_DONE BIT(0) > +#define RCF_SOFTFUNC_STATUS_INIT_DONE BIT(1) > +#define RCF_SOFTFUNC_STATUS_SEU_ERROR BIT(3) > +#define RCF_PIN_STATUS_NSTATUS BIT(31) > +/************************************************************************/ > + > +int mbox_send_cmd(u8 id, u32 cmd, u8 is_indirect, u32 len, u32 *arg, u8 > urgent, > + u32 *resp_buf_len, u32 *resp_buf); > +int mbox_send_cmd_psci(u8 id, u32 cmd, u8 is_indirect, u32 len, u32 *arg, > + u8 urgent, u32 *resp_buf_len, u32 *resp_buf); > +int mbox_send_cmd_only(u8 id, u32 cmd, u8 is_indirect, u32 len, u32 *arg); > +int mbox_send_cmd_only_psci(u8 id, u32 cmd, u8 is_indirect, u32 len, u32 > *arg); > +int mbox_rcv_resp(u32 *resp_buf, u32 resp_buf_max_len); > +int mbox_rcv_resp_psci(u32 *resp_buf, u32 resp_buf_max_len); > +int mbox_init(void); > + > +#ifdef CONFIG_CADENCE_QSPI > +int mbox_qspi_close(void); > +int mbox_qspi_open(void); > +#endif > + > +int mbox_reset_cold(void); > + > +#endif /* _MAILBOX_S10_H_ */ > diff --git a/arch/arm/mach-socfpga/mailbox_s10.c > b/arch/arm/mach-socfpga/mailbox_s10.c > new file mode 100644 > index 0000000..ed713a9 > --- /dev/null > +++ b/arch/arm/mach-socfpga/mailbox_s10.c > @@ -0,0 +1,378 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * Copyright (C) 2017-2018 Intel Corporation <www.intel.com> > + * > + */ > + > +#include <common.h> > +#include <wait_bit.h> > +#include <asm/io.h> > +#include <asm/arch/mailbox_s10.h> > +#include <asm/arch/system_manager.h> > +#include <asm/secure.h> > + > +DECLARE_GLOBAL_DATA_PTR; > + > +static __always_inline int mbox_polling_resp(u32 rout) > +{ > + static const struct socfpga_mailbox *mbox_base = > + (void *)SOCFPGA_MAILBOX_ADDRESS; > + u32 rin; > + unsigned long i = ~0; > + > + while (i) { > + rin = readl(&mbox_base->rin); > + if (rout != rin) > + return 0; This looks like include/wait_bit.h reimplementation > + i--; > + } > + > + return -ETIMEDOUT; > +} > + > +/* Check for available slot and write to circular buffer. > + * It also update command valid offset (cin) register. > + */ > +static __always_inline int mbox_fill_cmd_circular_buff(u32 header, u32 len, > + u32 *arg) > +{ > + static const struct socfpga_mailbox *mbox_base = > + (void *)SOCFPGA_MAILBOX_ADDRESS; > + u32 cin; > + u32 cout; > + u32 i; > + > + cin = readl(&mbox_base->cin) % MBOX_CMD_BUFFER_SIZE; > + cout = readl(&mbox_base->cout) % MBOX_CMD_BUFFER_SIZE; > + > + /* if command buffer is full or not enough free space > + * to fit the data > + */ > + if (((cin + 1) % MBOX_CMD_BUFFER_SIZE) == cout || > + ((MBOX_CMD_BUFFER_SIZE - cin + cout - 1) % > + MBOX_CMD_BUFFER_SIZE) < len) > + return -ENOMEM; > + > + /* write header to circular buffer */ > + writel(header, &mbox_base->cmd_buf[cin++]); > + /* wrapping around when it reach the buffer size */ > + cin %= MBOX_CMD_BUFFER_SIZE; > + > + /* write arguments */ > + for (i = 0; i < len; i++) { > + writel(arg[i], &mbox_base->cmd_buf[cin++]); > + /* wrapping around when it reach the buffer size */ > + cin %= MBOX_CMD_BUFFER_SIZE; > + } > + > + /* write command valid offset */ > + writel(cin, &mbox_base->cin); > + > + return 0; > +} > + > +/* Check the command and fill it into circular buffer */ > +static __always_inline int mbox_prepare_cmd_only(u8 id, u32 cmd, > + u8 is_indirect, u32 len, > + u32 *arg) > +{ > + u32 header; > + int ret; > + > + /* Total length is command + argument length */ > + if ((len + 1) > MBOX_CMD_BUFFER_SIZE) > + return -EINVAL; > + > + if (cmd > MBOX_MAX_CMD_INDEX) > + return -EINVAL; > + > + header = MBOX_CMD_HEADER(MBOX_CLIENT_ID_UBOOT, id, len, > + (is_indirect) ? 1 : 0, cmd); > + > + ret = mbox_fill_cmd_circular_buff(header, len, arg); > + > + return ret; > +} > + > +/* Send command only without waiting for responses from SDM */ > +static __always_inline int __mbox_send_cmd_only(u8 id, u32 cmd, > + u8 is_indirect, u32 len, > + u32 *arg) > +{ > + int ret = mbox_prepare_cmd_only(id, cmd, is_indirect, len, arg); > + /* write doorbell */ > + writel(1, MBOX_DOORBELL_TO_SDM_REG); > + > + return ret; > +} > + > +/* Return number of responses received in buffer */ > +static __always_inline int __mbox_rcv_resp(u32 *resp_buf, u32 > resp_buf_max_len) __always_inline is nonsense, drop it. Let the compiler do it's thing. > +{ > + static const struct socfpga_mailbox *mbox_base = > + (void *)SOCFPGA_MAILBOX_ADDRESS; > + u32 rin; > + u32 rout; > + u32 resp_len = 0; > + > + /* clear doorbell from SDM if it was SET */ > + if (readl((const u32 *)MBOX_DOORBELL_FROM_SDM_REG) & 1) > + writel(0, MBOX_DOORBELL_FROM_SDM_REG); > + > + /* read current response offset */ > + rout = readl(&mbox_base->rout); > + /* read response valid offset */ > + rin = readl(&mbox_base->rin); > + > + while (rin != rout && (resp_len < resp_buf_max_len)) { > + /* Response received */ > + if (resp_buf) > + resp_buf[resp_len++] = > + readl(&mbox_base->resp_buf[rout]); > + rout++; > + /* wrapping around when it reach the buffer size */ > + rout %= MBOX_RESP_BUFFER_SIZE; > + /* update next ROUT */ > + writel(rout, &mbox_base->rout); > + } > + > + return resp_len; > +} > + > +/* Support one command and up to 31 words argument length only */ > +static __always_inline int __mbox_send_cmd(u8 id, u32 cmd, u8 is_indirect, > + u32 len, u32 *arg, u8 urgent, > + u32 *resp_buf_len, u32 *resp_buf) > +{ > + static const struct socfpga_mailbox *mbox_base = > + (void *)SOCFPGA_MAILBOX_ADDRESS; > + > + u32 rin; > + u32 resp; > + u32 rout; > + u32 status; > + u32 resp_len; > + u32 buf_len; > + int ret; > + > + ret = mbox_prepare_cmd_only(id, cmd, is_indirect, len, arg); > + if (ret) > + return ret; > + > + if (urgent) { > + /* Read status because it is toggled */ > + status = readl(&mbox_base->status) & MBOX_STATUS_UA_MSK; > + /* Send command as urgent command */ > + writel(1, &mbox_base->urg); > + } > + > + /* write doorbell */ > + writel(1, MBOX_DOORBELL_TO_SDM_REG); > + > + while (1) { > + ret = ~0; > + > + /* Wait for doorbell from SDM */ > + while (!readl(MBOX_DOORBELL_FROM_SDM_REG) && ret--) > + ; > + if (!ret) > + return -ETIMEDOUT; wait_for_bit... > + /* clear interrupt */ > + writel(0, MBOX_DOORBELL_FROM_SDM_REG); > + > + if (urgent) { > + u32 new_status = readl(&mbox_base->status); > + /* urgent command doesn't have response */ > + writel(0, &mbox_base->urg); > + /* Urgent ACK is toggled */ > + if ((new_status & MBOX_STATUS_UA_MSK) ^ status) > + return 0; > + > + return -ECOMM; > + } > + > + /* read current response offset */ > + rout = readl(&mbox_base->rout); > + > + /* read response valid offset */ > + rin = readl(&mbox_base->rin); > + > + if (rout != rin) { > + /* Response received */ > + resp = readl(&mbox_base->resp_buf[rout]); > + rout++; > + /* wrapping around when it reach the buffer size */ > + rout %= MBOX_RESP_BUFFER_SIZE; > + /* update next ROUT */ > + writel(rout, &mbox_base->rout); > + > + /* check client ID and ID */ > + if ((MBOX_RESP_CLIENT_GET(resp) == > MBOX_CLIENT_ID_UBOOT) && > + (MBOX_RESP_ID_GET(resp) == id)) { > + ret = MBOX_RESP_ERR_GET(resp); > + if (ret) > + return ret; > + > + if (resp_buf_len) { > + buf_len = *resp_buf_len; > + *resp_buf_len = 0; > + } else { > + buf_len = 0; > + } > + > + resp_len = MBOX_RESP_LEN_GET(resp); > + while (resp_len) { > + ret = mbox_polling_resp(rout); > + if (ret) > + return ret; > + /* we need to process response buffer > + * even caller doesn't need it > + */ > + resp = > readl(&mbox_base->resp_buf[rout]); > + rout++; > + resp_len--; > + rout %= MBOX_RESP_BUFFER_SIZE; > + writel(rout, &mbox_base->rout); > + if (buf_len) { > + /* copy response to buffer */ > + resp_buf[*resp_buf_len] = resp; > + (*resp_buf_len)++; > + buf_len--; > + } > + } > + return ret; > + } > + } > + }; > + > + return -EIO; > +} > + > +int mbox_init(void) > +{ > + static const struct socfpga_mailbox *mbox_base = > + (void *)SOCFPGA_MAILBOX_ADDRESS; > + int ret; > + > + /* enable mailbox interrupts */ > + writel(MBOX_ALL_INTRS, &mbox_base->flags); > + > + /* Ensure urgent request is cleared */ > + writel(0, &mbox_base->urg); > + > + /* Ensure the Doorbell Interrupt is cleared */ > + writel(0, MBOX_DOORBELL_FROM_SDM_REG); > + > + ret = mbox_send_cmd(MBOX_ID_UBOOT, MBOX_RESTART, MBOX_CMD_DIRECT, 0, > + NULL, 1, 0, NULL); > + if (ret) > + return ret; > + > + /* Renable mailbox interrupts after MBOX_RESTART */ > + writel(MBOX_ALL_INTRS, &mbox_base->flags); > + > + return 0; > +} > + > +#ifdef CONFIG_CADENCE_QSPI > +int mbox_qspi_close(void) > +{ > + return mbox_send_cmd(MBOX_ID_UBOOT, MBOX_QSPI_CLOSE, MBOX_CMD_DIRECT, > + 0, NULL, 0, 0, NULL); > +} > + > +int mbox_qspi_open(void) > +{ > + static const struct socfpga_system_manager *sysmgr_regs = > + (struct socfpga_system_manager *)SOCFPGA_SYSMGR_ADDRESS; > + > + int ret; > + u32 resp_buf[1]; > + u32 resp_buf_len; > + > + ret = mbox_send_cmd(MBOX_ID_UBOOT, MBOX_QSPI_OPEN, MBOX_CMD_DIRECT, > + 0, NULL, 0, 0, NULL); > + if (ret) { > + /* retry again by closing and reopen the QSPI again */ > + ret = mbox_qspi_close(); > + if (ret) > + return ret; > + > + ret = mbox_send_cmd(MBOX_ID_UBOOT, MBOX_QSPI_OPEN, > + MBOX_CMD_DIRECT, 0, NULL, 0, 0, NULL); > + if (ret) > + return ret; > + } > + > + /* HPS will directly control the QSPI controller, no longer mailbox */ > + resp_buf_len = 1; > + ret = mbox_send_cmd(MBOX_ID_UBOOT, MBOX_QSPI_DIRECT, MBOX_CMD_DIRECT, > + 0, NULL, 0, (u32 *)&resp_buf_len, > + (u32 *)&resp_buf); > + if (ret) > + goto error; > + > + /* We are getting QSPI ref clock and set into sysmgr boot register */ > + printf("QSPI: Reference clock at %d Hz\n", resp_buf[0]); Certainly something I can get out of clock or clk command, drop the print. > + writel(resp_buf[0], &sysmgr_regs->boot_scratch_cold0); > + > + return 0; > + > +error: > + mbox_qspi_close(); > + > + return ret; > +} > +#endif /* CONFIG_CADENCE_QSPI */ > + > +int mbox_reset_cold(void) > +{ > + int ret; > + > + ret = mbox_send_cmd(MBOX_ID_UBOOT, MBOX_REBOOT_HPS, MBOX_CMD_DIRECT, > + 0, NULL, 0, 0, NULL); > + if (ret) { > + /* mailbox sent failure, wait for watchdog to kick in */ > + while (1) > + ; Is this supposed to be hang() ? > + } > + return 0; > +} > + > +int mbox_send_cmd(u8 id, u32 cmd, u8 is_indirect, u32 len, u32 *arg, > + u8 urgent, u32 *resp_buf_len, u32 *resp_buf) > +{ > + return __mbox_send_cmd(id, cmd, is_indirect, len, arg, urgent, > + resp_buf_len, resp_buf); > +} __anything is reserved for compiler, drop the leading underscores > -- Best regards, Marek Vasut _______________________________________________ U-Boot mailing list U-Boot@lists.denx.de https://lists.denx.de/listinfo/u-boot