mirror of
				https://xff.cz/git/u-boot/
				synced 2025-10-22 10:31:56 +02:00 
			
		
		
		
	arm: socfpga: stratix10: Add mailbox support for Stratix10 SoC
Add mailbox support for Stratix SoC Signed-off-by: Ley Foon Tan <ley.foon.tan@intel.com> Signed-off-by: Chin Liang See <chin.liang.see@intel.com> Reviewed-by: Marek Vasut <marex@denx.de>
This commit is contained in:
		
				
					committed by
					
						 Marek Vasut
						Marek Vasut
					
				
			
			
				
	
			
			
			
						parent
						
							d559130e36
						
					
				
				
					commit
					a280e9db64
				
			| @@ -30,6 +30,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 | ||||
|   | ||||
							
								
								
									
										144
									
								
								arch/arm/mach-socfpga/include/mach/mailbox_s10.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										144
									
								
								arch/arm/mach-socfpga/include/mach/mailbox_s10.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,144 @@ | ||||
| /* 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 | ||||
|  | ||||
| /* Mailbox registers */ | ||||
| #define MBOX_CIN			0	/* command valid offset */ | ||||
| #define MBOX_ROUT			4	/* response output offset */ | ||||
| #define MBOX_URG			8	/* urgent command */ | ||||
| #define MBOX_FLAGS			0x0c	/* interrupt enables */ | ||||
| #define MBOX_COUT			0x20	/* command free offset */ | ||||
| #define MBOX_RIN			0x24	/* respond valid offset */ | ||||
| #define MBOX_STATUS			0x2c	/* mailbox status */ | ||||
| #define MBOX_CMD_BUF			0x40	/* circular command buffer */ | ||||
| #define MBOX_RESP_BUF			0xc0	/* circular response buffer */ | ||||
| #define MBOX_DOORBELL_TO_SDM		0x400	/* Doorbell to SDM */ | ||||
| #define MBOX_DOORBELL_FROM_SDM		0x480	/* Doorbell from SDM */ | ||||
|  | ||||
| /* 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_ */ | ||||
							
								
								
									
										380
									
								
								arch/arm/mach-socfpga/mailbox_s10.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										380
									
								
								arch/arm/mach-socfpga/mailbox_s10.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,380 @@ | ||||
