123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137 |
- /*
- * Copyright (c) 2017-2020 Nordic Semiconductor ASA
- * Copyright (c) 2015 Runtime Inc
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #include <stddef.h>
- #include <string.h>
- #include <fs/fcb.h>
- #include "fcb_priv.h"
- static struct flash_sector *
- fcb_new_sector(struct fcb *fcb, int cnt)
- {
- struct flash_sector *prev;
- struct flash_sector *cur;
- int i;
- prev = NULL;
- i = 0;
- cur = fcb->f_active.fe_sector;
- do {
- cur = fcb_getnext_sector(fcb, cur);
- if (!prev) {
- prev = cur;
- }
- if (cur == fcb->f_oldest) {
- return NULL;
- }
- } while (i++ < cnt);
- return prev;
- }
- /*
- * Take one of the scratch blocks into use, if at all possible.
- */
- int
- fcb_append_to_scratch(struct fcb *fcb)
- {
- struct flash_sector *sector;
- int rc;
- sector = fcb_new_sector(fcb, 0);
- if (!sector) {
- return -ENOSPC;
- }
- rc = fcb_sector_hdr_init(fcb, sector, fcb->f_active_id + 1);
- if (rc) {
- return rc;
- }
- fcb->f_active.fe_sector = sector;
- fcb->f_active.fe_elem_off = sizeof(struct fcb_disk_area);
- fcb->f_active_id++;
- return 0;
- }
- int
- fcb_append(struct fcb *fcb, uint16_t len, struct fcb_entry *append_loc)
- {
- struct flash_sector *sector;
- struct fcb_entry *active;
- int cnt;
- int rc;
- uint8_t tmp_str[8];
- cnt = fcb_put_len(fcb, tmp_str, len);
- if (cnt < 0) {
- return cnt;
- }
- cnt = fcb_len_in_flash(fcb, cnt);
- len = fcb_len_in_flash(fcb, len) + fcb_len_in_flash(fcb, FCB_CRC_SZ);
- __ASSERT_NO_MSG(cnt <= sizeof(tmp_str));
- rc = k_mutex_lock(&fcb->f_mtx, K_FOREVER);
- if (rc) {
- return -EINVAL;
- }
- active = &fcb->f_active;
- if (active->fe_elem_off + len + cnt > active->fe_sector->fs_size) {
- sector = fcb_new_sector(fcb, fcb->f_scratch_cnt);
- if (!sector || (sector->fs_size <
- sizeof(struct fcb_disk_area) + len + cnt)) {
- rc = -ENOSPC;
- goto err;
- }
- rc = fcb_sector_hdr_init(fcb, sector, fcb->f_active_id + 1);
- if (rc) {
- goto err;
- }
- fcb->f_active.fe_sector = sector;
- fcb->f_active.fe_elem_off = sizeof(struct fcb_disk_area);
- fcb->f_active_id++;
- }
- rc = fcb_flash_write(fcb, active->fe_sector, active->fe_elem_off, tmp_str, cnt);
- if (rc) {
- rc = -EIO;
- goto err;
- }
- append_loc->fe_sector = active->fe_sector;
- append_loc->fe_elem_off = active->fe_elem_off;
- append_loc->fe_data_off = active->fe_elem_off + cnt;
- active->fe_elem_off = append_loc->fe_data_off + len;
- k_mutex_unlock(&fcb->f_mtx);
- return 0;
- err:
- k_mutex_unlock(&fcb->f_mtx);
- return rc;
- }
- int
- fcb_append_finish(struct fcb *fcb, struct fcb_entry *loc)
- {
- int rc;
- uint8_t crc8[fcb->f_align];
- off_t off;
- (void)memset(crc8, 0xFF, sizeof(crc8));
- rc = fcb_elem_crc8(fcb, loc, &crc8[0]);
- if (rc) {
- return rc;
- }
- off = loc->fe_data_off + fcb_len_in_flash(fcb, loc->fe_data_len);
- rc = fcb_flash_write(fcb, loc->fe_sector, off, crc8, fcb->f_align);
- if (rc) {
- return -EIO;
- }
- return 0;
- }
|