summaryrefslogtreecommitdiff
path: root/drivers/ddr/altera/uibssm_mailbox.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/ddr/altera/uibssm_mailbox.c')
-rw-r--r--drivers/ddr/altera/uibssm_mailbox.c321
1 files changed, 321 insertions, 0 deletions
diff --git a/drivers/ddr/altera/uibssm_mailbox.c b/drivers/ddr/altera/uibssm_mailbox.c
new file mode 100644
index 00000000000..deef61f1178
--- /dev/null
+++ b/drivers/ddr/altera/uibssm_mailbox.c
@@ -0,0 +1,321 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (C) 2025 Altera Corporation <www.altera.com>
+ */
+
+#include <hang.h>
+#include <wait_bit.h>
+#include <asm/io.h>
+#include <linux/bitfield.h>
+#include "uibssm_mailbox.h"
+
+#define MAX_RETRIES 3
+
+int uib_bist_mem_init_start(struct uib_info *uib_ctrl)
+{
+ struct uib_mb_resp usr_resp;
+ bool bist_start, bist_success;
+ u32 start;
+
+ for (int i = 0; i < uib_ctrl->num_instance; i++) {
+ bist_start = false;
+ bist_success = false;
+
+ /*
+ * Full memory initialization BIST performed on all UIB channels
+ * start memory initialization BIST on full memory address
+ */
+ uib_mb_req(uib_ctrl->uib[i].uib_csr_addr,
+ UIB_CMD_TRIG_CONTROLLER_OP,
+ UIB_BIST_MEM_INIT_START,
+ UIB_BIST_FULL_MEM, &usr_resp);
+
+ bist_start = UIBSSM_CMD_RESPONSE_DATA_SHORT(usr_resp.cmd_resp_status) &
+ UIB_BIST_INITIATE_PASS;
+
+ if (!bist_start) {
+ printf("%s: Failed to initialize memory on UIB instance %d\n",
+ __func__, i);
+
+ return -EINVAL;
+ }
+
+ /* Polling for the initiated memory initialization BIST status */
+ start = get_timer(0);
+ while (!bist_success) {
+ udelay(1);
+
+ /*
+ * cmd_param_0 is not used in BIST status request,
+ * hence set the value to 0
+ */
+ uib_mb_req(uib_ctrl->uib[i].uib_csr_addr,
+ UIB_CMD_TRIG_CONTROLLER_OP,
+ UIB_BIST_MEM_INIT_STATUS,
+ 0, &usr_resp);
+
+ bist_success = UIBSSM_CMD_RESPONSE_DATA_SHORT(usr_resp.cmd_resp_status) &
+ BIT(0);
+
+ if (!bist_success && (get_timer(start) > TIMEOUT)) {
+ printf("%s: Timeout initializing memory on UIB instance %d\n",
+ __func__, i);
+
+ return -ETIMEDOUT;
+ }
+ }
+
+ debug("%s: Memory initialized successfully on UIB instance %d\n", __func__, i);
+ }
+
+ return 0;
+}
+
+int uib_cal_status(phys_addr_t addr)
+{
+ int ret = 0;
+ phys_addr_t status_addr = addr + UIB_R_INITSTS_OFFSET;
+
+ /* Ensure calibration completed */
+ ret = wait_for_bit_le32((const void *)status_addr, UIB_R_INITSTS_INITSTS_PASS, true,
+ TIMEOUT, false);
+ if (ret)
+ printf("%s: HBM calibration UIB instance 0x%llx timeout\n", __func__, status_addr);
+
+ return ret;
+}
+
+void uib_init_mem_cal(struct uib_info *uib_ctrl)
+{
+ int i, ret;
+
+ if (!uib_ctrl->num_instance) {
+ uib_ctrl->overall_cal_status = false;
+ } else {
+ uib_ctrl->overall_cal_status = true;
+
+ /* Check initial calibration status for the assigned UIB */
+ for (i = 0; i < uib_ctrl->num_instance; i++) {
+ ret = uib_cal_status(uib_ctrl->uib[i].uib_csr_addr);
+ if (ret) {
+ uib_ctrl->uib[i].cal_status = false;
+ uib_ctrl->overall_cal_status = false;
+
+ printf("%s: Initial HBM calibration UIB_%d failed\n", __func__, i);
+ break;
+ }
+
+ uib_ctrl->uib[i].cal_status = true;
+
+ debug("%s: Initial HBM calibration UIB_%d succeed\n", __func__, i);
+ }
+ }
+}
+
+/* Trying 3 times re-calibration if initial calibration failed */
+void uib_trig_mem_cal(struct uib_info *uib_ctrl)
+{
+ int i, j, cal_stat;
+
+ if (!uib_ctrl->num_instance) {
+ uib_ctrl->overall_cal_status = false;
+ } else {
+ uib_ctrl->overall_cal_status = true;
+
+ for (i = 0; i < uib_ctrl->num_instance; i++) {
+ uib_ctrl->uib[i].cal_status = false;
+
+ /* Initiate Re-calibration */
+ for (j = 0; j < MAX_RETRIES; j++) {
+ clrsetbits_le32(uib_ctrl->uib[i].uib_csr_addr +
+ UIB_R_INITCTL_OFFSET,
+ UIB_R_INITCTL_INITTYPE_MASK |
+ UIB_R_INITCTL_INITREQ_MASK,
+ UIB_R_INITCTL_INITTYPE(UIB_RST_REQUEST_WITH_CAL) |
+ UIB_R_INITCTL_INITREQ(1));
+
+ cal_stat = uib_cal_status(uib_ctrl->uib[i].uib_csr_addr);
+ if (cal_stat)
+ continue;
+
+ debug("%s: HBM re-calibration UIB_%d succeed\n", __func__, i);
+
+ uib_ctrl->uib[i].cal_status = true;
+ break;
+ }
+
+ if (!uib_ctrl->uib[i].cal_status) {
+ uib_ctrl->overall_cal_status = false;
+
+ printf("%s: HBM re-calibration UIB_%d failed\n", __func__, i);
+ break;
+ }
+ }
+ }
+}
+
+static void uib_mailbox_write_request(u32 data, u32 target_write_addr, phys_addr_t csr_addr)
+{
+ int ret;
+
+ /*
+ * Read from chms0020 MBWRADDR_VALID and ensure its not set or
+ * wait till it get cleared by controller
+ */
+ debug("%s: #1 Read MBWRADDR_VALID from UIB_R_MBWRCTL\n", __func__);
+ ret = wait_for_bit_le32((const void *)csr_addr + UIB_R_MBWRCTL,
+ UIB_R_MBWRCTL_MBWRADDR_VALID, false, TIMEOUT, false);
+ if (ret) {
+ printf("%s: TIMEOUT!!! MBWRADDR_VALID is not zero\n", __func__);
+
+ hang();
+ }
+
+ /* Write <target write address> to chms0024 MBWRADDR */
+ debug("%s: #2 Write 0x%x to UIB_R_MBWRADDR\n", __func__, target_write_addr);
+ writel(target_write_addr, csr_addr + UIB_R_MBWRADDR);
+
+ /*
+ * Write 1 to chms0020 MBWRADDR_VALID to indicate the address is now valid
+ * for FW to read
+ */
+ debug("%s: #3 Write 1 to MBWRADDR_VALID for FW to read address\n", __func__);
+ setbits_le32(csr_addr + UIB_R_MBWRCTL, UIB_R_MBWRCTL_MBWRADDR_VALID);
+
+ /*
+ * Read from chms0020 MBWRDATA_VALID and ensure its not set or
+ * wait till it get cleared by controller
+ */
+ debug("%s: #4 Read MBWRDATA_VALID from UIB_R_MBWRCTL\n", __func__);
+ ret = wait_for_bit_le32((const void *)csr_addr + UIB_R_MBWRCTL,
+ UIB_R_MBWRCTL_MBWRDATA_VALID, false, TIMEOUT, false);
+ if (ret) {
+ printf("%s: TIMEOUT!!! MBWRADDR_VALID is not zero\n", __func__);
+
+ hang();
+ }
+
+ /*
+ * Read from chms0020 MBWRDATA_END and ensure its not set or
+ * wait till it get cleared by controller
+ */
+ debug("%s: #5 Read R_MBWRCTL_MBWRDATA_END from UIB_R_MBWRCTL\n", __func__);
+ ret = wait_for_bit_le32((const void *)csr_addr + UIB_R_MBWRCTL,
+ UIB_R_MBWRCTL_MBWRDATA_END, false, TIMEOUT, false);
+ if (ret) {
+ printf("%s: TIMEOUT!!! MBWRDATA_END is not zero\n", __func__);
+
+ hang();
+ }
+
+ /* Write <write data> to chms0028 MMR_MBWRDATA */
+ debug("%s: #6 Write 0x%x to UIB_R_MBWRDATA\n", __func__, data);
+ writel(data, csr_addr + UIB_R_MBWRDATA);
+
+ /*
+ * Write 1 to chms0020 MBWRDATA_END to indicate if the <write data> is the last burst
+ * for FW to read for the <target write address>
+ */
+ debug("%s: #7 Write 1 to MBWRDATA_END to inform FW this is last burst of data to read\n",
+ __func__);
+ setbits_le32(csr_addr + UIB_R_MBWRCTL, UIB_R_MBWRCTL_MBWRDATA_END);
+
+ /* Write 1 to chms0020 MBWRDATA_VALID to indicate the data is now valid for FW to read */
+ debug("%s: #8 Write 1 to MBWRDATA_VALID for FW to read data\n", __func__);
+ setbits_le32(csr_addr + UIB_R_MBWRCTL, UIB_R_MBWRCTL_MBWRDATA_VALID);
+}
+
+static u32 uib_mailbox_read_request(u32 target_read_addr, phys_addr_t csr_addr)
+{
+ int ret;
+ u32 reg, rd_data = 0;
+
+ /*
+ * Read from chms0030 MBRDADDR_VALID and ensure its not set or
+ * wait till it get cleared by controller
+ */
+ debug("%s: #1 Read MBRDADDR_VALID from UIB_R_MBRDCTL\n", __func__);
+ ret = wait_for_bit_le32((const void *)csr_addr + UIB_R_MBRDCTL,
+ UIB_R_MBRDCTL_MBRDADDR_VALID, false, TIMEOUT, false);
+ if (ret) {
+ printf("%s: TIMEOUT!!! MBRDADDR_VALID is not zero\n", __func__);
+
+ hang();
+ }
+
+ /* Write <target read address> to chms0034 MBRDADDR */
+ debug("%s: #2 Write 0x%x to UIB_R_MBRDADDR\n", __func__, target_read_addr);
+ writel(target_read_addr, csr_addr + UIB_R_MBRDADDR);
+
+ /*
+ * Write 1 to chms0030 MBRDADDR_VALID to indicate the address is now valid
+ * for FW to read
+ */
+ debug("%s: #3 Write 1 to MBRDADDR_VALID for FW to read address\n", __func__);
+ setbits_le32(csr_addr + UIB_R_MBRDCTL, UIB_R_MBRDCTL_MBRDADDR_VALID);
+
+ /*
+ * Continuously poll the chms0030 MBRDDATA_VALID. If MBRDDATA_VALID are set, read
+ * chms0038 MBRDDATA and chms0030 MBRDDATA_END to retrieve the <read data> and
+ * <end of read burst> status accordingly
+ */
+ debug("%s: #4 Read MBRDDATA_VALID from UIB_R_MBRDCTL\n", __func__);
+ ret = wait_for_bit_le32((const void *)csr_addr + UIB_R_MBRDCTL,
+ UIB_R_MBRDCTL_MBRDDATA_VALID, true, TIMEOUT, false);
+ if (ret) {
+ printf("%s: TIMEOUT!!! MBRDDATA_VALID is zero\n", __func__);
+
+ hang();
+ }
+
+ reg = readl(csr_addr + UIB_R_MBRRDATA);
+ debug("%s: #5 Read data from UIB_R_MBRRDATA = 0x%x\n", __func__, reg);
+ rd_data = reg;
+
+ reg = readl(csr_addr + UIB_R_MBRDCTL);
+ debug("%s: #6 Read end of read burst status from UIB_R_MBRDCTL = 0x%x\n", __func__, reg);
+
+ /*
+ * Once done retrieving the data, write 1 to chms0030 MBRDDATA_VALID,
+ * chms0030 MBRDDATA_END to clear the register
+ */
+ debug("%s: #7 Write 1 to MBRDDATA_VALID for FW to read address\n", __func__);
+ setbits_le32(csr_addr + UIB_R_MBRDCTL, UIB_R_MBRDCTL_MBRDDATA_VALID |
+ UIB_R_MBWRCTL_MBWRDATA_END);
+
+ return rd_data;
+}
+
+int uib_mb_req(phys_addr_t uib_csr_addr, u32 usr_cmd_type, u32 usr_cmd_opcode,
+ u32 cmd_param_0, struct uib_mb_resp *resp)
+{
+ u32 cmd_req;
+
+ /* Initialized zeros for responses */
+ resp->cmd_resp_status = 0;
+
+ /* Write CMD_REQ (CMD_TYPE and CMD_OPCODE) */
+ cmd_req = FIELD_PREP(CMD_TYPE_MASK, usr_cmd_type) |
+ FIELD_PREP(CMD_OPCODE_MASK, usr_cmd_opcode);
+ uib_mailbox_write_request(cmd_req, 0, uib_csr_addr);
+
+ debug("%s: Write 0x%x to UIBSSM_CMD_REQ_OFFSET 0x%llx\n", __func__, cmd_req, uib_csr_addr);
+
+ /* Write CMD_PARAM_* */
+ if (cmd_param_0)
+ uib_mailbox_write_request(cmd_param_0, 0, uib_csr_addr);
+ else
+ debug("%s: cmd_param_0 is NULL\n", __func__);
+
+ /* read CMD_RESPONSE_STATUS */
+ resp->cmd_resp_status = uib_mailbox_read_request(0, uib_csr_addr);
+
+ debug("%s: CMD_RESPONSE_STATUS 0x%llx: 0x%x\n", __func__,
+ uib_csr_addr, resp->cmd_resp_status);
+ debug("%s: STATUS_CMD_RESPONSE_ERROR: 0x%lx\n", __func__,
+ UIBSSM_CMD_RESPONSE_ERROR(resp->cmd_resp_status));
+ debug("%s: STATUS_GENERAL_ERROR: 0x%lx\n", __func__,
+ UIBSSM_GENERAL_ERROR(resp->cmd_resp_status));
+
+ return 0;
+}