| // 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; | ||||
|  | ||||
| #define MBOX_READL(reg)			\ | ||||
| 	 readl(SOCFPGA_MAILBOX_ADDRESS + (reg)) | ||||
|  | ||||
| #define MBOX_WRITEL(data, reg)		\ | ||||
| 	writel(data, SOCFPGA_MAILBOX_ADDRESS + (reg)) | ||||
|  | ||||
| #define MBOX_READ_RESP_BUF(rout)	\ | ||||
| 	MBOX_READL(MBOX_RESP_BUF + ((rout) * sizeof(u32))) | ||||
|  | ||||
| #define MBOX_WRITE_CMD_BUF(data, cin)	\ | ||||
| 	MBOX_WRITEL(data, MBOX_CMD_BUF + ((cin) * sizeof(u32))) | ||||
|  | ||||
| static __always_inline int mbox_polling_resp(u32 rout) | ||||
| { | ||||
| 	u32 rin; | ||||
| 	unsigned long i = ~0; | ||||
|  | ||||
| 	while (i) { | ||||
| 		rin = MBOX_READL(MBOX_RIN); | ||||
| 		if (rout != rin) | ||||
| 			return 0; | ||||
|  | ||||
| 		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) | ||||
| { | ||||
| 	u32 cin; | ||||
| 	u32 cout; | ||||
| 	u32 i; | ||||
|  | ||||
| 	cin = MBOX_READL(MBOX_CIN) % MBOX_CMD_BUFFER_SIZE; | ||||
| 	cout = MBOX_READL(MBOX_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 */ | ||||
| 	MBOX_WRITE_CMD_BUF(header, cin++); | ||||
| 	/* wrapping around when it reach the buffer size */ | ||||
| 	cin %= MBOX_CMD_BUFFER_SIZE; | ||||
|  | ||||
| 	/* write arguments */ | ||||
| 	for (i = 0; i < len; i++) { | ||||
| 		MBOX_WRITE_CMD_BUF(arg[i], cin++); | ||||
| 		/* wrapping around when it reach the buffer size */ | ||||
| 		cin %= MBOX_CMD_BUFFER_SIZE; | ||||
| 	} | ||||
|  | ||||
| 	/* write command valid offset */ | ||||
| 	MBOX_WRITEL(cin, MBOX_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_common(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 */ | ||||
| 	MBOX_WRITEL(1, MBOX_DOORBELL_TO_SDM); | ||||
|  | ||||
| 	return ret; | ||||
| } | ||||
|  | ||||
| /* Return number of responses received in buffer */ | ||||
| static __always_inline int __mbox_rcv_resp(u32 *resp_buf, u32 resp_buf_max_len) | ||||
| { | ||||
| 	u32 rin; | ||||
| 	u32 rout; | ||||
| 	u32 resp_len = 0; | ||||
|  | ||||
| 	/* clear doorbell from SDM if it was SET */ | ||||
| 	if (MBOX_READL(MBOX_DOORBELL_FROM_SDM) & 1) | ||||
| 		MBOX_WRITEL(0, MBOX_DOORBELL_FROM_SDM); | ||||
|  | ||||
| 	/* read current response offset */ | ||||
| 	rout = MBOX_READL(MBOX_ROUT); | ||||
| 	/* read response valid offset */ | ||||
| 	rin = MBOX_READL(MBOX_RIN); | ||||
|  | ||||
| 	while (rin != rout && (resp_len < resp_buf_max_len)) { | ||||
| 		/* Response received */ | ||||
| 		if (resp_buf) | ||||
| 			resp_buf[resp_len++] = MBOX_READ_RESP_BUF(rout); | ||||
|  | ||||
| 		rout++; | ||||
| 		/* wrapping around when it reach the buffer size */ | ||||
| 		rout %= MBOX_RESP_BUFFER_SIZE; | ||||
| 		/* update next ROUT */ | ||||
| 		MBOX_WRITEL(rout, MBOX_ROUT); | ||||
| 	} | ||||
|  | ||||
| 	return resp_len; | ||||
| } | ||||
|  | ||||
| /* Support one command and up to 31 words argument length only */ | ||||
| static __always_inline int mbox_send_cmd_common(u8 id, u32 cmd, u8 is_indirect, | ||||
| 						u32 len, u32 *arg, u8 urgent, | ||||
| 						u32 *resp_buf_len, | ||||
| 						u32 *resp_buf) | ||||
| { | ||||
| 	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 = MBOX_READL(MBOX_STATUS) & MBOX_STATUS_UA_MSK; | ||||
| 		/* Send command as urgent command */ | ||||
| 		MBOX_WRITEL(1, MBOX_URG); | ||||
| 	} | ||||
|  | ||||
| 	/* write doorbell */ | ||||
| 	MBOX_WRITEL(1, MBOX_DOORBELL_TO_SDM); | ||||
|  | ||||
| 	while (1) { | ||||
| 		ret = ~0; | ||||
|  | ||||
| 		/* Wait for doorbell from SDM */ | ||||
| 		while (!MBOX_READL(MBOX_DOORBELL_FROM_SDM) && ret--) | ||||
| 			; | ||||
| 		if (!ret) | ||||
| 			return -ETIMEDOUT; | ||||
|  | ||||
| 		/* clear interrupt */ | ||||
| 		MBOX_WRITEL(0, MBOX_DOORBELL_FROM_SDM); | ||||
|  | ||||
| 		if (urgent) { | ||||
| 			u32 new_status = MBOX_READL(MBOX_STATUS); | ||||
| 			/* urgent command doesn't have response */ | ||||
| 			MBOX_WRITEL(0, MBOX_URG); | ||||
| 			/* Urgent ACK is toggled */ | ||||
| 			if ((new_status & MBOX_STATUS_UA_MSK) ^ status) | ||||
| 				return 0; | ||||
|  | ||||
| 			return -ECOMM; | ||||
| 		} | ||||
|  | ||||
| 		/* read current response offset */ | ||||
| 		rout = MBOX_READL(MBOX_ROUT); | ||||
|  | ||||
| 		/* read response valid offset */ | ||||
| 		rin = MBOX_READL(MBOX_RIN); | ||||
|  | ||||
| 		if (rout != rin) { | ||||
| 			/* Response received */ | ||||
| 			resp = MBOX_READ_RESP_BUF(rout); | ||||
| 			rout++; | ||||
| 			/* wrapping around when it reach the buffer size */ | ||||
| 			rout %= MBOX_RESP_BUFFER_SIZE; | ||||
| 			/* update next ROUT */ | ||||
| 			MBOX_WRITEL(rout, MBOX_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 = MBOX_READ_RESP_BUF(rout); | ||||
| 					rout++; | ||||
| 					resp_len--; | ||||
| 					rout %= MBOX_RESP_BUFFER_SIZE; | ||||
| 					MBOX_WRITEL(rout, MBOX_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) | ||||
| { | ||||
| 	int ret; | ||||
|  | ||||
| 	/* enable mailbox interrupts */ | ||||
| 	MBOX_WRITEL(MBOX_ALL_INTRS, MBOX_FLAGS); | ||||
|  | ||||
| 	/* Ensure urgent request is cleared */ | ||||
| 	MBOX_WRITEL(0, MBOX_URG); | ||||
|  | ||||
| 	/* Ensure the Doorbell Interrupt is cleared */ | ||||
| 	MBOX_WRITEL(0, MBOX_DOORBELL_FROM_SDM); | ||||
|  | ||||
| 	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 */ | ||||
| 	MBOX_WRITEL(MBOX_ALL_INTRS, MBOX_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]); | ||||
| 	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 */ | ||||
| 		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_common(id, cmd, is_indirect, len, arg, urgent, | ||||
| 			       resp_buf_len, resp_buf); | ||||
| } | ||||
|  | ||||
| int __secure mbox_send_cmd_psci(u8 id, u32 cmd, u8 is_indirect, u32 len, | ||||
| 				u32 *arg, u8 urgent, u32 *resp_buf_len, | ||||
| 				u32 *resp_buf) | ||||
| { | ||||
| 	return mbox_send_cmd_common(id, cmd, is_indirect, len, arg, urgent, | ||||
| 			       resp_buf_len, resp_buf); | ||||
| } | ||||
|  | ||||
| int mbox_send_cmd_only(u8 id, u32 cmd, u8 is_indirect, u32 len, u32 *arg) | ||||
| { | ||||
| 	return mbox_send_cmd_only_common(id, cmd, is_indirect, len, arg); | ||||
| } | ||||
|  | ||||
| int __secure mbox_send_cmd_only_psci(u8 id, u32 cmd, u8 is_indirect, u32 len, | ||||
| 				     u32 *arg) | ||||
| { | ||||
| 	return mbox_send_cmd_only_common(id, cmd, is_indirect, len, arg); | ||||
| } | ||||
|  | ||||
| int mbox_rcv_resp(u32 *resp_buf, u32 resp_buf_max_len) | ||||
| { | ||||
| 	return __mbox_rcv_resp(resp_buf, resp_buf_max_len); | ||||
| } | ||||
|  | ||||
| int __secure mbox_rcv_resp_psci(u32 *resp_buf, u32 resp_buf_max_len) | ||||
| { | ||||
| 	return __mbox_rcv_resp(resp_buf, resp_buf_max_len); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user