blob: 7154caa5f15e22c2335ce9fecdc06f11003a32fb [file] [log] [blame]
/*
* Copyright (c) 2017-2020 Nordic Semiconductor ASA
* Copyright (c) 2015 Runtime Inc
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <limits.h>
#include <stdlib.h>
#include <zephyr/fs/fcb.h>
#include "fcb_priv.h"
#include "string.h"
#include <errno.h>
#include <zephyr/device.h>
#include <zephyr/drivers/flash.h>
uint8_t
fcb_get_align(const struct fcb *fcb)
{
uint8_t align;
if (fcb->fap == NULL) {
return 0;
}
align = flash_area_align(fcb->fap);
return align;
}
int fcb_flash_read(const struct fcb *fcb, const struct flash_sector *sector,
off_t off, void *dst, size_t len)
{
int rc;
if (off + len > sector->fs_size) {
return -EINVAL;
}
if (fcb->fap == NULL) {
return -EIO;
}
rc = flash_area_read(fcb->fap, sector->fs_off + off, dst, len);
if (rc != 0) {
return -EIO;
}
return 0;
}
int fcb_flash_write(const struct fcb *fcb, const struct flash_sector *sector,
off_t off, const void *src, size_t len)
{
int rc;
if (off + len > sector->fs_size) {
return -EINVAL;
}
if (fcb->fap == NULL) {
return -EIO;
}
rc = flash_area_write(fcb->fap, sector->fs_off + off, src, len);
if (rc != 0) {
return -EIO;
}
return 0;
}
int
fcb_erase_sector(const struct fcb *fcb, const struct flash_sector *sector)
{
int rc;
if (fcb->fap == NULL) {
return -EIO;
}
rc = flash_area_flatten(fcb->fap, sector->fs_off, sector->fs_size);
if (rc != 0) {
return -EIO;
}
return 0;
}
int
fcb_init(int f_area_id, struct fcb *fcb)
{
struct flash_sector *sector;
int rc;
int i;
uint8_t align;
int oldest = -1, newest = -1;
struct flash_sector *oldest_sector = NULL, *newest_sector = NULL;
struct fcb_disk_area fda;
const struct flash_parameters *fparam;
if (!fcb->f_sectors || fcb->f_sector_cnt - fcb->f_scratch_cnt < 1) {
return -EINVAL;
}
rc = flash_area_open(f_area_id, &fcb->fap);
if (rc != 0) {
return -EINVAL;
}
fparam = flash_get_parameters(fcb->fap->fa_dev);
fcb->f_erase_value = fparam->erase_value;
align = fcb_get_align(fcb);
if (align == 0U) {
return -EINVAL;
}
/* Fill last used, first used */
for (i = 0; i < fcb->f_sector_cnt; i++) {
sector = &fcb->f_sectors[i];
rc = fcb_sector_hdr_read(fcb, sector, &fda);
if (rc < 0) {
return rc;
}
if (rc == 0) {
continue;
}
if (oldest < 0) {
oldest = newest = fda.fd_id;
oldest_sector = newest_sector = sector;
continue;
}
if (FCB_ID_GT(fda.fd_id, newest)) {
newest = fda.fd_id;
newest_sector = sector;
} else if (FCB_ID_GT(oldest, fda.fd_id)) {
oldest = fda.fd_id;
oldest_sector = sector;
}
}
if (oldest < 0) {
/*
* No initialized areas.
*/
oldest_sector = newest_sector = &fcb->f_sectors[0];
rc = fcb_sector_hdr_init(fcb, oldest_sector, 0);
if (rc) {
return rc;
}
newest = oldest = 0;
}
fcb->f_align = align;
fcb->f_oldest = oldest_sector;
fcb->f_active.fe_sector = newest_sector;
fcb->f_active.fe_elem_off = fcb_len_in_flash(fcb, sizeof(struct fcb_disk_area));
fcb->f_active_id = newest;
while (1) {
rc = fcb_getnext_in_sector(fcb, &fcb->f_active);
if (rc == -ENOTSUP) {
rc = 0;
break;
}
if (rc != 0) {
break;
}
}
k_mutex_init(&fcb->f_mtx);
return rc;
}
int
fcb_free_sector_cnt(struct fcb *fcb)
{
int i;
struct flash_sector *fa;
fa = fcb->f_active.fe_sector;
for (i = 0; i < fcb->f_sector_cnt; i++) {
fa = fcb_getnext_sector(fcb, fa);
if (fa == fcb->f_oldest) {
break;
}
}
return i;
}
int
fcb_is_empty(struct fcb *fcb)
{
return (fcb->f_active.fe_sector == fcb->f_oldest &&
fcb->f_active.fe_elem_off == fcb_len_in_flash(fcb, sizeof(struct fcb_disk_area)));
}
/**
* Length of an element is encoded in 1 or 2 bytes.
* 1 byte for lengths < 128 bytes, 2 bytes for < 16384.
*
* The storage of length has been originally designed to work with 0xff erasable
* flash devices and gives length 0xffff special meaning: that there is no value
* written; this is smart way to utilize value in non-written flash to figure
* out where data ends. Additionally it sets highest bit of first byte of
* the length to 1, to mark that there is second byte to be read.
* Above poses some problems when non-0xff erasable flash is used. To solve
* the problem all length values are xored with not of erase value for given
* flash:
* len' = len ^ ~erase_value;
* To obtain original value, the logic is reversed:
* len = len' ^ ~erase_value;
*
* In case of 0xff erased flash this does not modify data that is written to
* flash; in case of other flash devices, e.g. that erase to 0x00, it allows
* to correctly use the first bit of byte to figure out how many bytes are there
* and if there is any data at all or both bytes are equal to erase value.
*/
int
fcb_put_len(const struct fcb *fcb, uint8_t *buf, uint16_t len)
{
if (len < 0x80) {
buf[0] = len ^ ~fcb->f_erase_value;
return 1;
} else if (len <= FCB_MAX_LEN) {
buf[0] = (len | 0x80) ^ ~fcb->f_erase_value;
buf[1] = (len >> 7) ^ ~fcb->f_erase_value;
return 2;
} else {
return -EINVAL;
}
}
int
fcb_get_len(const struct fcb *fcb, uint8_t *buf, uint16_t *len)
{
int rc;
uint8_t buf0_xor;
uint8_t buf1_xor;
buf0_xor = buf[0] ^ ~fcb->f_erase_value;
if (buf0_xor & 0x80) {
if ((buf[0] == fcb->f_erase_value) &&
(buf[1] == fcb->f_erase_value)) {
return -ENOTSUP;
}
buf1_xor = buf[1] ^ ~fcb->f_erase_value;
*len = (uint16_t)((buf0_xor & 0x7f) | ((uint16_t)buf1_xor << 7));
rc = 2;
} else {
*len = (uint16_t)(buf0_xor);
rc = 1;
}
return rc;
}
/**
* Initialize erased sector for use.
*/
int
fcb_sector_hdr_init(struct fcb *fcb, struct flash_sector *sector, uint16_t id)
{
struct fcb_disk_area fda;
int rc;
fda.fd_magic = fcb_flash_magic(fcb);
fda.fd_ver = fcb->f_version;
fda._pad = fcb->f_erase_value;
fda.fd_id = id;
rc = fcb_flash_write(fcb, sector, 0, &fda, sizeof(fda));
if (rc != 0) {
return -EIO;
}
return 0;
}
/**
* Checks whether FCB sector contains data or not.
* Returns <0 in error.
* Returns 0 if sector is unused;
* Returns 1 if sector has data.
*/
int fcb_sector_hdr_read(struct fcb *fcb, struct flash_sector *sector,
struct fcb_disk_area *fdap)
{
struct fcb_disk_area fda;
int rc;
if (!fdap) {
fdap = &fda;
}
rc = fcb_flash_read(fcb, sector, 0, fdap, sizeof(*fdap));
if (rc) {
return -EIO;
}
if (fdap->fd_magic == MK32(fcb->f_erase_value)) {
return 0;
}
if (fdap->fd_magic != fcb_flash_magic(fcb)) {
return -ENOMSG;
}
return 1;
}
/**
* Finds the fcb entry that gives back upto n entries at the end.
* @param0 ptr to fcb
* @param1 n number of fcb entries the user wants to get
* @param2 ptr to the fcb_entry to be returned
* @return 0 on there are any fcbs aviable; -ENOENT otherwise
*/
int
fcb_offset_last_n(struct fcb *fcb, uint8_t entries,
struct fcb_entry *last_n_entry)
{
struct fcb_entry loc;
int i;
int rc;
/* assure a minimum amount of entries */
if (!entries) {
entries = 1U;
}
i = 0;
(void)memset(&loc, 0, sizeof(loc));
while (!fcb_getnext(fcb, &loc)) {
if (i == 0) {
/* Start from the beginning of fcb entries */
*last_n_entry = loc;
}
/* Update last_n_entry after n entries and keep updating */
else if (i > (entries - 1)) {
rc = fcb_getnext(fcb, last_n_entry);
if (rc) {
/* A fcb history must have been erased,
* wanted entry doesn't exist anymore.
*/
return -ENOENT;
}
}
i++;
}
return (i == 0) ? -ENOENT : 0;
}
/**
* Clear fcb
* @param fcb
* @return 0 on success; non-zero on failure
*/
int
fcb_clear(struct fcb *fcb)
{
int rc;
rc = 0;
while (!fcb_is_empty(fcb)) {
rc = fcb_rotate(fcb);
if (rc) {
break;
}
}
return rc;
}