tbm-mcu/source/drivers/spi_flash.c

194 lines
3.8 KiB
C
Raw Normal View History

2017-03-11 12:56:46 +00:00
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
2017-04-07 19:35:51 +02:00
#include <string.h>
2017-03-11 12:56:46 +00:00
2017-06-14 14:59:07 +02:00
#include <bitops.h>
2017-03-11 12:56:46 +00:00
#include <flash.h>
#include <macros.h>
#include <spi.h>
2017-04-07 17:32:43 +02:00
#include <spi_flash.h>
2017-03-11 12:56:46 +00:00
2017-04-07 19:35:51 +02:00
static size_t spi_flash_get_size(struct flash_dev *dev);
2017-06-14 14:13:23 +02:00
static size_t spi_flash_read(struct flash_dev *dev, uint32_t addr, void *data,
2017-03-11 12:56:46 +00:00
size_t len);
2017-06-14 14:13:23 +02:00
static size_t spi_flash_write(struct flash_dev *dev, uint32_t addr,
2017-03-11 12:56:46 +00:00
const void *data, size_t len);
2017-06-14 14:13:23 +02:00
static int spi_flash_erase(struct flash_dev *dev, uint32_t addr);
static void spi_flash_release(struct flash_dev *dev);
2017-03-11 12:56:46 +00:00
2017-06-14 14:13:23 +02:00
static struct flash_ops spi_flash_ops = {
.release = spi_flash_release,
2017-04-07 19:35:51 +02:00
.get_size = spi_flash_get_size,
.get_capacity = spi_flash_get_size,
2017-03-11 12:56:46 +00:00
.read = spi_flash_read,
.write = spi_flash_write,
2017-06-16 15:43:21 +02:00
.copy = default_flash_copy,
.is_erased = default_flash_is_erased,
2017-03-11 12:56:46 +00:00
.erase = spi_flash_erase,
};
static void spi_flash_addr(char *cmd, uint32_t addr)
{
cmd[1] = addr >> 16;
cmd[2] = addr >> 8;
cmd[3] = addr;
}
static uint8_t spi_flash_get_status1(struct flash_dev *dev)
{
struct spi_dev *spi_dev = dev->priv;
uint8_t cmd = SPI_FLASH_READ_SR1;
uint8_t ret;
spi_tx_rx(spi_dev, &ret, sizeof ret, &cmd, sizeof cmd);
return ret;
}
static int spi_flash_wait(struct flash_dev *dev)
{
while (spi_flash_get_status1(dev) & SPI_FLASH_SR1_BUSY);
return 0;
}
2017-03-11 12:56:46 +00:00
static void spi_flash_write_enable(struct flash_dev *dev)
{
2017-04-07 17:32:43 +02:00
char cmd[1] = { SPI_FLASH_WRITE_ENABLE };
2017-03-11 12:56:46 +00:00
struct spi_dev *spi_dev = dev->priv;
spi_tx_rx(spi_dev, NULL, 0, cmd, sizeof cmd);
2017-03-11 12:56:46 +00:00
}
static void spi_flash_write_disable(struct flash_dev *dev)
{
2017-04-07 17:32:43 +02:00
char cmd[1] = { SPI_FLASH_WRITE_DISABLE };
2017-03-11 12:56:46 +00:00
struct spi_dev *spi_dev = dev->priv;
spi_tx_rx(spi_dev, NULL, 0, cmd, sizeof cmd);
2017-03-11 12:56:46 +00:00
}
2017-06-14 14:13:23 +02:00
static size_t spi_flash_get_jedec_id(struct flash_dev *dev, uint8_t *jedec_id,
2017-04-07 19:35:51 +02:00
size_t len)
{
uint8_t cmd[1] = { SPI_FLASH_JEDEC_ID };
uint8_t buf[3];
2017-04-07 19:35:51 +02:00
struct spi_dev *spi_dev = dev->priv;
spi_tx_rx(spi_dev, buf, sizeof buf, cmd, sizeof cmd);
2017-04-07 19:35:51 +02:00
memcpy(jedec_id, buf, min(sizeof buf, len));
2017-04-07 19:35:51 +02:00
2017-06-14 14:13:23 +02:00
return len;
2017-04-07 19:35:51 +02:00
}
static size_t spi_flash_get_size(struct flash_dev *dev)
{
uint8_t jedec_id[4];
spi_flash_get_jedec_id(dev, jedec_id, sizeof jedec_id);
return SPI_FLASH_SIZE(jedec_id[2]);
}
2017-06-14 14:13:23 +02:00
static size_t spi_flash_read(struct flash_dev *dev, uint32_t addr, void *data,
2017-03-11 12:56:46 +00:00
size_t len)
{
char cmd[4];
2017-03-11 12:56:46 +00:00
struct spi_dev *spi_dev = dev->priv;
size_t nbytes, ret = 0;
2017-03-11 12:56:46 +00:00
while (len) {
cmd[0] = SPI_FLASH_READ;
spi_flash_addr(cmd, addr);
2017-03-11 12:56:46 +00:00
nbytes = min(len, 32);
2017-03-11 12:56:46 +00:00
spi_tx_rx(spi_dev, data, len, cmd, sizeof cmd);
addr += nbytes;
data += nbytes;
len -= nbytes;
ret += nbytes;
}
return ret;
2017-03-11 12:56:46 +00:00
}
2017-06-14 14:13:23 +02:00
static size_t spi_flash_write(struct flash_dev *dev, uint32_t addr,
2017-03-11 12:56:46 +00:00
const void *data, size_t len)
{
char cmd[4 + 32];
2017-03-11 12:56:46 +00:00
struct spi_dev *spi_dev = dev->priv;
size_t nbytes, ret = 0;
2017-03-11 12:56:46 +00:00
while (len) {
spi_flash_write_enable(dev);
cmd[0] = SPI_FLASH_PAGE_PROGRAM;
spi_flash_addr(cmd, addr);
2017-03-11 12:56:46 +00:00
nbytes = min(len, 32);
2017-06-26 12:18:44 +02:00
nbytes = min(nbytes, align_up(addr + 1, ilog2(256)) - addr);
memmove(cmd + 4, data, nbytes);
spi_tx_rx(spi_dev, NULL, 0, cmd, 4 + nbytes);
spi_flash_wait(dev);
addr += nbytes;
data += nbytes;
len -= nbytes;
ret += nbytes;
}
return ret;
2017-03-11 12:56:46 +00:00
}
2017-06-14 14:13:23 +02:00
static int spi_flash_erase(struct flash_dev *dev, uint32_t addr)
2017-03-11 12:56:46 +00:00
{
char cmd[4];
struct spi_dev *spi_dev = dev->priv;
spi_flash_write_enable(dev);
2017-06-14 14:13:23 +02:00
cmd[0] = SPI_FLASH_ERASE_4K;
spi_flash_addr(cmd, addr << dev->log2_block_size);
spi_tx_rx(spi_dev, NULL, 0, cmd, sizeof cmd);
spi_flash_wait(dev);
2017-06-14 14:13:23 +02:00
2017-03-11 12:56:46 +00:00
spi_flash_write_disable(dev);
2017-06-14 14:13:23 +02:00
2017-03-11 12:56:46 +00:00
return 0;
}
struct flash_dev *flash_probe(void)
{
struct flash_dev *dev;
if (!(dev = malloc(sizeof *dev)))
return NULL;
if (!(dev->priv = spi_probe()))
goto err_free_dev;
dev->ops = &spi_flash_ops;
2017-06-14 14:59:07 +02:00
dev->log2_block_size = ilog2(4 * KIB);
2017-03-11 12:56:46 +00:00
return dev;
err_free_dev:
free(dev);
return NULL;
}
2017-06-14 14:13:23 +02:00
static void spi_flash_release(struct flash_dev *dev)
{
2017-04-07 17:32:43 +02:00
if (!dev)
return;
spi_release(dev->priv);
free(dev);
}