tbm-mcu/source/drivers/sandbox_flash.c

149 lines
2.8 KiB
C
Raw Normal View History

2017-03-11 13:32:30 +00:00
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/mman.h>
#include <sys/stat.h>
2017-03-11 13:32:30 +00:00
#include <bitops.h>
2017-03-11 13:32:30 +00:00
#include <flash.h>
#include <macros.h>
#define CONFIG_FLASH_SIZE 4 * MIB
struct stdio_flash_priv {
FILE *fp;
2017-03-11 13:32:30 +00:00
char *data;
size_t size;
};
2017-04-07 19:35:51 +02:00
static size_t stdio_flash_get_size(struct flash_dev *dev);
static size_t stdio_flash_read(struct flash_dev *dev, uint32_t addr, void *data,
2017-03-11 13:32:30 +00:00
size_t len);
static size_t stdio_flash_write(struct flash_dev *dev, uint32_t addr,
2017-03-11 13:32:30 +00:00
const void *data, size_t len);
2017-05-19 17:04:45 +02:00
static int stdio_flash_erase(struct flash_dev *dev, uint32_t addr);
static void stdio_flash_release(struct flash_dev *dev);
2017-03-11 13:32:30 +00:00
static struct flash_ops stdio_flash_ops = {
.release = stdio_flash_release,
2017-04-07 19:35:51 +02:00
.get_size = stdio_flash_get_size,
2017-03-11 13:32:30 +00:00
.read = stdio_flash_read,
.write = stdio_flash_write,
.copy = default_flash_copy,
.is_erased = default_flash_is_erased,
2017-03-11 13:32:30 +00:00
.erase = stdio_flash_erase,
};
2017-04-07 19:35:51 +02:00
static size_t stdio_flash_get_size(struct flash_dev *dev)
{
struct stdio_flash_priv *priv = dev->priv;
return priv->size;
}
static size_t stdio_flash_read(struct flash_dev *dev, uint32_t addr, void *data,
2017-03-11 13:32:30 +00:00
size_t len)
{
struct stdio_flash_priv *priv = dev->priv;
addr = min(addr, priv->size);
len = min(len, priv->size - addr);
if (!len)
return 0;
2017-03-11 13:32:30 +00:00
memcpy(data, priv->data + addr, len);
return len;
2017-03-11 13:32:30 +00:00
}
static size_t stdio_flash_write(struct flash_dev *dev, uint32_t addr,
2017-03-11 13:32:30 +00:00
const void *data, size_t len)
{
struct stdio_flash_priv *priv = dev->priv;
const char *src;
char *dst;
size_t i;
addr = min(addr, priv->size);
len = min(len, priv->size - addr);
if (!len)
return 0;
2017-03-11 13:32:30 +00:00
src = data;
dst = priv->data + addr;
for (i = 0; i < len; ++i) {
dst[i] &= src[i];
}
return len;
2017-03-11 13:32:30 +00:00
}
2017-05-19 17:04:45 +02:00
static int stdio_flash_erase(struct flash_dev *dev, uint32_t addr)
{
2017-03-11 13:32:30 +00:00
struct stdio_flash_priv *priv = dev->priv;
2017-05-19 17:04:45 +02:00
memset(priv->data + (addr << dev->log2_block_size), 0xff,
1 << dev->log2_block_size);
2017-03-11 13:32:30 +00:00
return 0;
}
struct flash_dev *flash_probe(void)
{
struct stat st;
2017-03-11 13:32:30 +00:00
struct flash_dev *dev;
struct stdio_flash_priv *priv;
if (!(dev = malloc(sizeof *dev)))
return NULL;
if (!(priv = malloc(sizeof *priv)))
goto err_free_dev;
if (!(priv->fp = fopen("flash.img", "r+b")))
2017-03-11 13:32:30 +00:00
goto err_free_priv;
if (fstat(fileno(priv->fp), &st) < 0)
goto err_close_fp;
priv->size = st.st_size;
if (!(priv->data = mmap(NULL, priv->size, PROT_READ | PROT_WRITE,
2017-03-31 16:11:12 +02:00
MAP_SHARED, fileno(priv->fp), 0)))
goto err_close_fp;
2017-03-11 13:32:30 +00:00
dev->priv = priv;
dev->ops = &stdio_flash_ops;
2017-05-19 17:04:45 +02:00
dev->log2_block_size = ilog2(4 * KIB);
2017-03-11 13:32:30 +00:00
return dev;
err_close_fp:
fclose(priv->fp);
2017-03-11 13:32:30 +00:00
err_free_priv:
free(priv);
err_free_dev:
free(dev);
return NULL;
}
static void stdio_flash_release(struct flash_dev *dev)
{
struct stdio_flash_priv *priv;
if (!dev)
return;
priv = dev->priv;
munmap(priv->data, priv->size);
fclose(priv->fp);
free(dev->priv);
free(dev);
}