a91ec6eb61
Original work by @ourairquality * Sysparam threadsafe and SPI access * Sysparam test cases * Fix for negative int8 * Sysparam getting bool without memory allocation. Bool tests. * SPI flash refactoring. * Extract common spiflash.c into core. * Use spiflash.c in sysparam. * Use memcpy in spiflash.c insted of hand-written version. * Tests for spiflash.c
187 lines
4.5 KiB
C
187 lines
4.5 KiB
C
/**
|
|
* ESP8266 SPIFFS HAL configuration.
|
|
*
|
|
* Part of esp-open-rtos
|
|
* Copyright (c) 2016 sheinz https://github.com/sheinz
|
|
* MIT License
|
|
*/
|
|
#include "esp_spiffs.h"
|
|
#include "spiffs.h"
|
|
#include <spiflash.h>
|
|
#include <stdbool.h>
|
|
#include <esp/uart.h>
|
|
#include <fcntl.h>
|
|
|
|
spiffs fs;
|
|
|
|
typedef struct {
|
|
void *buf;
|
|
uint32_t size;
|
|
} fs_buf_t;
|
|
|
|
static spiffs_config config = {0};
|
|
static fs_buf_t work_buf = {0};
|
|
static fs_buf_t fds_buf = {0};
|
|
static fs_buf_t cache_buf = {0};
|
|
|
|
/**
|
|
* Number of file descriptors opened at the same time
|
|
*/
|
|
#define ESP_SPIFFS_FD_NUMBER 5
|
|
|
|
#define ESP_SPIFFS_CACHE_PAGES 5
|
|
|
|
static s32_t esp_spiffs_read(u32_t addr, u32_t size, u8_t *dst)
|
|
{
|
|
if (!spiflash_read(addr, dst, size)) {
|
|
return SPIFFS_ERR_INTERNAL;
|
|
}
|
|
|
|
return SPIFFS_OK;
|
|
}
|
|
|
|
static s32_t esp_spiffs_write(u32_t addr, u32_t size, u8_t *src)
|
|
{
|
|
if (!spiflash_write(addr, src, size)) {
|
|
return SPIFFS_ERR_INTERNAL;
|
|
}
|
|
|
|
return SPIFFS_OK;
|
|
}
|
|
|
|
static s32_t esp_spiffs_erase(u32_t addr, u32_t size)
|
|
{
|
|
uint32_t sectors = size / SPI_FLASH_SECTOR_SIZE;
|
|
|
|
for (uint32_t i = 0; i < sectors; i++) {
|
|
if (!spiflash_erase_sector(addr + (SPI_FLASH_SECTOR_SIZE * i))) {
|
|
return SPIFFS_ERR_INTERNAL;
|
|
}
|
|
}
|
|
|
|
return SPIFFS_OK;
|
|
}
|
|
|
|
#if SPIFFS_SINGLETON == 1
|
|
void esp_spiffs_init()
|
|
{
|
|
#else
|
|
void esp_spiffs_init(uint32_t addr, uint32_t size)
|
|
{
|
|
config.phys_addr = addr;
|
|
config.phys_size = size;
|
|
|
|
config.phys_erase_block = SPIFFS_ESP_ERASE_SIZE;
|
|
config.log_page_size = SPIFFS_LOG_PAGE_SIZE;
|
|
config.log_block_size = SPIFFS_LOG_BLOCK_SIZE;
|
|
|
|
// Initialize fs.cfg so the following helper functions work correctly
|
|
memcpy(&fs.cfg, &config, sizeof(spiffs_config));
|
|
#endif
|
|
work_buf.size = 2 * SPIFFS_LOG_PAGE_SIZE;
|
|
fds_buf.size = SPIFFS_buffer_bytes_for_filedescs(&fs, ESP_SPIFFS_FD_NUMBER);
|
|
cache_buf.size= SPIFFS_buffer_bytes_for_cache(&fs, ESP_SPIFFS_CACHE_PAGES);
|
|
|
|
work_buf.buf = malloc(work_buf.size);
|
|
fds_buf.buf = malloc(fds_buf.size);
|
|
cache_buf.buf = malloc(cache_buf.size);
|
|
|
|
config.hal_read_f = esp_spiffs_read;
|
|
config.hal_write_f = esp_spiffs_write;
|
|
config.hal_erase_f = esp_spiffs_erase;
|
|
|
|
config.fh_ix_offset = 3;
|
|
|
|
}
|
|
|
|
void esp_spiffs_deinit()
|
|
{
|
|
free(work_buf.buf);
|
|
work_buf.buf = 0;
|
|
|
|
free(fds_buf.buf);
|
|
fds_buf.buf = 0;
|
|
|
|
free(cache_buf.buf);
|
|
cache_buf.buf = 0;
|
|
}
|
|
|
|
int32_t esp_spiffs_mount()
|
|
{
|
|
printf("SPIFFS memory, work_buf_size=%d, fds_buf_size=%d, cache_buf_size=%d\n",
|
|
work_buf.size, fds_buf.size, cache_buf.size);
|
|
|
|
int32_t err = SPIFFS_mount(&fs, &config, (uint8_t*)work_buf.buf,
|
|
(uint8_t*)fds_buf.buf, fds_buf.size,
|
|
cache_buf.buf, cache_buf.size, 0);
|
|
|
|
if (err != SPIFFS_OK) {
|
|
printf("Error spiffs mount: %d\n", err);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
// This implementation replaces implementation in core/newlib_syscals.c
|
|
long _write_filesystem_r(struct _reent *r, int fd, const char *ptr, int len )
|
|
{
|
|
return SPIFFS_write(&fs, (spiffs_file)fd, (char*)ptr, len);
|
|
}
|
|
|
|
// This implementation replaces implementation in core/newlib_syscals.c
|
|
long _read_filesystem_r( struct _reent *r, int fd, char *ptr, int len )
|
|
{
|
|
return SPIFFS_read(&fs, (spiffs_file)fd, ptr, len);
|
|
}
|
|
|
|
int _open_r(struct _reent *r, const char *pathname, int flags, int mode)
|
|
{
|
|
uint32_t spiffs_flags = SPIFFS_RDONLY;
|
|
|
|
if (flags & O_CREAT) spiffs_flags |= SPIFFS_CREAT;
|
|
if (flags & O_APPEND) spiffs_flags |= SPIFFS_APPEND;
|
|
if (flags & O_TRUNC) spiffs_flags |= SPIFFS_TRUNC;
|
|
if (flags & O_RDONLY) spiffs_flags |= SPIFFS_RDONLY;
|
|
if (flags & O_WRONLY) spiffs_flags |= SPIFFS_WRONLY;
|
|
if (flags & O_EXCL) spiffs_flags |= SPIFFS_EXCL;
|
|
/* if (flags & O_DIRECT) spiffs_flags |= SPIFFS_DIRECT; no support in newlib */
|
|
|
|
return SPIFFS_open(&fs, pathname, spiffs_flags, mode);
|
|
}
|
|
|
|
int _close_r(struct _reent *r, int fd)
|
|
{
|
|
return SPIFFS_close(&fs, (spiffs_file)fd);
|
|
}
|
|
|
|
int _unlink_r(struct _reent *r, const char *path)
|
|
{
|
|
return SPIFFS_remove(&fs, path);
|
|
}
|
|
|
|
int _fstat_r(struct _reent *r, int fd, void *buf)
|
|
{
|
|
spiffs_stat s;
|
|
struct stat *sb = (struct stat*)buf;
|
|
|
|
int result = SPIFFS_fstat(&fs, (spiffs_file)fd, &s);
|
|
sb->st_size = s.size;
|
|
|
|
return result;
|
|
}
|
|
|
|
int _stat_r(struct _reent *r, const char *pathname, void *buf)
|
|
{
|
|
spiffs_stat s;
|
|
struct stat *sb = (struct stat*)buf;
|
|
|
|
int result = SPIFFS_stat(&fs, pathname, &s);
|
|
sb->st_size = s.size;
|
|
|
|
return result;
|
|
}
|
|
|
|
off_t _lseek_r(struct _reent *r, int fd, off_t offset, int whence)
|
|
{
|
|
return SPIFFS_lseek(&fs, (spiffs_file)fd, offset, whence);
|
|
}
|