blob: 773334e167ae36308264c53234fc71d19f6ae1c5 [file] [log] [blame]
Damien George784e0232017-01-24 16:56:03 +11001/*
2 * This file is part of the MicroPython project, http://micropython.org/
3 *
4 * The MIT License (MIT)
5 *
Damien George4e487002018-03-02 16:01:18 +11006 * Copyright (c) 2016-2018 Damien P. George
Damien George784e0232017-01-24 16:56:03 +11007 *
8 * Permission is hereby granted, free of charge, to any person obtaining a copy
9 * of this software and associated documentation files (the "Software"), to deal
10 * in the Software without restriction, including without limitation the rights
11 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 * copies of the Software, and to permit persons to whom the Software is
13 * furnished to do so, subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included in
16 * all copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 * THE SOFTWARE.
25 */
26
27#include <stdio.h>
28#include <string.h>
29
30#include "py/mperrno.h"
31#include "py/mphal.h"
Damien George784e0232017-01-24 16:56:03 +110032#include "drivers/memory/spiflash.h"
33
Damien George4e487002018-03-02 16:01:18 +110034#define QSPI_QE_MASK (0x02)
35#define USE_WR_DELAY (1)
36
37#define CMD_WRSR (0x01)
38#define CMD_WRITE (0x02)
39#define CMD_READ (0x03)
40#define CMD_RDSR (0x05)
41#define CMD_WREN (0x06)
42#define CMD_SEC_ERASE (0x20)
43#define CMD_RDCR (0x35)
44#define CMD_RD_DEVID (0x9f)
45#define CMD_CHIP_ERASE (0xc7)
46#define CMD_C4READ (0xeb)
iabdalkaderb5e80fa2024-11-29 16:56:20 +010047#define CMD_RSTEN (0x66)
48#define CMD_RESET (0x99)
Damien George4e487002018-03-02 16:01:18 +110049
Andrew Leech30501d32020-01-28 14:59:05 +110050// 32 bit addressing commands
51#define CMD_WRITE_32 (0x12)
52#define CMD_READ_32 (0x13)
53#define CMD_SEC_ERASE_32 (0x21)
54#define CMD_C4READ_32 (0xec)
55
Damien George784e0232017-01-24 16:56:03 +110056#define WAIT_SR_TIMEOUT (1000000)
57
58#define PAGE_SIZE (256) // maximum bytes we can write in one SPI transfer
Damien George86fe73b2018-06-07 14:09:10 +100059#define SECTOR_SIZE MP_SPIFLASH_ERASE_BLOCK_SIZE
Damien George784e0232017-01-24 16:56:03 +110060
Angus Grattondecf8e62024-02-27 15:32:29 +110061static void mp_spiflash_acquire_bus(mp_spiflash_t *self) {
Damien George4e487002018-03-02 16:01:18 +110062 const mp_spiflash_config_t *c = self->config;
63 if (c->bus_kind == MP_SPIFLASH_BUS_QSPI) {
64 c->bus.u_qspi.proto->ioctl(c->bus.u_qspi.data, MP_QSPI_IOCTL_BUS_ACQUIRE);
65 }
Damien George784e0232017-01-24 16:56:03 +110066}
67
Angus Grattondecf8e62024-02-27 15:32:29 +110068static void mp_spiflash_release_bus(mp_spiflash_t *self) {
Damien George4e487002018-03-02 16:01:18 +110069 const mp_spiflash_config_t *c = self->config;
70 if (c->bus_kind == MP_SPIFLASH_BUS_QSPI) {
71 c->bus.u_qspi.proto->ioctl(c->bus.u_qspi.data, MP_QSPI_IOCTL_BUS_RELEASE);
72 }
Damien George784e0232017-01-24 16:56:03 +110073}
74
Angus Grattondecf8e62024-02-27 15:32:29 +110075static int mp_spiflash_write_cmd_data(mp_spiflash_t *self, uint8_t cmd, size_t len, uint32_t data) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +110076 int ret = 0;
Damien George4e487002018-03-02 16:01:18 +110077 const mp_spiflash_config_t *c = self->config;
78 if (c->bus_kind == MP_SPIFLASH_BUS_SPI) {
79 // Note: len/data are unused for standard SPI
80 mp_hal_pin_write(c->bus.u_spi.cs, 0);
81 c->bus.u_spi.proto->transfer(c->bus.u_spi.data, 1, &cmd, NULL);
82 mp_hal_pin_write(c->bus.u_spi.cs, 1);
83 } else {
Andrew Leech7ee5afe2021-03-05 10:15:29 +110084 ret = c->bus.u_qspi.proto->write_cmd_data(c->bus.u_qspi.data, cmd, len, data);
Damien George4e487002018-03-02 16:01:18 +110085 }
Andrew Leech7ee5afe2021-03-05 10:15:29 +110086 return ret;
Damien George4e487002018-03-02 16:01:18 +110087}
88
Angus Grattondecf8e62024-02-27 15:32:29 +110089static int mp_spiflash_transfer_cmd_addr_data(mp_spiflash_t *self, uint8_t cmd, uint32_t addr, size_t len, const uint8_t *src, uint8_t *dest) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +110090 int ret = 0;
Damien George4e487002018-03-02 16:01:18 +110091 const mp_spiflash_config_t *c = self->config;
92 if (c->bus_kind == MP_SPIFLASH_BUS_SPI) {
Andrew Leech30501d32020-01-28 14:59:05 +110093 uint8_t buf[5] = {cmd, 0};
94 uint8_t buff_len = 1 + mp_spi_set_addr_buff(&buf[1], addr);
Damien George4e487002018-03-02 16:01:18 +110095 mp_hal_pin_write(c->bus.u_spi.cs, 0);
Andrew Leech30501d32020-01-28 14:59:05 +110096 c->bus.u_spi.proto->transfer(c->bus.u_spi.data, buff_len, buf, NULL);
97 if (len && (src != NULL)) {
Damien George4e487002018-03-02 16:01:18 +110098 c->bus.u_spi.proto->transfer(c->bus.u_spi.data, len, src, NULL);
Andrew Leech30501d32020-01-28 14:59:05 +110099 } else if (len && (dest != NULL)) {
100 c->bus.u_spi.proto->transfer(c->bus.u_spi.data, len, dest, dest);
Damien George4e487002018-03-02 16:01:18 +1100101 }
Andrew Leech30501d32020-01-28 14:59:05 +1100102
Damien George4e487002018-03-02 16:01:18 +1100103 mp_hal_pin_write(c->bus.u_spi.cs, 1);
104 } else {
Andrew Leech30501d32020-01-28 14:59:05 +1100105 if (dest != NULL) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100106 ret = c->bus.u_qspi.proto->read_cmd_qaddr_qdata(c->bus.u_qspi.data, cmd, addr, len, dest);
Andrew Leech30501d32020-01-28 14:59:05 +1100107 } else {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100108 ret = c->bus.u_qspi.proto->write_cmd_addr_data(c->bus.u_qspi.data, cmd, addr, len, src);
Andrew Leech30501d32020-01-28 14:59:05 +1100109 }
Damien George4e487002018-03-02 16:01:18 +1100110 }
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100111 return ret;
Damien George4e487002018-03-02 16:01:18 +1100112}
113
Angus Grattondecf8e62024-02-27 15:32:29 +1100114static int mp_spiflash_read_cmd(mp_spiflash_t *self, uint8_t cmd, size_t len, uint32_t *dest) {
Damien George4e487002018-03-02 16:01:18 +1100115 const mp_spiflash_config_t *c = self->config;
116 if (c->bus_kind == MP_SPIFLASH_BUS_SPI) {
Damien George4e487002018-03-02 16:01:18 +1100117 mp_hal_pin_write(c->bus.u_spi.cs, 0);
118 c->bus.u_spi.proto->transfer(c->bus.u_spi.data, 1, &cmd, NULL);
Damien Georgeb042fd52022-12-09 12:28:54 +1100119 c->bus.u_spi.proto->transfer(c->bus.u_spi.data, len, (void*)dest, (void*)dest);
Damien George4e487002018-03-02 16:01:18 +1100120 mp_hal_pin_write(c->bus.u_spi.cs, 1);
Damien Georgeb042fd52022-12-09 12:28:54 +1100121 return 0;
Damien George4e487002018-03-02 16:01:18 +1100122 } else {
Damien Georgeb042fd52022-12-09 12:28:54 +1100123 return c->bus.u_qspi.proto->read_cmd(c->bus.u_qspi.data, cmd, len, dest);
Damien George4e487002018-03-02 16:01:18 +1100124 }
125}
126
Angus Grattondecf8e62024-02-27 15:32:29 +1100127static int mp_spiflash_read_data(mp_spiflash_t *self, uint32_t addr, size_t len, uint8_t *dest) {
Damien George4e487002018-03-02 16:01:18 +1100128 const mp_spiflash_config_t *c = self->config;
Andrew Leech30501d32020-01-28 14:59:05 +1100129 uint8_t cmd;
Damien George4e487002018-03-02 16:01:18 +1100130 if (c->bus_kind == MP_SPIFLASH_BUS_SPI) {
Damien George54f16942022-06-01 18:50:43 +1000131 cmd = MICROPY_HW_SPI_ADDR_IS_32BIT(addr) ? CMD_READ_32 : CMD_READ;
Damien George4e487002018-03-02 16:01:18 +1100132 } else {
Damien George54f16942022-06-01 18:50:43 +1000133 cmd = MICROPY_HW_SPI_ADDR_IS_32BIT(addr) ? CMD_C4READ_32 : CMD_C4READ;
Damien George4e487002018-03-02 16:01:18 +1100134 }
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100135 return mp_spiflash_transfer_cmd_addr_data(self, cmd, addr, len, NULL, dest);
Damien George4e487002018-03-02 16:01:18 +1100136}
137
Angus Grattondecf8e62024-02-27 15:32:29 +1100138static int mp_spiflash_write_cmd(mp_spiflash_t *self, uint8_t cmd) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100139 return mp_spiflash_write_cmd_data(self, cmd, 0, 0);
Damien George4e487002018-03-02 16:01:18 +1100140}
141
Angus Grattondecf8e62024-02-27 15:32:29 +1100142static int mp_spiflash_wait_sr(mp_spiflash_t *self, uint8_t mask, uint8_t val, uint32_t timeout) {
Andrew Leech2ed2ec12019-01-29 15:20:01 +1100143 do {
Damien Georgeb042fd52022-12-09 12:28:54 +1100144 uint32_t sr;
145 int ret = mp_spiflash_read_cmd(self, CMD_RDSR, 1, &sr);
146 if (ret != 0) {
147 return ret;
148 }
Damien George4e487002018-03-02 16:01:18 +1100149 if ((sr & mask) == val) {
Andrew Leech2ed2ec12019-01-29 15:20:01 +1100150 return 0; // success
Damien George784e0232017-01-24 16:56:03 +1100151 }
Andrew Leech2ed2ec12019-01-29 15:20:01 +1100152 } while (timeout--);
153
154 return -MP_ETIMEDOUT;
Damien George784e0232017-01-24 16:56:03 +1100155}
156
Angus Grattondecf8e62024-02-27 15:32:29 +1100157static int mp_spiflash_wait_wel1(mp_spiflash_t *self) {
Damien George784e0232017-01-24 16:56:03 +1100158 return mp_spiflash_wait_sr(self, 2, 2, WAIT_SR_TIMEOUT);
159}
160
Angus Grattondecf8e62024-02-27 15:32:29 +1100161static int mp_spiflash_wait_wip0(mp_spiflash_t *self) {
Damien George784e0232017-01-24 16:56:03 +1100162 return mp_spiflash_wait_sr(self, 1, 0, WAIT_SR_TIMEOUT);
163}
164
Damien George8cde5fa2019-07-03 01:03:25 +1000165static inline void mp_spiflash_deepsleep_internal(mp_spiflash_t *self, int value) {
166 mp_spiflash_write_cmd(self, value ? 0xb9 : 0xab); // sleep/wake
167}
168
Damien George4e487002018-03-02 16:01:18 +1100169void mp_spiflash_init(mp_spiflash_t *self) {
170 self->flags = 0;
171
172 if (self->config->bus_kind == MP_SPIFLASH_BUS_SPI) {
173 mp_hal_pin_write(self->config->bus.u_spi.cs, 1);
174 mp_hal_pin_output(self->config->bus.u_spi.cs);
Damien Georgea739b352018-03-09 17:32:28 +1100175 self->config->bus.u_spi.proto->ioctl(self->config->bus.u_spi.data, MP_SPI_IOCTL_INIT);
Damien George4e487002018-03-02 16:01:18 +1100176 } else {
177 self->config->bus.u_qspi.proto->ioctl(self->config->bus.u_qspi.data, MP_QSPI_IOCTL_INIT);
178 }
179
180 mp_spiflash_acquire_bus(self);
181
Damien George8cde5fa2019-07-03 01:03:25 +1000182 // Ensure SPI flash is out of sleep mode
183 mp_spiflash_deepsleep_internal(self, 0);
184
iabdalkaderb5e80fa2024-11-29 16:56:20 +0100185 // Software reset.
186 #if MICROPY_HW_SPIFLASH_SOFT_RESET
187 mp_spiflash_write_cmd(self, CMD_RSTEN);
188 mp_spiflash_write_cmd(self, CMD_RESET);
189 mp_spiflash_wait_wip0(self);
190 mp_hal_delay_ms(1);
191 #endif
192
Damien George4e487002018-03-02 16:01:18 +1100193 #if defined(CHECK_DEVID)
194 // Validate device id
Damien Georgeb042fd52022-12-09 12:28:54 +1100195 uint32_t devid;
196 int ret = mp_spiflash_read_cmd(self, CMD_RD_DEVID, 3, &devid);
197 if (ret != 0 || devid != CHECK_DEVID) {
198 mp_spiflash_release_bus(self);
199 return;
Damien George4e487002018-03-02 16:01:18 +1100200 }
201 #endif
202
203 if (self->config->bus_kind == MP_SPIFLASH_BUS_QSPI) {
204 // Set QE bit
Damien Georgeb042fd52022-12-09 12:28:54 +1100205 uint32_t sr = 0, cr = 0;
206 int ret = mp_spiflash_read_cmd(self, CMD_RDSR, 1, &sr);
207 if (ret == 0) {
208 ret = mp_spiflash_read_cmd(self, CMD_RDCR, 1, &cr);
209 }
210 uint32_t data = (sr & 0xff) | (cr & 0xff) << 8;
211 if (ret == 0 && !(data & (QSPI_QE_MASK << 8))) {
Damien Georgecc34b082018-03-11 11:25:38 +1100212 data |= QSPI_QE_MASK << 8;
Damien George4e487002018-03-02 16:01:18 +1100213 mp_spiflash_write_cmd(self, CMD_WREN);
214 mp_spiflash_write_cmd_data(self, CMD_WRSR, 2, data);
215 mp_spiflash_wait_wip0(self);
216 }
217 }
218
219 mp_spiflash_release_bus(self);
Damien George784e0232017-01-24 16:56:03 +1100220}
221
Damien George8cde5fa2019-07-03 01:03:25 +1000222void mp_spiflash_deepsleep(mp_spiflash_t *self, int value) {
223 if (value) {
224 mp_spiflash_acquire_bus(self);
225 }
226 mp_spiflash_deepsleep_internal(self, value);
227 if (!value) {
228 mp_spiflash_release_bus(self);
229 }
230}
231
Angus Grattondecf8e62024-02-27 15:32:29 +1100232static int mp_spiflash_erase_block_internal(mp_spiflash_t *self, uint32_t addr) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100233 int ret = 0;
Damien George784e0232017-01-24 16:56:03 +1100234 // enable writes
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100235 ret = mp_spiflash_write_cmd(self, CMD_WREN);
236 if (ret != 0) {
237 return ret;
238 }
Damien George784e0232017-01-24 16:56:03 +1100239
240 // wait WEL=1
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100241 ret = mp_spiflash_wait_wel1(self);
Damien George784e0232017-01-24 16:56:03 +1100242 if (ret != 0) {
243 return ret;
244 }
245
246 // erase the sector
Damien George54f16942022-06-01 18:50:43 +1000247 uint8_t cmd = MICROPY_HW_SPI_ADDR_IS_32BIT(addr) ? CMD_SEC_ERASE_32 : CMD_SEC_ERASE;
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100248 ret = mp_spiflash_transfer_cmd_addr_data(self, cmd, addr, 0, NULL, NULL);
249 if (ret != 0) {
250 return ret;
251 }
Damien George784e0232017-01-24 16:56:03 +1100252
253 // wait WIP=0
254 return mp_spiflash_wait_wip0(self);
255}
256
Angus Grattondecf8e62024-02-27 15:32:29 +1100257static int mp_spiflash_write_page(mp_spiflash_t *self, uint32_t addr, size_t len, const uint8_t *src) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100258 int ret = 0;
Damien George784e0232017-01-24 16:56:03 +1100259 // enable writes
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100260 ret = mp_spiflash_write_cmd(self, CMD_WREN);
261 if (ret != 0) {
262 return ret;
263 }
Damien George784e0232017-01-24 16:56:03 +1100264
265 // wait WEL=1
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100266 ret = mp_spiflash_wait_wel1(self);
Damien George784e0232017-01-24 16:56:03 +1100267 if (ret != 0) {
268 return ret;
269 }
270
271 // write the page
Damien George54f16942022-06-01 18:50:43 +1000272 uint8_t cmd = MICROPY_HW_SPI_ADDR_IS_32BIT(addr) ? CMD_WRITE_32 : CMD_WRITE;
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100273 ret = mp_spiflash_transfer_cmd_addr_data(self, cmd, addr, len, src, NULL);
274 if (ret != 0) {
275 return ret;
276 }
Damien George784e0232017-01-24 16:56:03 +1100277
278 // wait WIP=0
279 return mp_spiflash_wait_wip0(self);
280}
281
Damien Georgecc5a9402018-06-07 15:36:27 +1000282/******************************************************************************/
Damien Georgeb78ca322018-06-07 15:39:46 +1000283// Interface functions that go direct to the SPI flash device
284
285int mp_spiflash_erase_block(mp_spiflash_t *self, uint32_t addr) {
286 mp_spiflash_acquire_bus(self);
287 int ret = mp_spiflash_erase_block_internal(self, addr);
288 mp_spiflash_release_bus(self);
289 return ret;
290}
291
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100292int mp_spiflash_read(mp_spiflash_t *self, uint32_t addr, size_t len, uint8_t *dest) {
Damien Georgeb78ca322018-06-07 15:39:46 +1000293 if (len == 0) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100294 return 0;
Damien Georgeb78ca322018-06-07 15:39:46 +1000295 }
296 mp_spiflash_acquire_bus(self);
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100297 int ret = mp_spiflash_read_data(self, addr, len, dest);
Damien Georgeb78ca322018-06-07 15:39:46 +1000298 mp_spiflash_release_bus(self);
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100299 return ret;
Damien Georgeb78ca322018-06-07 15:39:46 +1000300}
301
302int mp_spiflash_write(mp_spiflash_t *self, uint32_t addr, size_t len, const uint8_t *src) {
303 mp_spiflash_acquire_bus(self);
304 int ret = 0;
305 uint32_t offset = addr & (PAGE_SIZE - 1);
306 while (len) {
307 size_t rest = PAGE_SIZE - offset;
308 if (rest > len) {
309 rest = len;
310 }
311 ret = mp_spiflash_write_page(self, addr, rest, src);
312 if (ret != 0) {
313 break;
314 }
315 len -= rest;
316 addr += rest;
317 src += rest;
318 offset = 0;
319 }
320 mp_spiflash_release_bus(self);
321 return ret;
322}
323
324/******************************************************************************/
Damien Georgecc5a9402018-06-07 15:36:27 +1000325// Interface functions that use the cache
326
Damien Georgee43a74a2020-12-17 16:59:54 +1100327#if MICROPY_HW_SPIFLASH_ENABLE_CACHE
328
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100329int mp_spiflash_cached_read(mp_spiflash_t *self, uint32_t addr, size_t len, uint8_t *dest) {
Damien George4e487002018-03-02 16:01:18 +1100330 if (len == 0) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100331 return 0;
Damien George4e487002018-03-02 16:01:18 +1100332 }
Damien George784e0232017-01-24 16:56:03 +1100333 mp_spiflash_acquire_bus(self);
Damien George86fe73b2018-06-07 14:09:10 +1000334 mp_spiflash_cache_t *cache = self->config->cache;
335 if (cache->user == self && cache->block != 0xffffffff) {
Damien George4e487002018-03-02 16:01:18 +1100336 uint32_t bis = addr / SECTOR_SIZE;
Damien Georgebdc875e2018-03-13 14:13:30 +1100337 uint32_t bie = (addr + len - 1) / SECTOR_SIZE;
Damien George86fe73b2018-06-07 14:09:10 +1000338 if (bis <= cache->block && cache->block <= bie) {
Damien Georgebdc875e2018-03-13 14:13:30 +1100339 // Read straddles current buffer
340 size_t rest = 0;
Damien George86fe73b2018-06-07 14:09:10 +1000341 if (bis < cache->block) {
Damien Georgebdc875e2018-03-13 14:13:30 +1100342 // Read direct from flash for first part
Damien George86fe73b2018-06-07 14:09:10 +1000343 rest = cache->block * SECTOR_SIZE - addr;
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100344 int ret = mp_spiflash_read_data(self, addr, rest, dest);
345 if (ret != 0) {
346 mp_spiflash_release_bus(self);
347 return ret;
348 }
Damien George4e487002018-03-02 16:01:18 +1100349 len -= rest;
Damien George4e487002018-03-02 16:01:18 +1100350 dest += rest;
351 addr += rest;
352 }
Damien Georgebdc875e2018-03-13 14:13:30 +1100353 uint32_t offset = addr & (SECTOR_SIZE - 1);
Damien George4e487002018-03-02 16:01:18 +1100354 rest = SECTOR_SIZE - offset;
355 if (rest > len) {
356 rest = len;
357 }
Damien George86fe73b2018-06-07 14:09:10 +1000358 memcpy(dest, &cache->buf[offset], rest);
Damien George4e487002018-03-02 16:01:18 +1100359 len -= rest;
Damien Georgebdc875e2018-03-13 14:13:30 +1100360 if (len == 0) {
Damien George4e487002018-03-02 16:01:18 +1100361 mp_spiflash_release_bus(self);
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100362 return 0;
Damien George4e487002018-03-02 16:01:18 +1100363 }
Damien Georgebdc875e2018-03-13 14:13:30 +1100364 dest += rest;
365 addr += rest;
Damien George4e487002018-03-02 16:01:18 +1100366 }
Damien George4e487002018-03-02 16:01:18 +1100367 }
368 // Read rest direct from flash
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100369 int ret = mp_spiflash_read_data(self, addr, len, dest);
Damien George784e0232017-01-24 16:56:03 +1100370 mp_spiflash_release_bus(self);
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100371 return ret;
Damien George784e0232017-01-24 16:56:03 +1100372}
373
Angus Grattondecf8e62024-02-27 15:32:29 +1100374static int mp_spiflash_cache_flush_internal(mp_spiflash_t *self) {
Damien George4e487002018-03-02 16:01:18 +1100375 #if USE_WR_DELAY
376 if (!(self->flags & 1)) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100377 return 0;
Damien George4e487002018-03-02 16:01:18 +1100378 }
Damien George784e0232017-01-24 16:56:03 +1100379
Damien George4e487002018-03-02 16:01:18 +1100380 self->flags &= ~1;
381
Damien George86fe73b2018-06-07 14:09:10 +1000382 mp_spiflash_cache_t *cache = self->config->cache;
383
Damien George4e487002018-03-02 16:01:18 +1100384 // Erase sector
Damien Georgeb78ca322018-06-07 15:39:46 +1000385 int ret = mp_spiflash_erase_block_internal(self, cache->block * SECTOR_SIZE);
Damien George4e487002018-03-02 16:01:18 +1100386 if (ret != 0) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100387 return ret;
Damien George4e487002018-03-02 16:01:18 +1100388 }
389
390 // Write
391 for (int i = 0; i < 16; i += 1) {
Damien Georgeb78ca322018-06-07 15:39:46 +1000392 uint32_t addr = cache->block * SECTOR_SIZE + i * PAGE_SIZE;
393 int ret = mp_spiflash_write_page(self, addr, PAGE_SIZE, cache->buf + i * PAGE_SIZE);
Damien George4e487002018-03-02 16:01:18 +1100394 if (ret != 0) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100395 return ret;
Damien George4e487002018-03-02 16:01:18 +1100396 }
397 }
398 #endif
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100399 return 0;
Damien George4e487002018-03-02 16:01:18 +1100400}
401
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100402int mp_spiflash_cache_flush(mp_spiflash_t *self) {
Damien George4e487002018-03-02 16:01:18 +1100403 mp_spiflash_acquire_bus(self);
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100404 int ret = mp_spiflash_cache_flush_internal(self);
Damien George4e487002018-03-02 16:01:18 +1100405 mp_spiflash_release_bus(self);
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100406 return ret;
Damien George4e487002018-03-02 16:01:18 +1100407}
408
Angus Grattondecf8e62024-02-27 15:32:29 +1100409static int mp_spiflash_cached_write_part(mp_spiflash_t *self, uint32_t addr, size_t len, const uint8_t *src) {
Damien George4e487002018-03-02 16:01:18 +1100410 // Align to 4096 sector
Damien George784e0232017-01-24 16:56:03 +1100411 uint32_t offset = addr & 0xfff;
Damien George4e487002018-03-02 16:01:18 +1100412 uint32_t sec = addr >> 12;
413 addr = sec << 12;
Damien George784e0232017-01-24 16:56:03 +1100414
Damien George4e487002018-03-02 16:01:18 +1100415 // Restriction for now, so we don't need to erase multiple pages
Damien George86fe73b2018-06-07 14:09:10 +1000416 if (offset + len > SECTOR_SIZE) {
Damien Georgecc5a9402018-06-07 15:36:27 +1000417 printf("mp_spiflash_cached_write_part: len is too large\n");
Damien George784e0232017-01-24 16:56:03 +1100418 return -MP_EIO;
419 }
420
Damien George86fe73b2018-06-07 14:09:10 +1000421 mp_spiflash_cache_t *cache = self->config->cache;
422
Damien George4e487002018-03-02 16:01:18 +1100423 // Acquire the sector buffer
Damien George86fe73b2018-06-07 14:09:10 +1000424 if (cache->user != self) {
425 if (cache->user != NULL) {
Damien Georgecc5a9402018-06-07 15:36:27 +1000426 mp_spiflash_cache_flush(cache->user);
Damien George4e487002018-03-02 16:01:18 +1100427 }
Damien George86fe73b2018-06-07 14:09:10 +1000428 cache->user = self;
429 cache->block = 0xffffffff;
Damien George784e0232017-01-24 16:56:03 +1100430 }
431
Damien George86fe73b2018-06-07 14:09:10 +1000432 if (cache->block != sec) {
Damien George4e487002018-03-02 16:01:18 +1100433 // Read sector
434 #if USE_WR_DELAY
Damien George86fe73b2018-06-07 14:09:10 +1000435 if (cache->block != 0xffffffff) {
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100436 int ret = mp_spiflash_cache_flush_internal(self);
437 if (ret != 0) {
438 return ret;
439 }
Damien George4e487002018-03-02 16:01:18 +1100440 }
441 #endif
Andrew Leech7ee5afe2021-03-05 10:15:29 +1100442 int ret = mp_spiflash_read_data(self, addr, SECTOR_SIZE, cache->buf);
443 if (ret != 0) {
444 return ret;
445 }
Damien George4e487002018-03-02 16:01:18 +1100446 }
Damien George784e0232017-01-24 16:56:03 +1100447
Damien George4e487002018-03-02 16:01:18 +1100448 #if USE_WR_DELAY
449
Damien George86fe73b2018-06-07 14:09:10 +1000450 cache->block = sec;
Damien George4e487002018-03-02 16:01:18 +1100451 // Just copy to buffer
Damien George86fe73b2018-06-07 14:09:10 +1000452 memcpy(cache->buf + offset, src, len);
Damien George4e487002018-03-02 16:01:18 +1100453 // And mark dirty
454 self->flags |= 1;
455
456 #else
457
458 uint32_t dirty = 0;
459 for (size_t i = 0; i < len; ++i) {
Damien George86fe73b2018-06-07 14:09:10 +1000460 if (cache->buf[offset + i] != src[i]) {
461 if (cache->buf[offset + i] != 0xff) {
Damien George4e487002018-03-02 16:01:18 +1100462 // Erase sector
Damien Georgeb78ca322018-06-07 15:39:46 +1000463 int ret = mp_spiflash_erase_block_internal(self, addr);
Damien George4e487002018-03-02 16:01:18 +1100464 if (ret != 0) {
465 return ret;
466 }
467 dirty = 0xffff;
468 break;
469 } else {
470 dirty |= (1 << ((offset + i) >> 8));
471 }
Damien George784e0232017-01-24 16:56:03 +1100472 }
473 }
474
Damien George86fe73b2018-06-07 14:09:10 +1000475 cache->block = sec;
Damien George4e487002018-03-02 16:01:18 +1100476 // Copy new block into buffer
Damien George86fe73b2018-06-07 14:09:10 +1000477 memcpy(cache->buf + offset, src, len);
Damien George4e487002018-03-02 16:01:18 +1100478
479 // Write sector in pages of 256 bytes
480 for (size_t i = 0; i < 16; ++i) {
481 if (dirty & (1 << i)) {
Damien Georgeb78ca322018-06-07 15:39:46 +1000482 int ret = mp_spiflash_write_page(self, addr + i * PAGE_SIZE, PAGE_SIZE, cache->buf + i * PAGE_SIZE);
Damien George4e487002018-03-02 16:01:18 +1100483 if (ret != 0) {
484 return ret;
485 }
486 }
487 }
488
489 #endif
490
Damien George784e0232017-01-24 16:56:03 +1100491 return 0; // success
492}
Damien George4e487002018-03-02 16:01:18 +1100493
Damien Georgecc5a9402018-06-07 15:36:27 +1000494int mp_spiflash_cached_write(mp_spiflash_t *self, uint32_t addr, size_t len, const uint8_t *src) {
Damien George4e487002018-03-02 16:01:18 +1100495 uint32_t bis = addr / SECTOR_SIZE;
496 uint32_t bie = (addr + len - 1) / SECTOR_SIZE;
497
498 mp_spiflash_acquire_bus(self);
Damien Georgebdc875e2018-03-13 14:13:30 +1100499
Damien George86fe73b2018-06-07 14:09:10 +1000500 mp_spiflash_cache_t *cache = self->config->cache;
501 if (cache->user == self && bis <= cache->block && bie >= cache->block) {
Damien Georgebdc875e2018-03-13 14:13:30 +1100502 // Write straddles current buffer
503 uint32_t pre;
504 uint32_t offset;
Damien George86fe73b2018-06-07 14:09:10 +1000505 if (cache->block * SECTOR_SIZE >= addr) {
506 pre = cache->block * SECTOR_SIZE - addr;
Damien George4e487002018-03-02 16:01:18 +1100507 offset = 0;
508 } else {
509 pre = 0;
Damien George86fe73b2018-06-07 14:09:10 +1000510 offset = addr - cache->block * SECTOR_SIZE;
Damien George4e487002018-03-02 16:01:18 +1100511 }
Damien Georgebdc875e2018-03-13 14:13:30 +1100512
513 // Write buffered part first
514 uint32_t len_in_buf = len - pre;
515 len = 0;
516 if (len_in_buf > SECTOR_SIZE - offset) {
517 len = len_in_buf - (SECTOR_SIZE - offset);
518 len_in_buf = SECTOR_SIZE - offset;
Damien George4e487002018-03-02 16:01:18 +1100519 }
Damien George86fe73b2018-06-07 14:09:10 +1000520 memcpy(&cache->buf[offset], &src[pre], len_in_buf);
Damien George4e487002018-03-02 16:01:18 +1100521 self->flags |= 1; // Mark dirty
Damien Georgebdc875e2018-03-13 14:13:30 +1100522
523 // Write part before buffer sector
Damien George4e487002018-03-02 16:01:18 +1100524 while (pre) {
525 int rest = pre & (SECTOR_SIZE - 1);
526 if (rest == 0) {
527 rest = SECTOR_SIZE;
528 }
Damien Georgecc5a9402018-06-07 15:36:27 +1000529 int ret = mp_spiflash_cached_write_part(self, addr, rest, src);
Damien Georgebdc875e2018-03-13 14:13:30 +1100530 if (ret != 0) {
531 mp_spiflash_release_bus(self);
532 return ret;
533 }
534 src += rest;
Damien George4e487002018-03-02 16:01:18 +1100535 addr += rest;
536 pre -= rest;
537 }
Damien Georgebdc875e2018-03-13 14:13:30 +1100538 src += len_in_buf;
539 addr += len_in_buf;
540
541 // Fall through to write remaining part
Damien George4e487002018-03-02 16:01:18 +1100542 }
Damien Georgebdc875e2018-03-13 14:13:30 +1100543
544 uint32_t offset = addr & (SECTOR_SIZE - 1);
545 while (len) {
546 int rest = SECTOR_SIZE - offset;
547 if (rest > len) {
548 rest = len;
549 }
Damien Georgecc5a9402018-06-07 15:36:27 +1000550 int ret = mp_spiflash_cached_write_part(self, addr, rest, src);
Damien Georgebdc875e2018-03-13 14:13:30 +1100551 if (ret != 0) {
552 mp_spiflash_release_bus(self);
553 return ret;
554 }
555 len -= rest;
556 addr += rest;
557 src += rest;
558 offset = 0;
559 }
560
Damien George4e487002018-03-02 16:01:18 +1100561 mp_spiflash_release_bus(self);
562 return 0;
563}
Damien Georgee43a74a2020-12-17 16:59:54 +1100564
565#endif // MICROPY_HW_SPIFLASH_ENABLE_CACHE