drivers/memory/spiflash: Fix bugs in and clean up read/write functions.

mp_spiflash_read had a bug in it where "dest" and "addr" were incremented
twice for a certain special case.  This was fixed, which then allowed the
function to be simplified to reduce code size.

mp_spiflash_write had a bug in it where "src" was not incremented correctly
for the case where the data to be written included the caching buffer as
well as some bytes after this buffer.  This was fixed and the resulting
code simplified.
This commit is contained in:
Damien George 2018-03-13 14:13:30 +11:00
parent e0bc438e4b
commit bdc875e602

View File

@ -233,50 +233,32 @@ void mp_spiflash_read(mp_spiflash_t *self, uint32_t addr, size_t len, uint8_t *d
mp_spiflash_acquire_bus(self); mp_spiflash_acquire_bus(self);
if (bufuser == self && bufsec != 0xffffffff) { if (bufuser == self && bufsec != 0xffffffff) {
uint32_t bis = addr / SECTOR_SIZE; uint32_t bis = addr / SECTOR_SIZE;
int rest = 0; uint32_t bie = (addr + len - 1) / SECTOR_SIZE;
if (bis < bufsec) { if (bis <= bufsec && bufsec <= bie) {
rest = bufsec * SECTOR_SIZE - addr; // Read straddles current buffer
if (rest > len) { size_t rest = 0;
rest = len; if (bis < bufsec) {
} // Read direct from flash for first part
mp_spiflash_read_data(self, addr, rest, dest); rest = bufsec * SECTOR_SIZE - addr;
len -= rest; mp_spiflash_read_data(self, addr, rest, dest);
if (len <= 0) {
mp_spiflash_release_bus(self);
return;
} else {
// Something from buffer...
addr = bufsec * SECTOR_SIZE;
dest += rest;
if (len > SECTOR_SIZE) {
rest = SECTOR_SIZE;
} else {
rest = len;
}
memcpy(dest, buf, rest);
len -= rest; len -= rest;
if (len <= 0) {
mp_spiflash_release_bus(self);
return;
}
dest += rest; dest += rest;
addr += rest; addr += rest;
} }
} else if (bis == bufsec) { uint32_t offset = addr & (SECTOR_SIZE - 1);
uint32_t offset = addr & (SECTOR_SIZE-1);
rest = SECTOR_SIZE - offset; rest = SECTOR_SIZE - offset;
if (rest > len) { if (rest > len) {
rest = len; rest = len;
} }
memcpy(dest, &buf[offset], rest); memcpy(dest, &buf[offset], rest);
len -= rest; len -= rest;
if (len <= 0) { if (len == 0) {
mp_spiflash_release_bus(self); mp_spiflash_release_bus(self);
return; return;
} }
dest += rest;
addr += rest;
} }
dest += rest;
addr += rest;
} }
// Read rest direct from flash // Read rest direct from flash
mp_spiflash_read_data(self, addr, len, dest); mp_spiflash_read_data(self, addr, len, dest);
@ -395,64 +377,67 @@ int mp_spiflash_write(mp_spiflash_t *self, uint32_t addr, size_t len, const uint
uint32_t bie = (addr + len - 1) / SECTOR_SIZE; uint32_t bie = (addr + len - 1) / SECTOR_SIZE;
mp_spiflash_acquire_bus(self); mp_spiflash_acquire_bus(self);
if (bufuser == self && bis <= bufsec && bie >= bufsec) { if (bufuser == self && bis <= bufsec && bie >= bufsec) {
// Current buffer affected, handle this part first // Write straddles current buffer
uint32_t taddr = (bufsec + 1) * SECTOR_SIZE; uint32_t pre;
int32_t offset = addr - bufsec * SECTOR_SIZE; uint32_t offset;
int32_t pre = bufsec * SECTOR_SIZE - addr; if (bufsec * SECTOR_SIZE >= addr) {
if (offset < 0) { pre = bufsec * SECTOR_SIZE - addr;
offset = 0; offset = 0;
} else { } else {
pre = 0; pre = 0;
offset = addr - bufsec * SECTOR_SIZE;
} }
int32_t rest = len - pre;
int32_t trail = 0; // Write buffered part first
if (rest > SECTOR_SIZE - offset) { uint32_t len_in_buf = len - pre;
trail = rest - (SECTOR_SIZE - offset); len = 0;
rest = SECTOR_SIZE - offset; if (len_in_buf > SECTOR_SIZE - offset) {
len = len_in_buf - (SECTOR_SIZE - offset);
len_in_buf = SECTOR_SIZE - offset;
} }
memcpy(&buf[offset], &src[pre], rest); memcpy(&buf[offset], &src[pre], len_in_buf);
self->flags |= 1; // Mark dirty self->flags |= 1; // Mark dirty
if ((pre | trail) == 0) {
mp_spiflash_release_bus(self); // Write part before buffer sector
return 0;
}
const uint8_t *p = src;
while (pre) { while (pre) {
int rest = pre & (SECTOR_SIZE - 1); int rest = pre & (SECTOR_SIZE - 1);
if (rest == 0) { if (rest == 0) {
rest = SECTOR_SIZE; rest = SECTOR_SIZE;
} }
mp_spiflash_write_part(self, addr, rest, p); int ret = mp_spiflash_write_part(self, addr, rest, src);
p += rest; if (ret != 0) {
mp_spiflash_release_bus(self);
return ret;
}
src += rest;
addr += rest; addr += rest;
pre -= rest; pre -= rest;
} }
while (trail) { src += len_in_buf;
int rest = trail; addr += len_in_buf;
if (rest > SECTOR_SIZE) {
rest = SECTOR_SIZE; // Fall through to write remaining part
}
mp_spiflash_write_part(self, taddr, rest, src);
src += rest;
taddr += rest;
trail -= rest;
}
} else {
// Current buffer not affected, business as usual
uint32_t offset = addr & (SECTOR_SIZE - 1);
while (len) {
int rest = SECTOR_SIZE - offset;
if (rest > len) {
rest = len;
}
mp_spiflash_write_part(self, addr, rest, src);
len -= rest;
addr += rest;
src += rest;
offset = 0;
}
} }
uint32_t offset = addr & (SECTOR_SIZE - 1);
while (len) {
int rest = SECTOR_SIZE - offset;
if (rest > len) {
rest = len;
}
int ret = mp_spiflash_write_part(self, addr, rest, src);
if (ret != 0) {
mp_spiflash_release_bus(self);
return ret;
}
len -= rest;
addr += rest;
src += rest;
offset = 0;
}
mp_spiflash_release_bus(self); mp_spiflash_release_bus(self);
return 0; return 0;
} }