nuttx/fs/spiffs/src/spiffs_gc.c
Petro Karashchenko d499ac9d58 nuttx: fix multiple 'FAR', 'CODE' and style issues
Signed-off-by: Petro Karashchenko <petro.karashchenko@gmail.com>
2024-08-25 19:22:15 +08:00

1277 lines
43 KiB
C

/****************************************************************************
* fs/spiffs/src/spiffs_gc.c
*
* Copyright (C) 2018 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <gnutt@nuttx.org>
*
* This is a port of version 0.3.7 of SPIFFS by Peter Andersion. That
* version was originally released under the MIT license but is here re-
* released under the NuttX BSD license.
*
* Copyright (c) 2013-2017 Peter Andersson (pelleplutt1976@gmail.com)
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* 3. Neither the name NuttX nor the names of its contributors may be
* used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <stdint.h>
#include <stdbool.h>
#include <string.h>
#include <debug.h>
#include <inttypes.h>
#include "spiffs.h"
#include "spiffs_core.h"
#include "spiffs_cache.h"
#include "spiffs_gc.h"
/****************************************************************************
* Private Types
****************************************************************************/
enum spiffs_gc_clean_state_e
{
FIND_OBJ_DATA,
MOVE_OBJ_DATA,
MOVE_OBJ_NDX,
FINISHED
};
struct spiffs_gc_s
{
enum spiffs_gc_clean_state_e state;
int16_t cur_objid;
int16_t cur_objndx_spndx;
int16_t cur_objndx_pgndx;
int16_t cur_data_pgndx;
int stored_scan_entry_index;
bool objid_found;
};
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Name: spiffs_gc_erase_block
*
* Description:
* Erases a logical block and updates the erase counter.
* If cache is enabled, all pages that might be cached in this block
* is dropped.
*
* Input Parameters:
* fs - A reference to the SPIFFS volume object instance
* blkndx - The block index to erase
*
* Returned Value:
* Zero (OK) is returned on success; A negated errno value is returned on
* any failure.
*
****************************************************************************/
static int spiffs_gc_erase_block(FAR struct spiffs_s *fs, int16_t blkndx)
{
int ret;
int i;
spiffs_gcinfo("Erase block=%04x\n", blkndx);
/* Erase the block */
ret = spiffs_erase_block(fs, blkndx);
if (ret < 0)
{
ferr("ERROR: spiffs_erase_block() blkndx=%d failed: %d\n",
blkndx, ret);
}
/* Then remove the pages from the cache. */
for (i = 0; i < SPIFFS_GEO_PAGES_PER_BLOCK(fs); i++)
{
spiffs_cache_drop_page(fs, SPIFFS_PAGE_FOR_BLOCK(fs, blkndx) + i);
}
return ret;
}
/****************************************************************************
* Name: spiffs_gc_epage_stats
*
* Description:
* Checks if garbage collecting is necessary. If so a candidate block is
* found, cleansed and erased
*
* This function will try to make room for given amount of bytes in the
* filesystem by moving pages and erasing blocks.
*
* If it is physically impossible, err_no will be set to -ENOSPC. If
* there already is this amount (or more) of free space, SPIFFS_gc will
* silently return. It is recommended to call statfs() before invoking
* this method in order to determine what amount of bytes to give.
*
* NB: the garbage collector is automatically called when spiffs needs free
* pages. The reason for this function is to give possibility to do
* background tidying when user knows the system is idle.
*
* Input Parameters:
* fs the file system struct
* len amount of data that should be freed
*
* Returned Value:
* Zero (OK) is returned on success; A negated errno value is returned on
* any failure. -ENODATA is returned if no blocks were deleted.
*
****************************************************************************/
/* Updates page statistics for a block that is about to be erased */
static int spiffs_gc_epage_stats(FAR struct spiffs_s *fs, int16_t blkndx)
{
FAR int16_t *objlu_buf = (FAR int16_t *)fs->lu_work;
uint32_t dele = 0;
uint32_t allo = 0;
int entries_per_page = (SPIFFS_GEO_PAGE_SIZE(fs) / sizeof(int16_t));
int cur_entry = 0;
int obj_lookup_page = 0;
int ret = OK;
/* Check each object lookup page */
while (ret >= 0 && obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs))
{
int entry_offset = obj_lookup_page * entries_per_page;
ret = spiffs_cache_read(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ, 0,
blkndx * SPIFFS_GEO_BLOCK_SIZE(fs) +
SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
SPIFFS_GEO_PAGE_SIZE(fs), fs->lu_work);
/* Check each entry */
while (ret >= 0 &&
cur_entry - entry_offset < entries_per_page &&
cur_entry <
(int)(SPIFFS_GEO_PAGES_PER_BLOCK(fs) -
SPIFFS_OBJ_LOOKUP_PAGES(fs)))
{
int16_t id = objlu_buf[cur_entry - entry_offset];
if (id == SPIFFS_OBJID_FREE)
{
}
else if (id == SPIFFS_OBJID_DELETED)
{
dele++;
}
else
{
allo++;
}
cur_entry++;
}
obj_lookup_page++;
}
spiffs_gcinfo("Wipe pallo=%" PRIu32 " pdele=%" PRIu32 "\n", allo, dele);
fs->alloc_pages -= allo;
fs->deleted_pages -= dele;
return ret;
}
/****************************************************************************
* Name: spiffs_gc_find_candidate
*
* Description:
* Finds block candidates to erase
*
* Input Parameters:
* fs - A reference to the SPIFFS volume object instance
* block_candidates - Memory to return an array of candidates
* candidate_count - Memory to return the number of candidates
* fs_crammed - True: Filesystem is full
*
* Returned Value:
* Zero (OK) is returned on success; A negated errno value is returned on
* any failure. -ENODATA is returned if no blocks were deleted.
*
****************************************************************************/
static int spiffs_gc_find_candidate(FAR struct spiffs_s *fs,
FAR int16_t **block_candidates,
FAR int *candidate_count,
bool fs_crammed)
{
FAR int32_t *cand_scores;
FAR int16_t *objlu_buf = (FAR int16_t *)fs->lu_work;
FAR int16_t *cand_blocks;
uint32_t blocks = SPIFFS_GEO_BLOCK_COUNT(fs);
int16_t cur_block = 0;
uint32_t cur_block_addr = 0;
int entries_per_page;
int max_candidates;
int cur_entry = 0;
int ret = OK;
/* Using fs->work area as sorted candidate memory,
* (int16_t)cand_blkndx/(int32_t)score
*/
max_candidates =
MIN(SPIFFS_GEO_BLOCK_COUNT(fs),
(SPIFFS_GEO_PAGE_SIZE(fs) - 8) /
(sizeof(int16_t) + sizeof(int32_t)));
*candidate_count = 0;
memset(fs->work, 0xff, SPIFFS_GEO_PAGE_SIZE(fs));
/* Divide up work area into block indices and scores */
cand_blocks = (FAR int16_t *)fs->work;
cand_scores =
(FAR int32_t *)(fs->work + max_candidates * sizeof(int16_t));
/* Align cand_scores on int32_t boundary */
cand_scores =
(FAR int32_t *)(((intptr_t)cand_scores + sizeof(intptr_t) - 1) &
~(sizeof(intptr_t) - 1));
*block_candidates = cand_blocks;
entries_per_page = (SPIFFS_GEO_PAGE_SIZE(fs) / sizeof(int16_t));
/* Check each block */
while (ret >= 0 && blocks--)
{
uint16_t deleted_pages_in_block = 0;
uint16_t used_pages_in_block = 0;
int obj_lookup_page = 0;
/* check each object lookup page */
while (ret >= 0 &&
obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs))
{
int entry_offset = obj_lookup_page * entries_per_page;
ret = spiffs_cache_read(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
0, cur_block_addr +
SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
SPIFFS_GEO_PAGE_SIZE(fs), fs->lu_work);
/* Check each entry */
while (ret >= 0 &&
cur_entry - entry_offset < entries_per_page &&
cur_entry <
(int)(SPIFFS_GEO_PAGES_PER_BLOCK(fs) -
SPIFFS_OBJ_LOOKUP_PAGES(fs)))
{
int16_t id = objlu_buf[cur_entry - entry_offset];
if (id == SPIFFS_OBJID_FREE)
{
/* When a free entry is encountered, scan logic ensures
* that all following entries are free also
*/
ret = 1; /* Kill object lu loop */
break;
}
else if (id == SPIFFS_OBJID_DELETED)
{
deleted_pages_in_block++;
}
else
{
used_pages_in_block++;
}
cur_entry++;
}
obj_lookup_page++;
}
if (ret == 1)
{
ret = OK;
}
/* Calculate score and insert into candidate table stoneage sort, but
* probably not so many blocks
*/
if (ret >= 0 /* && deleted_pages_in_block > 0 */)
{
int32_t score;
int16_t erase_count;
int16_t erase_age;
int candndx;
/* Read erase count */
ret = spiffs_cache_read(fs, SPIFFS_OP_C_READ | SPIFFS_OP_T_OBJ_LU2,
0, SPIFFS_ERASE_COUNT_PADDR(fs, cur_block),
sizeof(int16_t), (uint8_t *)&erase_count);
if (ret < 0)
{
ferr("ERROR: spiffs_cache_read() failed: %d\n", ret);
return ret;
}
/* Calculate the erase age */
if (fs->max_erase_count > erase_count)
{
erase_age = fs->max_erase_count - erase_count;
}
else
{
erase_age =
SPIFFS_OBJID_FREE - (erase_count - fs->max_erase_count);
}
score =
deleted_pages_in_block * CONFIG_SPIFFS_GC_DELWGT +
used_pages_in_block * CONFIG_SPIFFS_GC_USEDWGT +
erase_age * (fs_crammed ? 0 : CONFIG_SPIFFS_GC_ERASEAGEWGT);
candndx = 0;
spiffs_gcinfo("blkndx=%04x del=%d use=%d score=%" PRIu32 "\n",
cur_block, deleted_pages_in_block,
used_pages_in_block, score);
while (candndx < max_candidates)
{
if (cand_blocks[candndx] == (int16_t) - 1)
{
cand_blocks[candndx] = cur_block;
cand_scores[candndx] = score;
break;
}
else if (cand_scores[candndx] < score)
{
int reorder_candndx = max_candidates - 2;
while (reorder_candndx >= candndx)
{
cand_blocks[reorder_candndx + 1] =
cand_blocks[reorder_candndx];
cand_scores[reorder_candndx + 1] =
cand_scores[reorder_candndx];
reorder_candndx--;
}
cand_blocks[candndx] = cur_block;
cand_scores[candndx] = score;
break;
}
candndx++;
}
(*candidate_count)++;
}
cur_entry = 0;
cur_block++;
cur_block_addr += SPIFFS_GEO_BLOCK_SIZE(fs);
}
return ret;
}
/****************************************************************************
* Name: spiffs_gc_clean
*
* Description:
* Empties given block by moving all data into free pages of another block
* Strategy:
* loop:
* - Scan object lookup for object data pages.
* - For first found id, check spndx and load corresponding object index
* page to memory.
* - Push object scan lookup entry index.
* Rescan object lookup, find data pages with same id and referenced
* by same object index.
* Move data page, update object index in memory.
* When reached end of lookup, store updated object index.
* - Pop object scan lookup entry index.
* - Repeat loop until end of object lookup.
* - Scan object lookup again for remaining object index pages, move to
* new page in other block.
*
* Input Parameters:
* fs the file system struct
* len amount of data that should be freed
*
* Returned Value:
* Zero (OK) is returned on success; A negated errno value is returned on
* any failure. -ENODATA is returned if no blocks were deleted.
*
****************************************************************************/
static int spiffs_gc_clean(FAR struct spiffs_s *fs, int16_t blkndx)
{
const int entries_per_page = (SPIFFS_GEO_PAGE_SIZE(fs) / sizeof(int16_t));
int ret = OK;
/* This is the global localizer being pushed and popped */
struct spiffs_gc_s gc; /* Our stack frame/state */
FAR int16_t *objlu_buf;
FAR struct spiffs_pgobj_ndxheader_s *objhdr;
FAR struct spiffs_page_objndx_s *objndx;
int16_t cur_pgndx = 0;
int cur_entry = 0;
spiffs_gcinfo("Cleaning block %04x\n", blkndx);
memset(&gc, 0, sizeof(struct spiffs_gc_s));
gc.state = FIND_OBJ_DATA;
objlu_buf = (FAR int16_t *)fs->lu_work;
objhdr = (FAR struct spiffs_pgobj_ndxheader_s *)fs->work;
objndx = (struct spiffs_page_objndx_s *)fs->work;
if (fs->free_blkndx == blkndx)
{
/* Move free cursor to next block, cannot use free pages from the block
* we want to clean
*/
fs->free_blkndx = (blkndx + 1) % SPIFFS_GEO_BLOCK_COUNT(fs);
fs->free_entry = 0;
spiffs_gcinfo("Move free cursor to block=%04x\n", fs->free_blkndx);
}
while (ret >= 0 && gc.state != FINISHED)
{
int obj_lookup_page;
uint8_t scan;
spiffs_gcinfo("state=%d entry=%d\n", gc.state, cur_entry);
gc.objid_found = false; /* Reset (to no found data page) */
/* Scan through lookup pages */
obj_lookup_page = cur_entry / entries_per_page;
scan = 1;
/* Check each object lookup page */
while (scan && ret >= 0 &&
obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs))
{
int entry_offset = obj_lookup_page * entries_per_page;
ret = spiffs_cache_read(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
0, blkndx * SPIFFS_GEO_BLOCK_SIZE(fs) +
SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
SPIFFS_GEO_PAGE_SIZE(fs), fs->lu_work);
/* Check each object lookup entry */
while (scan && ret >= 0 &&
cur_entry - entry_offset < entries_per_page &&
cur_entry <
(int)(SPIFFS_GEO_PAGES_PER_BLOCK(fs) -
SPIFFS_OBJ_LOOKUP_PAGES(fs)))
{
int16_t id = objlu_buf[cur_entry - entry_offset];
cur_pgndx =
SPIFFS_OBJ_LOOKUP_ENTRY_TO_PGNDX(fs, blkndx, cur_entry);
/* Act upon object id depending on gc state */
switch (gc.state)
{
case FIND_OBJ_DATA:
/* Find a data page. */
if (id != SPIFFS_OBJID_DELETED &&
id != SPIFFS_OBJID_FREE &&
((id & SPIFFS_OBJID_NDXFLAG) == 0))
{
/* Found a data page, stop scanning and handle in
* switch case below
*/
spiffs_gcinfo(
"Found data page, state=%d, objid=%04x\n",
gc.state, id);
gc.objid_found = true;
gc.cur_objid = id;
gc.cur_data_pgndx = cur_pgndx;
scan = 0;
}
break;
case MOVE_OBJ_DATA:
/* Evacuate found data pages for corresponding object index
* we have in memory, update memory representation
*/
if (id == gc.cur_objid)
{
struct spiffs_page_header_s phdr;
ret = spiffs_cache_read(fs,
SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
0, SPIFFS_PAGE_TO_PADDR(fs, cur_pgndx),
sizeof(struct spiffs_page_header_s),
(FAR uint8_t *)&phdr);
if (ret < 0)
{
ferr("ERROR: spiffs_cache_read() failed: %d\n",
ret);
return ret;
}
spiffs_gcinfo("Found data page %04x:%04x @%04x\n",
gc.cur_objid, phdr.spndx, cur_pgndx);
if (SPIFFS_OBJNDX_ENTRY_SPNDX(fs, phdr.spndx) !=
gc.cur_objndx_spndx)
{
spiffs_gcinfo(
"No objndx spndx match, take in another run\n");
}
else
{
int16_t new_data_pgndx;
if (phdr.flags & SPIFFS_PH_FLAG_DELET)
{
/* Move page */
ret = spiffs_page_move(fs, 0, 0, id, &phdr,
cur_pgndx, &new_data_pgndx);
spiffs_gcinfo(
"Move objndx=%04x:%04x page=%04x to %04x\n",
gc.cur_objid, phdr.spndx,
cur_pgndx, new_data_pgndx);
if (ret < 0)
{
ferr("ERROR: spiffs_page_move(): %d\n",
ret);
return ret;
}
/* Move wipes obj_lu, reload it */
ret =
spiffs_cache_read(fs,
SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
0,
blkndx * SPIFFS_GEO_BLOCK_SIZE(fs) +
SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
SPIFFS_GEO_PAGE_SIZE(fs),
fs->lu_work);
if (ret < 0)
{
ferr("ERROR: spiffs_cache_read(): %d\n",
ret);
return ret;
}
}
else
{
/* Page is deleted but not deleted in lookup.
* Scrap it - might seem unnecessary as we will
* erase this block, but we might get aborted
*/
spiffs_gcinfo(
"Wipe objndx=%04x:%04x page=%04x\n",
id, phdr.spndx, cur_pgndx);
ret = spiffs_page_delete(fs, cur_pgndx);
if (ret < 0)
{
ferr("ERROR: spiffs_page_delete(): %d\n",
ret);
return ret;
}
new_data_pgndx = SPIFFS_OBJID_FREE;
}
/* Update memory representation of object index
* page with new data page
*/
if (gc.cur_objndx_spndx == 0)
{
/* Update object index header page */
((FAR int16_t *)((FAR uint8_t *)objhdr +
sizeof(struct spiffs_pgobj_ndxheader_s)))
[phdr.spndx] = new_data_pgndx;
spiffs_gcinfo(
"Wrote page=%04x to objhdr entry=%04x\n",
new_data_pgndx,
(int)SPIFFS_OBJNDX_ENTRY(fs, phdr.spndx));
}
else
{
/* Update object index page */
((FAR int16_t *)((FAR uint8_t *)objndx +
sizeof(struct spiffs_page_objndx_s)))
[SPIFFS_OBJNDX_ENTRY(fs, phdr.spndx)] =
new_data_pgndx;
spiffs_gcinfo(
"Wrote page=%04x to objndx entry=%04x\n",
new_data_pgndx,
(int)SPIFFS_OBJNDX_ENTRY(fs, phdr.spndx));
}
}
}
break;
case MOVE_OBJ_NDX:
/* Find and evacuate object index pages */
if (id != SPIFFS_OBJID_DELETED &&
id != SPIFFS_OBJID_FREE &&
(id & SPIFFS_OBJID_NDXFLAG) != 0)
{
/* Found an index object id */
struct spiffs_page_header_s phdr;
int16_t new_pgndx;
/* Load header */
ret = spiffs_cache_read(fs,
SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
0,
SPIFFS_PAGE_TO_PADDR(fs, cur_pgndx),
sizeof(struct spiffs_page_header_s),
(FAR uint8_t *)&phdr);
if (ret < 0)
{
ferr("ERROR: spiffs_cache_read() failed: %d\n",
ret);
return ret;
}
if (phdr.flags & SPIFFS_PH_FLAG_DELET)
{
/* Move page */
ret = spiffs_page_move(fs, 0, 0, id, &phdr,
cur_pgndx, &new_pgndx);
spiffs_gcinfo(
"Move objndx=%04x:%04x page=%04x to %04x\n",
id, phdr.spndx, cur_pgndx, new_pgndx);
if (ret < 0)
{
ferr("ERROR: spiffs_page_move() failed: %d\n",
ret);
return ret;
}
spiffs_fobj_event(fs,
(FAR struct spiffs_page_objndx_s *)&phdr,
SPIFFS_EV_NDXMOV, id,
phdr.spndx, new_pgndx, 0);
/* Move wipes obj_lu, reload it */
ret =
spiffs_cache_read(fs,
SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
0,
blkndx * SPIFFS_GEO_BLOCK_SIZE(fs) +
SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
SPIFFS_GEO_PAGE_SIZE(fs), fs->lu_work);
if (ret < 0)
{
ferr("ERROR: spiffs_cache_read() failed: %d\n",
ret);
return ret;
}
}
else
{
/* Page is deleted but not deleted in lookup, scrap
* it - might seem unnecessary as we will erase
* this block, but we might get aborted
*/
spiffs_gcinfo("Wipe objndx=%04x:%04x page=%04x\n",
id, phdr.spndx, cur_pgndx);
ret = spiffs_page_delete(fs, cur_pgndx);
if (ret < 0)
{
ferr("ERROR: spiffs_fobj_event() failed: %d\n",
ret);
return ret;
}
spiffs_fobj_event(fs, NULL,
SPIFFS_EV_NDXDEL, id,
phdr.spndx, cur_pgndx, 0);
}
}
break;
default:
scan = 0;
break;
}
cur_entry++;
}
obj_lookup_page++; /* No need to check scan variable here,
* obj_lookup_page is set in start of loop
*/
}
if (ret < 0)
{
break;
}
/* State finalization and switch */
switch (gc.state)
{
case FIND_OBJ_DATA:
if (gc.objid_found)
{
struct spiffs_page_header_s phdr;
int16_t objndx_pgndx;
/* Handle found data page. Find out corresponding objndx page
* and load it to memory
*/
gc.stored_scan_entry_index = cur_entry; /* Push cursor */
cur_entry = 0; /* Sestart scan from start */
gc.state = MOVE_OBJ_DATA;
ret = spiffs_cache_read(fs,
SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
0, SPIFFS_PAGE_TO_PADDR(fs, cur_pgndx),
sizeof(struct spiffs_page_header_s),
(FAR uint8_t *)&phdr);
if (ret < 0)
{
ferr("ERROR: spiffs_fobj_event() failed: %d\n", ret);
return ret;
}
gc.cur_objndx_spndx =
SPIFFS_OBJNDX_ENTRY_SPNDX(fs, phdr.spndx);
spiffs_gcinfo("Find objndx spndx=%04x\n", gc.cur_objndx_spndx);
ret = spiffs_objlu_find_id_and_span(fs,
gc.cur_objid | SPIFFS_OBJID_NDXFLAG,
gc.cur_objndx_spndx, 0,
&objndx_pgndx);
if (ret == -ENOENT)
{
/* On borked systems we might get an ERR_NOT_FOUND here -
* this is handled by simply deleting the page as it is not
* referenced from anywhere
*/
spiffs_gcinfo("objndx not found! Wipe page %04x\n",
gc.cur_data_pgndx);
ret = spiffs_page_delete(fs, gc.cur_data_pgndx);
if (ret < 0)
{
ferr("ERROR: spiffs_page_delete() failed: %d\n", ret);
return ret;
}
/* Then we restore states and continue scanning for data
* pages
*/
cur_entry = gc.stored_scan_entry_index; /* pop cursor */
gc.state = FIND_OBJ_DATA;
break;
}
if (ret < 0)
{
return ret;
}
spiffs_gcinfo("Found object index at page=%04x\n",
objndx_pgndx);
ret = spiffs_cache_read(fs,
SPIFFS_OP_T_OBJ_LU2 | SPIFFS_OP_C_READ,
0,
SPIFFS_PAGE_TO_PADDR(fs, objndx_pgndx),
SPIFFS_GEO_PAGE_SIZE(fs), fs->work);
if (ret < 0)
{
ferr("ERROR: spiffs_cache_read() failed: %d\n", ret);
return ret;
}
/* Cannot allow a gc if the presumed index in fact is no
* index, a check must run or lot of data may be lost
*/
ret = spiffs_validate_objndx(&objndx->phdr,
gc.cur_objid | SPIFFS_OBJID_NDXFLAG,
gc.cur_objndx_spndx);
if (ret < 0)
{
ferr("ERROR: spiffs_validate_objndx() failed: %d\n", ret);
return ret;
}
gc.cur_objndx_pgndx = objndx_pgndx;
}
else
{
/* No more data pages found, passed through all block, start
* evacuating object indices
*/
gc.state = MOVE_OBJ_NDX;
cur_entry = 0; /* Restart entry scan index */
}
break;
case MOVE_OBJ_DATA:
{
int16_t new_objndx_pgndx;
/* Store modified objndx (hdr) page residing in memory now that
* all data pages belonging to this object index and residing in
* the block we want to evacuate
*/
gc.state = FIND_OBJ_DATA;
cur_entry = gc.stored_scan_entry_index; /* pop cursor */
if (gc.cur_objndx_spndx == 0)
{
/* Store object index header page */
ret = spiffs_fobj_update_ndxhdr(fs, 0,
gc.cur_objid | SPIFFS_OBJID_NDXFLAG,
gc.cur_objndx_pgndx, fs->work, 0,
0, &new_objndx_pgndx);
spiffs_gcinfo("Store modified objhdr page=%04x:%04x\n",
new_objndx_pgndx, 0);
if (ret < 0)
{
ferr("ERROR: spiffs_fobj_update_ndxhdr() failed: %d\n",
ret);
return ret;
}
}
else
{
/* Store object index page */
ret = spiffs_page_move(fs, 0, fs->work,
gc.cur_objid | SPIFFS_OBJID_NDXFLAG,
0, gc.cur_objndx_pgndx,
&new_objndx_pgndx);
spiffs_gcinfo("Store modified objndx page=%04x:%04x\n",
new_objndx_pgndx, objndx->phdr.spndx);
if (ret < 0)
{
ferr("ERROR: spiffs_page_move() failed: %d\n", ret);
return ret;
}
spiffs_fobj_event(fs,
(FAR struct spiffs_page_objndx_s *)fs->work,
SPIFFS_EV_NDXUPD, gc.cur_objid,
objndx->phdr.spndx,
new_objndx_pgndx, 0);
}
}
break;
case MOVE_OBJ_NDX:
/* Scanned through all blocks, no more object indices found - our
* work here is done
*/
gc.state = FINISHED;
break;
default:
cur_entry = 0;
break;
}
spiffs_gcinfo("state=%d\n", gc.state);
}
return ret;
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: spiffs_gc_quick
*
* Description:
* Searches for blocks where all entries are deleted - if one is found,
* the block is erased. Compared to the non-quick gc, the quick one ensures
* that no updates are needed on existing objects on pages that are erased.
*
* Input Parameters:
* fs - A reference to the SPIFFS volume object instance
* max_free_pages - The maximum pages to free
*
* Returned Value:
* Zero (OK) is returned on success; A negated errno value is returned on
* any failure. -ENODATA is returned if no blocks were deleted.
*
****************************************************************************/
int spiffs_gc_quick(FAR struct spiffs_s *fs, uint16_t max_free_pages)
{
FAR int16_t *objlu_buf = (FAR int16_t *)fs->lu_work;
uint32_t cur_block_addr = 0;
uint32_t blocks = SPIFFS_GEO_BLOCK_COUNT(fs);
int16_t cur_block = 0;
int entries_per_page = (SPIFFS_GEO_PAGE_SIZE(fs) / sizeof(int16_t));
int cur_entry = 0;
int ret = OK;
spiffs_gcinfo("max_free_pages=%u\n", max_free_pages);
#ifdef CONFIG_SPIFFS_GCDBG
fs->stats_gc_runs++;
#endif
/* Find fully deleted blocks */
while (ret >= 0 && blocks--)
{
uint16_t deleted_pages_in_block = 0;
uint16_t free_pages_in_block = 0;
int obj_lookup_page = 0;
/* Check each object lookup page */
while (ret >= 0 &&
obj_lookup_page < (int)SPIFFS_OBJ_LOOKUP_PAGES(fs))
{
int entry_offset = obj_lookup_page * entries_per_page;
ret = spiffs_cache_read(fs, SPIFFS_OP_T_OBJ_LU | SPIFFS_OP_C_READ,
0, cur_block_addr +
SPIFFS_PAGE_TO_PADDR(fs, obj_lookup_page),
SPIFFS_GEO_PAGE_SIZE(fs), fs->lu_work);
/* Check each entry */
while (ret >= 0 &&
cur_entry - entry_offset < entries_per_page &&
cur_entry <
(int)(SPIFFS_GEO_PAGES_PER_BLOCK(fs) -
SPIFFS_OBJ_LOOKUP_PAGES(fs)))
{
int16_t id = objlu_buf[cur_entry - entry_offset];
if (id == SPIFFS_OBJID_DELETED)
{
deleted_pages_in_block++;
}
else if (id == SPIFFS_OBJID_FREE)
{
/* Kill scan, go for next block */
free_pages_in_block++;
if (free_pages_in_block > max_free_pages)
{
obj_lookup_page = SPIFFS_OBJ_LOOKUP_PAGES(fs);
ret = 1; /* Kill object lu loop */
break;
}
}
else
{
/* Kill scan, go for next block */
obj_lookup_page = SPIFFS_OBJ_LOOKUP_PAGES(fs);
ret = 1; /* Kill object lu loop */
break;
}
cur_entry++;
}
obj_lookup_page++;
}
if (ret == 1)
{
ret = OK;
}
if (ret >= 0 &&
deleted_pages_in_block + free_pages_in_block ==
SPIFFS_GEO_PAGES_PER_BLOCK(fs) - SPIFFS_OBJ_LOOKUP_PAGES(fs) &&
free_pages_in_block <= max_free_pages)
{
/* Found a fully deleted block */
fs->deleted_pages -= deleted_pages_in_block;
ret = spiffs_gc_erase_block(fs, cur_block);
return ret;
}
cur_entry = 0;
cur_block++;
cur_block_addr += SPIFFS_GEO_BLOCK_SIZE(fs);
}
if (ret >= 0)
{
/* No deleted blocks */
ret = -ENODATA;
}
return ret;
}
/****************************************************************************
* Name: spiffs_gc_check
*
* Description:
* Checks if garbage collecting is necessary. If so a candidate block is
* found, cleansed and erased
*
* This function will try to make room for given amount of bytes in the
* filesystem by moving pages and erasing blocks.
*
* If it is physically impossible, err_no will be set to -ENOSPC. If
* there already is this amount (or more) of free space, SPIFFS_gc will
* silently return. It is recommended to call statfs() before invoking
* this method in order to determine what amount of bytes to give.
*
* NB: the garbage collector is automatically called when spiffs needs free
* pages. The reason for this function is to give possibility to do
* background tidying when user knows the system is idle.
*
* Input Parameters:
* fs the file system struct
* len amount of data that should be freed
*
* Returned Value:
* Zero (OK) is returned on success; A negated errno value is returned on
* any failure. -ENODATA is returned if no blocks were deleted.
*
****************************************************************************/
int spiffs_gc_check(FAR struct spiffs_s *fs, off_t len)
{
int32_t free_pages;
uint32_t needed_pages;
int tries = 0;
int ret;
/* Get the number of free pages */
free_pages = (SPIFFS_GEO_PAGES_PER_BLOCK(fs) -
SPIFFS_OBJ_LOOKUP_PAGES(fs)) *
(SPIFFS_GEO_BLOCK_COUNT(fs) - 2) -
fs->alloc_pages - fs->deleted_pages;
spiffs_gcinfo("len=%ld free_blocks=%lu free_pages=%ld\n",
(long)len, (unsigned long)fs->free_blocks,
(long)free_pages);
if (fs->free_blocks > 3 &&
(int32_t)len < free_pages * (int32_t)SPIFFS_DATA_PAGE_SIZE(fs))
{
spiffs_gcinfo("Sufficient free space is available Do nothing.\n");
return OK;
}
/* Get the number of pages needed */
needed_pages = (len + SPIFFS_DATA_PAGE_SIZE(fs) - 1) /
SPIFFS_DATA_PAGE_SIZE(fs);
#if 0
if (fs->free_blocks <= 2 && (int32_t)needed_pages > free_pages)
{
spiffs_gcinfo("Full freeblk=%d needed=%d free=%d dele=%d\n",
fs->free_blocks, needed_pages, free_pages,
fs->deleted_pages);
return -ENOSPC;
}
#endif
if ((int32_t)needed_pages > (int32_t)(free_pages + fs->deleted_pages))
{
spiffs_gcinfo("Full freeblk=%" PRIu32 " needed=%" PRIu32 " "
"free=%" PRIu32 " dele=%" PRIu32 "\n",
fs->free_blocks, needed_pages, free_pages,
fs->deleted_pages);
return -ENOSPC;
}
do
{
FAR int16_t *cands;
int count;
int16_t cand;
int32_t prev_free_pages = free_pages;
spiffs_gcinfo("#%d: run gc free_blocks=%" PRIu32 " pfree="
"%" PRIu32 " pallo=%" PRIu32 " pdele=%" PRIu32 ""
" [%" PRIu32 "] len=%" PRIuOFF " of %" PRIu32 "\n",
tries, fs->free_blocks, free_pages,
fs->alloc_pages, fs->deleted_pages,
(free_pages + fs->alloc_pages + fs->deleted_pages),
len, (uint32_t)(free_pages * SPIFFS_DATA_PAGE_SIZE(fs)));
/* If the fs is crammed, ignore block age when selecting candidate -
* kind of a bad state
*/
ret = spiffs_gc_find_candidate(fs, &cands, &count, free_pages <= 0);
if (ret < 0)
{
ferr("ERROR: spiffs_gc_find_candidate() failed: %d\n", ret);
return ret;
}
if (count == 0)
{
spiffs_gcinfo("No candidates, return\n");
return (int32_t) needed_pages < free_pages ? OK : -ENOSPC;
}
#ifdef CONFIG_SPIFFS_GCDBG
fs->stats_gc_runs++;
#endif
cand = cands[0];
ret = spiffs_gc_clean(fs, cand);
spiffs_gcinfo("Cleaning block %d, result=%d\n", cand, ret);
if (ret < 0)
{
ferr("ERROR: spiffs_gc_clean() failed: %d\n", ret);
return ret;
}
ret = spiffs_gc_epage_stats(fs, cand);
if (ret < 0)
{
ferr("ERROR: spiffs_gc_epage_stats() failed: %d\n", ret);
return ret;
}
ret = spiffs_gc_erase_block(fs, cand);
if (ret < 0)
{
ferr("ERROR: spiffs_gc_erase_block() failed: %d\n", ret);
return ret;
}
free_pages = (SPIFFS_GEO_PAGES_PER_BLOCK(fs) -
SPIFFS_OBJ_LOOKUP_PAGES(fs)) *
(SPIFFS_GEO_BLOCK_COUNT(fs) - 2) -
fs->alloc_pages - fs->deleted_pages;
if (prev_free_pages <= 0 && prev_free_pages == free_pages)
{
/* Abort early to reduce wear, at least tried once */
spiffs_gcinfo("Early abort, no result on gc when fs crammed\n");
break;
}
}
while (++tries < CONFIG_SPIFFS_GC_MAXRUNS &&
(fs->free_blocks <= 2 ||
(int32_t) len > free_pages * (int32_t) SPIFFS_DATA_PAGE_SIZE(fs)));
/* Re-calculate the number of free pages */
free_pages = (SPIFFS_GEO_PAGES_PER_BLOCK(fs) -
SPIFFS_OBJ_LOOKUP_PAGES(fs)) *
(SPIFFS_GEO_BLOCK_COUNT(fs) - 2) -
fs->alloc_pages - fs->deleted_pages;
if ((int32_t) len > free_pages * (int32_t)SPIFFS_DATA_PAGE_SIZE(fs))
{
ret = -ENOSPC;
}
spiffs_gcinfo("Finished, %" PRIu32 " dirty, blocks, %" PRIu32 " "
"free, %" PRIu32 " pages free, %d tries, ret=%d\n",
fs->alloc_pages + fs->deleted_pages,
fs->free_blocks, free_pages, tries, ret);
return ret;
}