📄 fatfs_ncache.c
字号:
// All done
return true;
}
pnode = lnode;
lnode = lnode->hash_next;
}
// Node not in list
return false;
}
#ifdef USE_XCHECKS
static void
node_hash_check(fatfs_hash_table_t *tbl)
{
int i, n;
n = 0;
for (i = 0; i < tbl->size; i++)
{
fatfs_node_t *lnode, *pnode;
pnode = NULL;
lnode = tbl->nodes[i];
while (lnode != NULL)
{
if (pnode != NULL)
{
int c = strcasecmp(pnode->dentry.filename, lnode->dentry.filename);
CYG_ASSERT(c <= 0, "hash table not sorted");
CYG_ASSERT(pnode->dentry.parent_cluster != lnode->dentry.parent_cluster ||
0 != c, "duplicated node in hash table");
}
n++;
pnode = lnode;
lnode = lnode->hash_next;
}
}
CYG_ASSERTC(tbl->n == n);
}
static void
node_hash_not_found_check(fatfs_disk_t *disk,
const char *name,
unsigned int namelen,
unsigned int parent_cluster)
{
fatfs_node_t* node;
node = node_list_get_head(&disk->live_nlist);
while (NULL != node)
{
if (node->dentry.parent_cluster == parent_cluster &&
namelen == strlen(node->dentry.filename) &&
0 == strncasecmp(name, node->dentry.filename, namelen))
CYG_ASSERT(false, "node not found in hash, "
"but exists in live list");
node = node_list_get_next(node);
}
node = node_list_get_head(&disk->dead_nlist);
while (NULL != node)
{
if (node->dentry.parent_cluster == parent_cluster &&
namelen == strlen(node->dentry.filename) &&
0 == strncasecmp(name, node->dentry.filename, namelen))
CYG_ASSERT(false, "node not found in hash, "
"but exists in dead list");
node = node_list_get_next(node);
}
}
static void
node_hash_found_check(fatfs_disk_t *disk,
const char *name,
unsigned int namelen,
unsigned int parent_cluster,
fatfs_node_t* node)
{
fatfs_node_t *n;
n = node_list_get_head(&disk->live_nlist);
while (NULL != n)
{
if (n == node)
{
if (node->dentry.parent_cluster != parent_cluster ||
namelen != strlen(node->dentry.filename) ||
0 != strncasecmp(name, node->dentry.filename, namelen))
CYG_ASSERT(false, "node_hash_find returned wrong node");
return;
}
n = node_list_get_next(n);
}
n = node_list_get_head(&disk->dead_nlist);
while (NULL != n)
{
if (n == node)
{
if (node->dentry.parent_cluster != parent_cluster ||
namelen != strlen(node->dentry.filename) ||
0 != strncasecmp(name, node->dentry.filename, namelen))
CYG_ASSERT(false, "node_hash_find returned wrong node");
return;
}
n = node_list_get_next(n);
}
CYG_ASSERT(false, "node not found in dead or live lists, "
"but exists in hash.");
}
#endif // USE_XCHECKS
//--------------------------------------------------------------------------
#ifdef USE_XCHECKS
# define SANITY_CHECK() \
CYG_MACRO_START \
node_lists_check(disk); \
node_hash_check(&disk->node_hash); \
CYG_ASSERTC((disk->live_nlist.size + disk->dead_nlist.size) == \
disk->node_hash.n); \
CYG_MACRO_END
# define NODE_FIND_CHECK() \
CYG_MACRO_START \
if (NULL == node) \
node_hash_not_found_check(disk, name, namelen, parent_cluster); \
else \
node_hash_found_check(disk, name, namelen, parent_cluster, node); \
CYG_MACRO_END
#else // USE_XCHECKS
# define SANITY_CHECK() CYG_EMPTY_STATEMENT
# define NODE_FIND_CHECK() CYG_EMPTY_STATEMENT
#endif // not USE_XCHECKS
//==========================================================================
// Node pool allocation functions
static void
node_pool_init(fatfs_disk_t *disk)
{
int i;
for (i = 0; i < FATFS_NODE_POOL_SIZE; i++)
disk->node_pool[i] = &disk->node_pool_base[i];
disk->node_pool_free_cnt = i;
}
static fatfs_node_t *
node_pool_alloc(fatfs_disk_t *disk)
{
fatfs_node_t *node = NULL;
if (disk->node_pool_free_cnt > 0)
node = disk->node_pool[--disk->node_pool_free_cnt];
return node;
}
static void
node_pool_free(fatfs_disk_t *disk, fatfs_node_t *node)
{
disk->node_pool[disk->node_pool_free_cnt++] = node;
}
//==========================================================================
//==========================================================================
// Exported functions
//--------------------------------------------------------------------------
// fatfs_node_cache_init()
// Initializes node cache of given disk.
void
fatfs_node_cache_init(fatfs_disk_t *disk)
{
CYG_CHECK_DATA_PTRC(disk);
node_list_init(&disk->live_nlist);
node_list_init(&disk->dead_nlist);
node_hash_init(&disk->node_hash);
node_pool_init(disk);
SANITY_CHECK();
}
//--------------------------------------------------------------------------
// fatfs_node_cache_flush()
// Frees all node cache of given disk.
void
fatfs_node_cache_flush(fatfs_disk_t *disk)
{
fatfs_node_t *node, *next_node;
node = node_list_get_head(&disk->live_nlist);
while (NULL != node)
{
next_node = node_list_get_next(node);
node_list_remove(&disk->live_nlist, node);
if (!node_hash_remove(&disk->node_hash, node))
CYG_ASSERT(false, "Node not in hash");
node_pool_free(disk, node);
node = next_node;
}
node = node_list_get_head(&disk->dead_nlist);
while (NULL != node)
{
next_node = node_list_get_next(node);
node_list_remove(&disk->dead_nlist, node);
if (!node_hash_remove(&disk->node_hash, node))
CYG_ASSERT(false, "Node not in hash");
node_pool_free(disk, node);
node = next_node;
}
SANITY_CHECK();
}
//--------------------------------------------------------------------------
// fatfs_node_alloc()
// Allocates a new node.
fatfs_node_t*
fatfs_node_alloc(fatfs_disk_t *disk, fatfs_dir_entry_t *dentry)
{
fatfs_node_t *node;
CYG_CHECK_DATA_PTRC(disk);
CYG_CHECK_DATA_PTRC(dentry);
node = node_pool_alloc(disk);
if (NULL == node)
{
CYG_TRACE2(TNC, "getting node from dead list (size=%d keep=%d)",
node_list_get_size(&disk->dead_nlist), DLIST_KEEP_NUM);
if (node_list_get_size(&disk->dead_nlist) <= DLIST_KEEP_NUM)
return NULL;
node = node_list_tail_get(&disk->dead_nlist);
if (NULL == node)
return NULL;
CYG_TRACE1(TNC, "recycling node='%s'", node->dentry.filename);
node_list_remove(&disk->dead_nlist, node);
if (!node_hash_remove(&disk->node_hash, node))
CYG_ASSERT(false, "node not in hash");
}
// Init new node
node->dentry = *dentry;
node->refcnt = 0;
node_list_head_add(&disk->dead_nlist, node);
if (!node_hash_add(&disk->node_hash, node))
CYG_ASSERT(false, "node already in hash");
SANITY_CHECK();
return node;
}
//--------------------------------------------------------------------------
// fatfs_node_touch()
// Moves given node to top of its list.
// (When reusing dead node it is always taken from the bottom of list)
void
fatfs_node_touch(fatfs_disk_t *disk, fatfs_node_t *node)
{
CYG_CHECK_DATA_PTRC(disk);
CYG_CHECK_DATA_PTRC(node);
CYG_TRACE2(TNC, "node='%s' refcnt=%d", node->dentry.filename, node->refcnt);
if (node->refcnt == 0)
{
node_list_remove(&disk->dead_nlist, node);
node_list_head_add(&disk->dead_nlist, node);
}
else
{
node_list_remove(&disk->live_nlist, node);
node_list_head_add(&disk->live_nlist, node);
}
SANITY_CHECK();
}
//--------------------------------------------------------------------------
// fatfs_node_ref()
// Increases the given node reference count.
// If the reference goes from 0 to 1, than the node
// is moved from dead list to live list.
void
fatfs_node_ref(fatfs_disk_t *disk, fatfs_node_t *node)
{
CYG_CHECK_DATA_PTRC(disk);
CYG_CHECK_DATA_PTRC(node);
CYG_TRACE2(TNC, "node='%s' refcnt=%d", node->dentry.filename, node->refcnt);
node->refcnt++;
if (1 == node->refcnt)
{
// First reference - move node from dead to live list
CYG_TRACE1(TNC, "node='%s' to live list", node->dentry.filename);
node_list_remove(&disk->dead_nlist, node);
node_list_head_add(&disk->live_nlist, node);
}
SANITY_CHECK();
}
//--------------------------------------------------------------------------
// fatfs_node_unref()
// Decreases the given node reference count.
// If the reference goes from 1 to 0, than the node
// is moved from live list to top of dead list.
// (When reusing dead node it is always taken from the bottom of list)
void
fatfs_node_unref(fatfs_disk_t *disk, fatfs_node_t *node)
{
CYG_CHECK_DATA_PTRC(disk);
CYG_CHECK_DATA_PTRC(node);
CYG_TRACE2(TNC, "node='%s' refcnt=%d", node->dentry.filename, node->refcnt);
CYG_ASSERT(node->refcnt > 0, "node->refcnt <= 0");
node->refcnt--;
if (0 == node->refcnt)
{
// No more references - move node from live to dead list
CYG_TRACE1(TNC, "node='%s' to dead list", node->dentry.filename);
node_list_remove(&disk->live_nlist, node);
node_list_head_add(&disk->dead_nlist, node);
}
SANITY_CHECK();
}
//--------------------------------------------------------------------------
// fatfs_node_free()
// Frees node memory and removes it from its list and hash.
// This function must not be called on a referenced node.
void
fatfs_node_free(fatfs_disk_t *disk, fatfs_node_t *node)
{
CYG_CHECK_DATA_PTRC(disk);
CYG_CHECK_DATA_PTRC(node);
CYG_TRACE2(TNC, "node='%s' refcnt=%d", node->dentry.filename, node->refcnt);
CYG_ASSERTC(node->refcnt == 0);
CYG_ASSERTC(node != disk->root);
// Remove from dead list, from hash and free ptr
node_list_remove(&disk->dead_nlist, node);
if (!node_hash_remove(&disk->node_hash, node))
CYG_ASSERT(false, "node not in hash");
node_pool_free(disk, node);
SANITY_CHECK();
}
//--------------------------------------------------------------------------
// fatfs_node_rehash()
// Recalculates given node hash key.
void
fatfs_node_rehash(fatfs_disk_t *disk, fatfs_node_t *node)
{
CYG_CHECK_DATA_PTRC(disk);
CYG_CHECK_DATA_PTRC(node);
if (!node_hash_remove_keyless(&disk->node_hash, node))
CYG_ASSERT(false, "node not in hash");
if (!node_hash_add(&disk->node_hash, node))
CYG_ASSERT(false, "node already in hash");
SANITY_CHECK();
}
//--------------------------------------------------------------------------
// fatfs_node_find()
// Finds node in hash by its name and parent cluster.
// If no node found NULL is returned.
fatfs_node_t*
fatfs_node_find(fatfs_disk_t *disk,
const char *name,
unsigned int namelen,
unsigned int parent_cluster)
{
fatfs_node_t *node;
CYG_CHECK_DATA_PTRC(disk);
CYG_CHECK_DATA_PTRC(name);
node = node_hash_find(&disk->node_hash, name, namelen, parent_cluster);
NODE_FIND_CHECK();
CYG_TRACE3(TNC, "node name=%s pcluster=%d %s found in cache\n",
name, parent_cluster, ((node != NULL) ? "" : "not"));
return node;
}
//--------------------------------------------------------------------------
// fatfs_get_live_node_count()
// Gets the number of live nodes.
int
fatfs_get_live_node_count(fatfs_disk_t *disk)
{
return node_list_get_size(&disk->live_nlist);
}
//--------------------------------------------------------------------------
// fatfs_get_dead_node_count()
// Gets the number of dead nodes.
int
fatfs_get_dead_node_count(fatfs_disk_t *disk)
{
return node_list_get_size(&disk->dead_nlist);
}
// -------------------------------------------------------------------------
// EOF fatfs_ncache.c
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -