xref: /rk3399_ARM-atf/drivers/cfi/v2m/v2m_flash.c (revision 76ce1028b0e0ef27e3aede9ad0fe43ca5af2b6c5)
1aa7877c4SAntonio Nino Diaz /*
2*2fe75a2dSZelalem  * Copyright (c) 2015-2020, ARM Limited and Contributors. All rights reserved.
3aa7877c4SAntonio Nino Diaz  *
4aa7877c4SAntonio Nino Diaz  * SPDX-License-Identifier: BSD-3-Clause
5aa7877c4SAntonio Nino Diaz  */
6aa7877c4SAntonio Nino Diaz 
7aa7877c4SAntonio Nino Diaz #include <errno.h>
809d40e0eSAntonio Nino Diaz 
909d40e0eSAntonio Nino Diaz #include <drivers/cfi/v2m_flash.h>
1009d40e0eSAntonio Nino Diaz #include <lib/mmio.h>
11aa7877c4SAntonio Nino Diaz 
12aa7877c4SAntonio Nino Diaz /*
13aa7877c4SAntonio Nino Diaz  * This file supplies a low level interface to the vexpress NOR flash
14aa7877c4SAntonio Nino Diaz  * memory of juno and fvp. This memory is organized as an interleaved
15aa7877c4SAntonio Nino Diaz  * memory of two chips with a 16 bit word. It means that every 32 bit
16aa7877c4SAntonio Nino Diaz  * access is going to access to two different chips. This is very
17aa7877c4SAntonio Nino Diaz  * important when we send commands or read status of the chips.
18aa7877c4SAntonio Nino Diaz  */
19aa7877c4SAntonio Nino Diaz 
20aa7877c4SAntonio Nino Diaz /*
21aa7877c4SAntonio Nino Diaz  * DWS ready poll retries. The number of retries in this driver have been
22aa7877c4SAntonio Nino Diaz  * obtained empirically from Juno. FVP implements a zero wait state NOR flash
23aa7877c4SAntonio Nino Diaz  * model
24aa7877c4SAntonio Nino Diaz  */
25aa7877c4SAntonio Nino Diaz #define DWS_WORD_PROGRAM_RETRIES	1000
26aa7877c4SAntonio Nino Diaz #define DWS_WORD_ERASE_RETRIES		3000000
27aa7877c4SAntonio Nino Diaz #define DWS_WORD_LOCK_RETRIES		1000
28aa7877c4SAntonio Nino Diaz 
29aa7877c4SAntonio Nino Diaz /* Helper macro to detect end of command */
30*2fe75a2dSZelalem #define NOR_CMD_END (NOR_DWS | (NOR_DWS << 16l))
31aa7877c4SAntonio Nino Diaz 
32aa7877c4SAntonio Nino Diaz /* Helper macros to access two flash banks in parallel */
33aa7877c4SAntonio Nino Diaz #define NOR_2X16(d)			((d << 16) | (d & 0xffff))
34aa7877c4SAntonio Nino Diaz 
nor_status(uintptr_t base_addr)35aa7877c4SAntonio Nino Diaz static unsigned int nor_status(uintptr_t base_addr)
36aa7877c4SAntonio Nino Diaz {
37aa7877c4SAntonio Nino Diaz 	unsigned long status;
38aa7877c4SAntonio Nino Diaz 
39aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_READ_STATUS_REG);
40aa7877c4SAntonio Nino Diaz 	status = mmio_read_32(base_addr);
41aa7877c4SAntonio Nino Diaz 	status |= status >> 16; /* merge status from both flash banks */
42aa7877c4SAntonio Nino Diaz 
43aa7877c4SAntonio Nino Diaz 	return status & 0xFFFF;
44aa7877c4SAntonio Nino Diaz }
45aa7877c4SAntonio Nino Diaz 
46aa7877c4SAntonio Nino Diaz /*
47aa7877c4SAntonio Nino Diaz  * Poll Write State Machine.
48aa7877c4SAntonio Nino Diaz  * Return values:
49aa7877c4SAntonio Nino Diaz  *    0      = WSM ready
50aa7877c4SAntonio Nino Diaz  *    -EBUSY = WSM busy after the number of retries
51aa7877c4SAntonio Nino Diaz  */
nor_poll_dws(uintptr_t base_addr,unsigned long int retries)52aa7877c4SAntonio Nino Diaz static int nor_poll_dws(uintptr_t base_addr, unsigned long int retries)
53aa7877c4SAntonio Nino Diaz {
54aa7877c4SAntonio Nino Diaz 	unsigned long status;
55aa7877c4SAntonio Nino Diaz 
56aa7877c4SAntonio Nino Diaz 	do {
57aa7877c4SAntonio Nino Diaz 		nor_send_cmd(base_addr, NOR_CMD_READ_STATUS_REG);
58aa7877c4SAntonio Nino Diaz 		status = mmio_read_32(base_addr);
59aa7877c4SAntonio Nino Diaz 		if ((status & NOR_CMD_END) == NOR_CMD_END)
60aa7877c4SAntonio Nino Diaz 			return 0;
61aa7877c4SAntonio Nino Diaz 	} while (retries-- > 0);
62aa7877c4SAntonio Nino Diaz 
63aa7877c4SAntonio Nino Diaz 	return -EBUSY;
64aa7877c4SAntonio Nino Diaz }
65aa7877c4SAntonio Nino Diaz 
66aa7877c4SAntonio Nino Diaz /*
67aa7877c4SAntonio Nino Diaz  * Return values:
68aa7877c4SAntonio Nino Diaz  *    0      = success
69aa7877c4SAntonio Nino Diaz  *    -EPERM = Device protected or Block locked
70aa7877c4SAntonio Nino Diaz  *    -EIO   = General I/O error
71aa7877c4SAntonio Nino Diaz  */
nor_full_status_check(uintptr_t base_addr)72aa7877c4SAntonio Nino Diaz static int nor_full_status_check(uintptr_t base_addr)
73aa7877c4SAntonio Nino Diaz {
74aa7877c4SAntonio Nino Diaz 	unsigned long status;
75aa7877c4SAntonio Nino Diaz 
76aa7877c4SAntonio Nino Diaz 	/* Full status check */
77aa7877c4SAntonio Nino Diaz 	status = nor_status(base_addr);
78aa7877c4SAntonio Nino Diaz 
79aa7877c4SAntonio Nino Diaz 	if (status & (NOR_PS | NOR_BLS | NOR_ESS | NOR_PSS))
80aa7877c4SAntonio Nino Diaz 		return -EPERM;
81aa7877c4SAntonio Nino Diaz 	if (status & (NOR_VPPS | NOR_ES))
82aa7877c4SAntonio Nino Diaz 		return -EIO;
83aa7877c4SAntonio Nino Diaz 	return 0;
84aa7877c4SAntonio Nino Diaz }
85aa7877c4SAntonio Nino Diaz 
nor_send_cmd(uintptr_t base_addr,unsigned long cmd)86aa7877c4SAntonio Nino Diaz void nor_send_cmd(uintptr_t base_addr, unsigned long cmd)
87aa7877c4SAntonio Nino Diaz {
88aa7877c4SAntonio Nino Diaz 	mmio_write_32(base_addr, NOR_2X16(cmd));
89aa7877c4SAntonio Nino Diaz }
90aa7877c4SAntonio Nino Diaz 
91aa7877c4SAntonio Nino Diaz /*
92aa7877c4SAntonio Nino Diaz  * This function programs a word in the flash. Be aware that it only
93aa7877c4SAntonio Nino Diaz  * can reset bits that were previously set. It cannot set bits that
94aa7877c4SAntonio Nino Diaz  * were previously reset. The resulting bits = old_bits & new bits.
95aa7877c4SAntonio Nino Diaz  * Return values:
96aa7877c4SAntonio Nino Diaz  *  0 = success
97aa7877c4SAntonio Nino Diaz  *  otherwise it returns a negative value
98aa7877c4SAntonio Nino Diaz  */
nor_word_program(uintptr_t base_addr,unsigned long data)99aa7877c4SAntonio Nino Diaz int nor_word_program(uintptr_t base_addr, unsigned long data)
100aa7877c4SAntonio Nino Diaz {
101aa7877c4SAntonio Nino Diaz 	uint32_t status;
102aa7877c4SAntonio Nino Diaz 	int ret;
103aa7877c4SAntonio Nino Diaz 
104aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_CLEAR_STATUS_REG);
105aa7877c4SAntonio Nino Diaz 
106aa7877c4SAntonio Nino Diaz 	/* Set the device in write word mode */
107aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_WORD_PROGRAM);
108aa7877c4SAntonio Nino Diaz 	mmio_write_32(base_addr, data);
109aa7877c4SAntonio Nino Diaz 
110aa7877c4SAntonio Nino Diaz 	ret = nor_poll_dws(base_addr, DWS_WORD_PROGRAM_RETRIES);
111aa7877c4SAntonio Nino Diaz 	if (ret == 0) {
112aa7877c4SAntonio Nino Diaz 		/* Full status check */
113aa7877c4SAntonio Nino Diaz 		nor_send_cmd(base_addr, NOR_CMD_READ_STATUS_REG);
114aa7877c4SAntonio Nino Diaz 		status = mmio_read_32(base_addr);
115aa7877c4SAntonio Nino Diaz 
116aa7877c4SAntonio Nino Diaz 		if (status & (NOR_PS | NOR_BLS)) {
117aa7877c4SAntonio Nino Diaz 			nor_send_cmd(base_addr, NOR_CMD_CLEAR_STATUS_REG);
118aa7877c4SAntonio Nino Diaz 			ret = -EPERM;
119aa7877c4SAntonio Nino Diaz 		}
120aa7877c4SAntonio Nino Diaz 	}
121aa7877c4SAntonio Nino Diaz 
122aa7877c4SAntonio Nino Diaz 	if (ret == 0)
123aa7877c4SAntonio Nino Diaz 		ret = nor_full_status_check(base_addr);
124aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_READ_ARRAY);
125aa7877c4SAntonio Nino Diaz 
126aa7877c4SAntonio Nino Diaz 	return ret;
127aa7877c4SAntonio Nino Diaz }
128aa7877c4SAntonio Nino Diaz 
129aa7877c4SAntonio Nino Diaz /*
130aa7877c4SAntonio Nino Diaz  * Erase a full 256K block
131aa7877c4SAntonio Nino Diaz  * Return values:
132aa7877c4SAntonio Nino Diaz  *  0 = success
133aa7877c4SAntonio Nino Diaz  *  otherwise it returns a negative value
134aa7877c4SAntonio Nino Diaz  */
nor_erase(uintptr_t base_addr)135aa7877c4SAntonio Nino Diaz int nor_erase(uintptr_t base_addr)
136aa7877c4SAntonio Nino Diaz {
137aa7877c4SAntonio Nino Diaz 	int ret;
138aa7877c4SAntonio Nino Diaz 
139aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_CLEAR_STATUS_REG);
140aa7877c4SAntonio Nino Diaz 
141aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_BLOCK_ERASE);
142aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_BLOCK_ERASE_ACK);
143aa7877c4SAntonio Nino Diaz 
144aa7877c4SAntonio Nino Diaz 	ret = nor_poll_dws(base_addr, DWS_WORD_ERASE_RETRIES);
145aa7877c4SAntonio Nino Diaz 	if (ret == 0)
146aa7877c4SAntonio Nino Diaz 		ret = nor_full_status_check(base_addr);
147aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_READ_ARRAY);
148aa7877c4SAntonio Nino Diaz 
149aa7877c4SAntonio Nino Diaz 	return ret;
150aa7877c4SAntonio Nino Diaz }
151aa7877c4SAntonio Nino Diaz 
152aa7877c4SAntonio Nino Diaz /*
153aa7877c4SAntonio Nino Diaz  * Lock a full 256 block
154aa7877c4SAntonio Nino Diaz  * Return values:
155aa7877c4SAntonio Nino Diaz  *  0 = success
156aa7877c4SAntonio Nino Diaz  *  otherwise it returns a negative value
157aa7877c4SAntonio Nino Diaz  */
nor_lock(uintptr_t base_addr)158aa7877c4SAntonio Nino Diaz int nor_lock(uintptr_t base_addr)
159aa7877c4SAntonio Nino Diaz {
160aa7877c4SAntonio Nino Diaz 	int ret;
161aa7877c4SAntonio Nino Diaz 
162aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_CLEAR_STATUS_REG);
163aa7877c4SAntonio Nino Diaz 
164aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_LOCK_UNLOCK);
165aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_LOCK_BLOCK);
166aa7877c4SAntonio Nino Diaz 
167aa7877c4SAntonio Nino Diaz 	ret = nor_poll_dws(base_addr, DWS_WORD_LOCK_RETRIES);
168aa7877c4SAntonio Nino Diaz 	if (ret == 0)
169aa7877c4SAntonio Nino Diaz 		ret = nor_full_status_check(base_addr);
170aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_READ_ARRAY);
171aa7877c4SAntonio Nino Diaz 
172aa7877c4SAntonio Nino Diaz 	return ret;
173aa7877c4SAntonio Nino Diaz }
174aa7877c4SAntonio Nino Diaz 
175aa7877c4SAntonio Nino Diaz /*
176aa7877c4SAntonio Nino Diaz  * unlock a full 256 block
177aa7877c4SAntonio Nino Diaz  * Return values:
178aa7877c4SAntonio Nino Diaz  *  0 = success
179aa7877c4SAntonio Nino Diaz  *  otherwise it returns a negative value
180aa7877c4SAntonio Nino Diaz  */
nor_unlock(uintptr_t base_addr)181aa7877c4SAntonio Nino Diaz int nor_unlock(uintptr_t base_addr)
182aa7877c4SAntonio Nino Diaz {
183aa7877c4SAntonio Nino Diaz 	int ret;
184aa7877c4SAntonio Nino Diaz 
185aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_CLEAR_STATUS_REG);
186aa7877c4SAntonio Nino Diaz 
187aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_LOCK_UNLOCK);
188aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_UNLOCK_BLOCK);
189aa7877c4SAntonio Nino Diaz 
190aa7877c4SAntonio Nino Diaz 	ret = nor_poll_dws(base_addr, DWS_WORD_LOCK_RETRIES);
191aa7877c4SAntonio Nino Diaz 	if (ret == 0)
192aa7877c4SAntonio Nino Diaz 		ret = nor_full_status_check(base_addr);
193aa7877c4SAntonio Nino Diaz 	nor_send_cmd(base_addr, NOR_CMD_READ_ARRAY);
194aa7877c4SAntonio Nino Diaz 
195aa7877c4SAntonio Nino Diaz 	return ret;
196aa7877c4SAntonio Nino Diaz }
197