rt-thread/components/dfs/dfs_v2/filesystems/cromfs/dfs_cromfs.c

1383 lines
34 KiB
C

/*
* Copyright (c) 2006-2020, RT-Thread Development Team
*
* SPDX-License-Identifier: Apache-2.0
*
* Change Logs:
* Date Author Notes
* 2020/08/21 ShaoJinchun first version
*/
#include <rtthread.h>
#include <dfs.h>
#include <dfs_fs.h>
#include <dfs_file.h>
#include <dfs_dentry.h>
#include <dfs_mnt.h>
#include "dfs_cromfs.h"
#include <stdint.h>
#include "zlib.h"
#ifdef RT_USING_PAGECACHE
#include "dfs_pcache.h"
#endif
/**********************************/
#define CROMFS_PATITION_HEAD_SIZE 256
#define CROMFS_DIRENT_CACHE_SIZE 8
#define CROMFS_MAGIC "CROMFSMG"
#define CROMFS_CT_ASSERT(name, x) \
struct assert_##name {char ary[2 * (x) - 1];}
#define CROMFS_POS_ROOT (0x0UL)
#define CROMFS_POS_ERROR (0x1UL)
typedef struct
{
uint8_t magic[8]; /* CROMFS_MAGIC */
uint32_t version;
uint32_t partition_attr; /* expand, now reserved 0 */
uint32_t partition_size; /* with partition head */
uint32_t root_dir_pos; /* root dir pos */
uint32_t root_dir_size;
} partition_head_data;
typedef struct
{
partition_head_data head;
uint8_t padding[CROMFS_PATITION_HEAD_SIZE - sizeof(partition_head_data)];
} partition_head;
enum
{
CROMFS_DIRENT_ATTR_FILE = 0x0UL,
CROMFS_DIRENT_ATTR_DIR = 0x1UL,
CROMFS_DIRENT_ATTR_SYMLINK = 0x2UL,
};
typedef struct
{
uint16_t attr; /* dir or file add other */
uint16_t name_size; /* name real size */
uint32_t file_size; /* file data size */
uint32_t file_origin_size; /* file size before compress */
uint32_t parition_pos; /* offset of data */
uint8_t name[0]; /* name data */
} cromfs_dirent;
#define CROMFS_ALIGN_SIZE_BIT 4
#define CROMFS_ALIGN_SIZE (1UL << CROMFS_ALIGN_SIZE_BIT) /* must be same as sizeof cromfs_dirent */
#define CROMFS_ALIGN_SIZE_MASK (CROMFS_ALIGN_SIZE - 1)
CROMFS_CT_ASSERT(align_size, CROMFS_ALIGN_SIZE == sizeof(cromfs_dirent));
typedef union
{
cromfs_dirent dirent;
uint8_t name[CROMFS_ALIGN_SIZE];
} cromfs_dirent_item;
/**********************************/
typedef struct
{
rt_list_t list;
uint32_t partition_pos;
uint32_t size;
uint8_t *buff;
} cromfs_dirent_cache;
typedef struct st_cromfs_info
{
rt_device_t device;
uint32_t partition_size;
uint32_t bytes_per_sector;
uint32_t (*read_bytes)(struct st_cromfs_info *ci, uint32_t pos, void *buf, uint32_t size);
partition_head_data part_info;
struct rt_mutex lock;
struct cromfs_avl_struct *cromfs_avl_root;
rt_list_t cromfs_dirent_cache_head;
int cromfs_dirent_cache_nr;
const void *data;
} cromfs_info;
typedef struct
{
uint32_t ref;
uint32_t partition_pos;
cromfs_info *ci;
uint32_t size;
uint8_t *buff;
uint32_t partition_size;
int data_valid;
} file_info;
/**********************************/
#define avl_key_t uint32_t
#define AVL_EMPTY (struct cromfs_avl_struct *)0
#define avl_maxheight 32
#define heightof(tree) ((tree) == AVL_EMPTY ? 0 : (tree)->avl_height)
struct cromfs_avl_struct
{
struct cromfs_avl_struct *avl_left;
struct cromfs_avl_struct *avl_right;
int avl_height;
avl_key_t avl_key;
file_info *fi;
};
static void cromfs_avl_remove(struct cromfs_avl_struct *node_to_delete, struct cromfs_avl_struct **ptree);
static void cromfs_avl_insert(struct cromfs_avl_struct *new_node, struct cromfs_avl_struct **ptree);
static struct cromfs_avl_struct* cromfs_avl_find(avl_key_t key, struct cromfs_avl_struct *ptree);
static void cromfs_avl_rebalance(struct cromfs_avl_struct ***nodeplaces_ptr, int count)
{
for (;count > 0; count--)
{
struct cromfs_avl_struct **nodeplace = *--nodeplaces_ptr;
struct cromfs_avl_struct *node = *nodeplace;
struct cromfs_avl_struct *nodeleft = node->avl_left;
struct cromfs_avl_struct *noderight = node->avl_right;
int heightleft = heightof(nodeleft);
int heightright = heightof(noderight);
if (heightright + 1 < heightleft)
{
struct cromfs_avl_struct * nodeleftleft = nodeleft->avl_left;
struct cromfs_avl_struct * nodeleftright = nodeleft->avl_right;
int heightleftright = heightof(nodeleftright);
if (heightof(nodeleftleft) >= heightleftright)
{
node->avl_left = nodeleftright;
nodeleft->avl_right = node;
nodeleft->avl_height = 1 + (node->avl_height = 1 + heightleftright);
*nodeplace = nodeleft;
}
else
{
nodeleft->avl_right = nodeleftright->avl_left;
node->avl_left = nodeleftright->avl_right;
nodeleftright->avl_left = nodeleft;
nodeleftright->avl_right = node;
nodeleft->avl_height = node->avl_height = heightleftright;
nodeleftright->avl_height = heightleft;
*nodeplace = nodeleftright;
}
}
else if (heightleft + 1 < heightright)
{
struct cromfs_avl_struct *noderightright = noderight->avl_right;
struct cromfs_avl_struct *noderightleft = noderight->avl_left;
int heightrightleft = heightof(noderightleft);
if (heightof(noderightright) >= heightrightleft)
{
node->avl_right = noderightleft;
noderight->avl_left = node;
noderight->avl_height = 1 + (node->avl_height = 1 + heightrightleft);
*nodeplace = noderight;
}
else
{
noderight->avl_left = noderightleft->avl_right;
node->avl_right = noderightleft->avl_left;
noderightleft->avl_right = noderight;
noderightleft->avl_left = node;
noderight->avl_height = node->avl_height = heightrightleft;
noderightleft->avl_height = heightright;
*nodeplace = noderightleft;
}
}
else {
int height = (heightleft<heightright ? heightright : heightleft) + 1;
if (height == node->avl_height)
{
break;
}
node->avl_height = height;
}
}
}
static void cromfs_avl_remove(struct cromfs_avl_struct *node_to_delete, struct cromfs_avl_struct **ptree)
{
avl_key_t key = node_to_delete->avl_key;
struct cromfs_avl_struct **nodeplace = ptree;
struct cromfs_avl_struct **stack[avl_maxheight];
uint32_t stack_count = 0;
struct cromfs_avl_struct ***stack_ptr = &stack[0]; /* = &stack[stackcount] */
struct cromfs_avl_struct **nodeplace_to_delete;
for (;;)
{
struct cromfs_avl_struct *node = *nodeplace;
if (node == AVL_EMPTY)
{
return;
}
*stack_ptr++ = nodeplace;
stack_count++;
if (key == node->avl_key)
{
break;
}
if (key < node->avl_key)
{
nodeplace = &node->avl_left;
}
else
{
nodeplace = &node->avl_right;
}
}
nodeplace_to_delete = nodeplace;
if (node_to_delete->avl_left == AVL_EMPTY)
{
*nodeplace_to_delete = node_to_delete->avl_right;
stack_ptr--;
stack_count--;
}
else
{
struct cromfs_avl_struct *** stack_ptr_to_delete = stack_ptr;
struct cromfs_avl_struct ** nodeplace = &node_to_delete->avl_left;
struct cromfs_avl_struct * node;
for (;;)
{
node = *nodeplace;
if (node->avl_right == AVL_EMPTY)
{
break;
}
*stack_ptr++ = nodeplace;
stack_count++;
nodeplace = &node->avl_right;
}
*nodeplace = node->avl_left;
node->avl_left = node_to_delete->avl_left;
node->avl_right = node_to_delete->avl_right;
node->avl_height = node_to_delete->avl_height;
*nodeplace_to_delete = node;
*stack_ptr_to_delete = &node->avl_left;
}
cromfs_avl_rebalance(stack_ptr,stack_count);
}
static void cromfs_avl_insert(struct cromfs_avl_struct *new_node, struct cromfs_avl_struct **ptree)
{
avl_key_t key = new_node->avl_key;
struct cromfs_avl_struct **nodeplace = ptree;
struct cromfs_avl_struct **stack[avl_maxheight];
int stack_count = 0;
struct cromfs_avl_struct ***stack_ptr = &stack[0]; /* = &stack[stackcount] */
for (;;)
{
struct cromfs_avl_struct * node = *nodeplace;
if (node == AVL_EMPTY)
{
break;
}
*stack_ptr++ = nodeplace;
stack_count++;
if (key < node->avl_key)
{
nodeplace = &node->avl_left;
}
else
{
nodeplace = &node->avl_right;
}
}
new_node->avl_left = AVL_EMPTY;
new_node->avl_right = AVL_EMPTY;
new_node->avl_height = 1;
*nodeplace = new_node;
cromfs_avl_rebalance(stack_ptr,stack_count);
}
static struct cromfs_avl_struct* cromfs_avl_find(avl_key_t key, struct cromfs_avl_struct* ptree)
{
for (;;)
{
if (ptree == AVL_EMPTY)
{
return (struct cromfs_avl_struct *)0;
}
if (key == ptree->avl_key)
{
break;
}
if (key < ptree->avl_key)
{
ptree = ptree->avl_left;
}
else
{
ptree = ptree->avl_right;
}
}
return ptree;
}
/**********************************/
static uint32_t cromfs_read_bytes(cromfs_info *ci, uint32_t pos, void *buf, uint32_t size)
{
if (pos >= ci->partition_size || pos + size > ci->partition_size)
{
return 0;
}
return ci->read_bytes(ci, pos, buf, size);
}
static uint32_t cromfs_noblk_read_bytes(cromfs_info *ci, uint32_t pos, void *buf, uint32_t size)
{
uint32_t ret = 0;
ret = rt_device_read(ci->device, pos, buf, size);
if (ret != size)
{
return 0;
}
else
{
return ret;
}
}
static uint32_t cromfs_data_read_bytes(cromfs_info *ci, uint32_t pos, void *buf, uint32_t size)
{
uint32_t ret = 0;
uint8_t *data = (uint8_t *)ci->data;
if (data)
{
memcpy(buf, data + pos, size);
ret = size;
}
return ret;
}
static uint32_t cromfs_blk_read_bytes(cromfs_info *ci, uint32_t pos, void *buf, uint32_t size)
{
uint32_t ret = 0;
uint32_t size_bak = size;
uint32_t start_blk = 0;
uint32_t end_blk = 0;
uint32_t off_s = 0;
uint32_t sector_nr = 0;
uint8_t *block_buff = NULL;
uint32_t ret_len = 0;
if (!size || !buf)
{
return 0;
}
block_buff = (uint8_t *)malloc(2 * ci->bytes_per_sector);
if (!block_buff)
{
return 0;
}
start_blk = pos / ci->bytes_per_sector;
off_s = pos % ci->bytes_per_sector;
end_blk = (pos + size - 1) / ci->bytes_per_sector;
sector_nr = end_blk - start_blk;
if (sector_nr < 2)
{
ret_len = rt_device_read(ci->device, start_blk, block_buff, sector_nr + 1);
if (ret_len != sector_nr + 1)
{
goto end;
}
memcpy(buf, block_buff + off_s, size);
}
else
{
ret_len = rt_device_read(ci->device, start_blk, block_buff, 1);
if (ret_len != 1)
{
goto end;
}
memcpy(buf, block_buff + off_s, ci->bytes_per_sector - off_s);
off_s = (ci->bytes_per_sector - off_s);
size -= off_s;
sector_nr--;
start_blk++;
if (sector_nr)
{
ret_len = rt_device_read(ci->device, start_blk, (char*)buf + off_s, sector_nr);
if (ret_len != sector_nr)
{
goto end;
}
start_blk += sector_nr;
off_s += (sector_nr * ci->bytes_per_sector);
size -= (sector_nr * ci->bytes_per_sector);
}
ret_len = rt_device_read(ci->device, start_blk, block_buff, 1);
if (ret_len != 1)
{
goto end;
}
memcpy((char*)buf + off_s, block_buff, size);
}
ret = size_bak;
end:
free(block_buff);
return ret;
}
/**********************************/
static uint8_t *cromfs_dirent_cache_get(cromfs_info *ci, uint32_t pos, uint32_t size)
{
rt_list_t *l = NULL;
cromfs_dirent_cache *dir = NULL;
uint32_t len = 0;
/* find */
for (l = ci->cromfs_dirent_cache_head.next; l != &ci->cromfs_dirent_cache_head; l = l->next)
{
dir = (cromfs_dirent_cache *)l;
if (dir->partition_pos == pos)
{
RT_ASSERT(dir->size == size);
rt_list_remove(l);
rt_list_insert_after(&ci->cromfs_dirent_cache_head, l);
return dir->buff;
}
}
/* not found */
if (ci->cromfs_dirent_cache_nr >= CROMFS_DIRENT_CACHE_SIZE)
{
l = ci->cromfs_dirent_cache_head.prev;
dir = (cromfs_dirent_cache *)l;
rt_list_remove(l);
free(dir->buff);
free(dir);
ci->cromfs_dirent_cache_nr--;
}
dir = (cromfs_dirent_cache *)malloc(sizeof *dir);
if (!dir)
{
return NULL;
}
dir->buff = (uint8_t *)malloc(size);
if (!dir->buff)
{
free(dir);
return NULL;
}
len = cromfs_read_bytes(ci, pos, dir->buff, size);
if (len != size)
{
free(dir->buff);
free(dir);
return NULL;
}
rt_list_insert_after(&ci->cromfs_dirent_cache_head, (rt_list_t *)dir);
ci->cromfs_dirent_cache_nr++;
dir->partition_pos = pos;
dir->size = size;
return dir->buff;
}
static void cromfs_dirent_cache_destroy(cromfs_info *ci)
{
rt_list_t *l = NULL;
cromfs_dirent_cache *dir = NULL;
while ((l = ci->cromfs_dirent_cache_head.next) != &ci->cromfs_dirent_cache_head)
{
rt_list_remove(l);
dir = (cromfs_dirent_cache *)l;
free(dir->buff);
free(dir);
ci->cromfs_dirent_cache_nr--;
}
}
/**********************************/
#ifdef RT_USING_PAGECACHE
static ssize_t dfs_cromfs_page_read(struct dfs_file *file, struct dfs_page *page);
static struct dfs_aspace_ops dfs_cromfs_aspace_ops =
{
.read = dfs_cromfs_page_read
};
#endif
static int dfs_cromfs_mount(struct dfs_mnt *mnt, unsigned long rwflag, const void *data)
{
struct rt_device_blk_geometry geometry;
uint32_t len = 0;
cromfs_info *ci = NULL;
ci = (cromfs_info *)malloc(sizeof *ci);
if (!ci)
{
return -ENOMEM;
}
memset(ci, 0, sizeof *ci);
ci->device = mnt->dev_id;
ci->partition_size = UINT32_MAX;
if (ci->device)
{
rt_err_t ret = rt_device_open(ci->device, RT_DEVICE_OFLAG_RDONLY);
if (ret != RT_EOK)
{
free(ci);
return ret;
}
if (ci->device->type == RT_Device_Class_Block)
{
rt_device_control(ci->device, RT_DEVICE_CTRL_BLK_GETGEOME, &geometry);
ci->bytes_per_sector = geometry.bytes_per_sector;
ci->read_bytes = cromfs_blk_read_bytes;
}
else
{
ci->read_bytes = cromfs_noblk_read_bytes;
}
}
else if (data)
{
ci->data = data;
ci->read_bytes = cromfs_data_read_bytes;
}
else
{
free(ci);
return -RT_EIO;
}
len = cromfs_read_bytes(ci, 0, &ci->part_info, sizeof ci->part_info);
if (len != sizeof ci->part_info ||
memcmp(ci->part_info.magic, CROMFS_MAGIC, sizeof ci->part_info.magic) != 0)
{
free(ci);
return -RT_ERROR;
}
ci->partition_size = ci->part_info.partition_size;
mnt->data = ci;
rt_mutex_init(&ci->lock, "crom", RT_IPC_FLAG_FIFO);
ci->cromfs_avl_root = NULL;
rt_list_init(&ci->cromfs_dirent_cache_head);
ci->cromfs_dirent_cache_nr = 0;
return RT_EOK;
}
static int dfs_cromfs_unmount(struct dfs_mnt *mnt)
{
rt_err_t result = RT_EOK;
cromfs_info *ci = NULL;
ci = (cromfs_info *)mnt->data;
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
return -RT_ERROR;
}
cromfs_dirent_cache_destroy(ci);
while (ci->cromfs_avl_root)
{
struct cromfs_avl_struct *node;
file_info *fi = NULL;
node = ci->cromfs_avl_root;
fi = node->fi;
cromfs_avl_remove(node, &ci->cromfs_avl_root);
free(node);
if (fi->buff)
{
free(fi->buff);
}
free(fi);
}
if (ci->device)
{
rt_device_close(ci->device);
}
rt_mutex_detach(&ci->lock);
free(ci);
return RT_EOK;
}
static uint32_t cromfs_lookup(cromfs_info *ci, const char *path, int* file_type, uint32_t *size, uint32_t *osize)
{
uint32_t cur_size = 0, cur_pos = 0, cur_osize = 0;
const char *subpath = NULL, *subpath_end = NULL;
void *di_mem = NULL;
int _file_type = 0;
if (path[0] == '\0')
{
return CROMFS_POS_ERROR;
}
cur_size = ci->part_info.root_dir_size;
cur_osize = 0;
cur_pos = ci->part_info.root_dir_pos;
_file_type = CROMFS_DIRENT_ATTR_DIR;
subpath_end = path;
while (1)
{
cromfs_dirent_item *di_iter = NULL;
int found = 0;
/* skip /// */
while (*subpath_end && *subpath_end == '/')
{
subpath_end++;
}
subpath = subpath_end;
while ((*subpath_end != '/') && *subpath_end)
{
subpath_end++;
}
if (*subpath == '\0')
{
break;
}
/* if not dir or empty dir, error */
if (_file_type != CROMFS_DIRENT_ATTR_DIR || !cur_size)
{
return CROMFS_POS_ERROR;
}
/* find subpath */
di_mem = cromfs_dirent_cache_get(ci, cur_pos, cur_size);
if (!di_mem)
{
return CROMFS_POS_ERROR;
}
found = 0;
di_iter = (cromfs_dirent_item *)di_mem;
while (1)
{
uint32_t name_len = subpath_end - subpath;
uint32_t name_block = 0;
if (di_iter->dirent.name_size == name_len)
{
if (memcmp(di_iter->dirent.name, subpath, name_len) == 0)
{
found = 1;
cur_size = di_iter->dirent.file_size;
cur_osize = di_iter->dirent.file_origin_size;
cur_pos = di_iter->dirent.parition_pos;
if (di_iter->dirent.attr == CROMFS_DIRENT_ATTR_FILE)
{
_file_type = CROMFS_DIRENT_ATTR_FILE;
}
else if (di_iter->dirent.attr == CROMFS_DIRENT_ATTR_DIR)
{
_file_type = CROMFS_DIRENT_ATTR_DIR;
}
else if (di_iter->dirent.attr == CROMFS_DIRENT_ATTR_SYMLINK)
{
_file_type = CROMFS_DIRENT_ATTR_SYMLINK;
}
else
{
RT_ASSERT(0);
}
break;
}
}
name_block = (di_iter->dirent.name_size + CROMFS_ALIGN_SIZE_MASK) >> CROMFS_ALIGN_SIZE_BIT;
di_iter += (1 + name_block);
if ((uint32_t)(intptr_t)di_iter - (uint32_t)(intptr_t)di_mem >= cur_size)
{
break;
}
}
if (!found)
{
return CROMFS_POS_ERROR;
}
}
*size = cur_size;
*osize = cur_osize;
*file_type = _file_type;
return cur_pos;
}
static uint32_t __dfs_cromfs_lookup(cromfs_info *ci, const char *path, int* file_type, uint32_t *size, uint32_t *osize)
{
rt_err_t result = RT_EOK;
uint32_t ret = 0;
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
return CROMFS_POS_ERROR;
}
ret = cromfs_lookup(ci, path, file_type, size, osize);
rt_mutex_release(&ci->lock);
return ret;
}
static int fill_file_data(file_info *fi)
{
int ret = -1;
cromfs_info *ci = NULL;
void *compressed_file_buff = NULL;
uint32_t size = 0, osize = 0;
if (!fi->data_valid)
{
RT_ASSERT(fi->buff != NULL);
ci = fi->ci;
osize = fi->size;
size = fi->partition_size;
compressed_file_buff = (void *)malloc(size);
if (!compressed_file_buff)
{
goto end;
}
if (cromfs_read_bytes(ci, fi->partition_pos, compressed_file_buff, size) != size)
{
goto end;
}
if (uncompress((uint8_t *)fi->buff, (uLongf *)&osize, (uint8_t *)compressed_file_buff, size) != Z_OK)
{
goto end;
}
fi->data_valid = 1;
}
ret = 0;
end:
if (compressed_file_buff)
{
free(compressed_file_buff);
}
return ret;
}
static ssize_t dfs_cromfs_read(struct dfs_file *file, void *buf, size_t count, off_t *pos)
{
rt_err_t result = RT_EOK;
file_info *fi = NULL;
cromfs_info *ci = NULL;
uint32_t length = 0;
ci = (cromfs_info *)file->dentry->mnt->data;
fi = (file_info *)file->vnode->data;
if (count < file->vnode->size - *pos)
{
length = count;
}
else
{
length = file->vnode->size - *pos;
}
if (length > 0)
{
RT_ASSERT(fi->size != 0);
if (fi->buff)
{
int fill_ret = 0;
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
return 0;
}
fill_ret = fill_file_data(fi);
rt_mutex_release(&ci->lock);
if (fill_ret < 0)
{
return 0;
}
memcpy(buf, fi->buff + *pos, length);
}
else
{
void *di_mem = NULL;
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
return 0;
}
di_mem = cromfs_dirent_cache_get(ci, fi->partition_pos, fi->size);
if (di_mem)
{
memcpy(buf, (char*)di_mem + *pos, length);
}
rt_mutex_release(&ci->lock);
if (!di_mem)
{
return 0;
}
}
/* update file current position */
*pos += length;
}
return length;
}
static file_info *get_file_info(cromfs_info *ci, uint32_t partition_pos, int inc_ref)
{
struct cromfs_avl_struct* node = cromfs_avl_find(partition_pos, ci->cromfs_avl_root);
if (node)
{
if (inc_ref)
{
node->fi->ref++;
}
return node->fi;
}
return NULL;
}
static file_info *inset_file_info(cromfs_info *ci, uint32_t partition_pos, int file_type, uint32_t size, uint32_t osize)
{
file_info *fi = NULL;
void *file_buff = NULL;
struct cromfs_avl_struct *node = NULL;
fi = (file_info *)malloc(sizeof *fi);
if (!fi)
{
goto err;
}
fi->partition_pos = partition_pos;
fi->ci = ci;
if (file_type == CROMFS_DIRENT_ATTR_DIR)
{
fi->size = size;
}
else
{
fi->size = osize;
fi->partition_size = size;
fi->data_valid = 0;
if (osize)
{
file_buff = (void *)malloc(osize);
if (!file_buff)
{
goto err;
}
}
}
fi->buff = file_buff;
fi->ref = 1;
node = (struct cromfs_avl_struct *)malloc(sizeof *node);
if (!node)
{
goto err;
}
node->avl_key = partition_pos;
node->fi = fi;
cromfs_avl_insert(node, &ci->cromfs_avl_root);
return fi;
err:
if (file_buff)
{
free(file_buff);
}
if (fi)
{
free(fi);
}
return NULL;
}
static void deref_file_info(cromfs_info *ci, uint32_t partition_pos)
{
struct cromfs_avl_struct* node = cromfs_avl_find(partition_pos, ci->cromfs_avl_root);
file_info *fi = NULL;
if (node)
{
node->fi->ref--;
if (node->fi->ref == 0)
{
fi = node->fi;
cromfs_avl_remove(node, &ci->cromfs_avl_root);
free(node);
if (fi->buff)
{
free(fi->buff);
}
free(fi);
}
}
}
static int dfs_cromfs_close(struct dfs_file *file)
{
file_info *fi = NULL;
cromfs_info *ci = NULL;
rt_err_t result = 0;
RT_ASSERT(file->vnode->ref_count > 0);
if (file->vnode->ref_count > 1)
{
return 0;
}
fi = (file_info *)file->vnode->data;
ci = (cromfs_info *)file->dentry->mnt->data;
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
return -RT_ERROR;
}
deref_file_info(ci, fi->partition_pos);
rt_mutex_release(&ci->lock);
file->vnode->data = NULL;
return RT_EOK;
}
static int dfs_cromfs_open(struct dfs_file *file)
{
int ret = 0;
file_info *fi = NULL;
cromfs_info *ci = NULL;
uint32_t file_pos = 0;
uint32_t size = 0, osize = 0;
int file_type = 0;
rt_err_t result = RT_EOK;
if (file->flags & (O_CREAT | O_WRONLY | O_APPEND | O_TRUNC | O_RDWR))
{
return -EINVAL;
}
RT_ASSERT(file->vnode->ref_count > 0);
if (file->vnode->ref_count > 1)
{
if (file->vnode->type == FT_DIRECTORY
&& !(file->flags & O_DIRECTORY))
{
return -ENOENT;
}
file->fpos = 0;
return 0;
}
ci = (cromfs_info *)file->dentry->mnt->data;
file_pos = __dfs_cromfs_lookup(ci, file->dentry->pathname, &file_type, &size, &osize);
if (file_pos == CROMFS_POS_ERROR)
{
ret = -ENOENT;
goto end;
}
/* entry is a directory file type */
if (file_type == CROMFS_DIRENT_ATTR_DIR)
{
if (!(file->flags & O_DIRECTORY))
{
ret = -ENOENT;
goto end;
}
file->vnode->type = FT_DIRECTORY;
}
else if (file_type == CROMFS_DIRENT_ATTR_SYMLINK)
{
file->vnode->type = FT_SYMLINK;
}
else
{
/* entry is a file, but open it as a directory */
if (file->flags & O_DIRECTORY)
{
ret = -ENOENT;
goto end;
}
file->vnode->type = FT_REGULAR;
}
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
ret = -EINTR;
goto end;
}
fi = get_file_info(ci, file_pos, 1);
if (!fi)
{
fi = inset_file_info(ci, file_pos, file_type, size, osize);
}
rt_mutex_release(&ci->lock);
if (!fi)
{
ret = -ENOENT;
goto end;
}
file->vnode->data = fi;
if (file_type)
{
file->vnode->size = size;
}
else
{
file->vnode->size = osize;
}
file->fpos = 0;
ret = RT_EOK;
end:
return ret;
}
static int dfs_cromfs_stat(struct dfs_dentry *dentry, struct stat *st)
{
uint32_t size = 0, osize = 0;
int file_type = 0;
cromfs_info *ci = NULL;
uint32_t file_pos = 0;
ci = (cromfs_info *)dentry->mnt->data;
file_pos = __dfs_cromfs_lookup(ci, dentry->pathname, &file_type, &size, &osize);
if (file_pos == CROMFS_POS_ERROR)
{
return -ENOENT;
}
st->st_dev = 0;
st->st_mode = S_IFREG | (0777);
if (file_type == CROMFS_DIRENT_ATTR_DIR)
{
st->st_mode &= ~S_IFREG;
st->st_mode |= S_IFDIR;
st->st_size = size;
}
else if(file_type == CROMFS_DIRENT_ATTR_SYMLINK)
{
st->st_mode &= ~S_IFREG;
st->st_mode |= S_IFLNK;
st->st_size = osize;
}
else
{
#ifdef RT_USING_PAGECACHE
st->st_size = (dentry->vnode && dentry->vnode->aspace) ? dentry->vnode->size : osize;
#else
st->st_size = osize;
#endif
}
st->st_mtime = 0;
return RT_EOK;
}
static int dfs_cromfs_getdents(struct dfs_file *file, struct dirent *dirp, uint32_t count)
{
uint32_t index = 0;
uint8_t *name = NULL;
struct dirent *d = NULL;
file_info *fi = NULL;
cromfs_info *ci = NULL;
cromfs_dirent_item *dirent = NULL, *sub_dirent = NULL;
void *di_mem = NULL;
rt_err_t result = RT_EOK;
fi = (file_info *)file->vnode->data;
ci = fi->ci;
RT_ASSERT(fi->buff == NULL);
if (!fi->size)
{
return -EINVAL;
}
dirent = (cromfs_dirent_item *)malloc(fi->size);
if (!dirent)
{
return -ENOMEM;
}
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
free(dirent);
return -EINTR;
}
di_mem = cromfs_dirent_cache_get(ci, fi->partition_pos, fi->size);
if (di_mem)
{
memcpy(dirent, di_mem, fi->size);
}
rt_mutex_release(&ci->lock);
if (!di_mem)
{
free(dirent);
return -ENOMEM;
}
/* make integer count */
count = (count / sizeof(struct dirent));
if (count == 0)
{
free(dirent);
return -EINVAL;
}
for (index = 0; index < count && file->fpos < file->vnode->size; index++)
{
uint32_t name_size = 0;
d = dirp + index;
sub_dirent = &dirent[file->fpos >> CROMFS_ALIGN_SIZE_BIT];
name = sub_dirent->dirent.name;
/* fill dirent */
if (sub_dirent->dirent.attr == CROMFS_DIRENT_ATTR_DIR)
{
d->d_type = DT_DIR;
}
else
{
d->d_type = DT_REG;
}
d->d_namlen = sub_dirent->dirent.name_size;
d->d_reclen = (rt_uint16_t)sizeof(struct dirent);
memcpy(d->d_name, (char *)name, sub_dirent->dirent.name_size);
d->d_name[sub_dirent->dirent.name_size] = '\0';
name_size = (sub_dirent->dirent.name_size + CROMFS_ALIGN_SIZE_MASK) & ~CROMFS_ALIGN_SIZE_MASK;
/* move to next position */
file->fpos += (name_size + sizeof *sub_dirent);
}
free(dirent);
return index * sizeof(struct dirent);
}
static struct dfs_vnode *dfs_cromfs_lookup (struct dfs_dentry *dentry)
{
struct dfs_vnode *vnode = RT_NULL;
cromfs_info *ci = NULL;
RT_ASSERT(dentry != RT_NULL);
RT_ASSERT(dentry->mnt != RT_NULL);
ci = (cromfs_info *)dentry->mnt->data;
if (ci)
{
uint32_t size = 0, osize = 0;
int file_type = 0;
uint32_t file_pos = __dfs_cromfs_lookup(ci, dentry->pathname, &file_type, &size, &osize);
if (file_pos != CROMFS_POS_ERROR)
{
vnode = dfs_vnode_create();
if (vnode)
{
vnode->nlink = 1;
if (file_type == CROMFS_DIRENT_ATTR_DIR)
{
vnode->mode = S_IFDIR | (0777);
vnode->type = FT_DIRECTORY;
vnode->size = size;
}
else if (file_type == CROMFS_DIRENT_ATTR_SYMLINK)
{
vnode->mode = S_IFLNK | (0777);
vnode->type = FT_SYMLINK;
vnode->size = osize;
}
else
{
vnode->mode = S_IFREG | (0777);
vnode->type = FT_REGULAR;
vnode->size = osize;
#ifdef RT_USING_PAGECACHE
vnode->aspace = dfs_aspace_create(dentry, vnode, &dfs_cromfs_aspace_ops);
#endif
}
vnode->mnt = dentry->mnt;
}
}
}
return vnode;
}
static int dfs_cromfs_free_vnode(struct dfs_vnode *vnode)
{
return 0;
}
static int cromfs_readlink(cromfs_info *ci, char *path, char *buf, int len)
{
int ret = 0;
file_info *fi = NULL;
uint32_t file_pos = 0;
int file_type = 0;
uint32_t size = 0, osize = 0;
rt_err_t result = RT_EOK;
file_pos = __dfs_cromfs_lookup(ci, path, &file_type, &size, &osize);
if (file_pos == CROMFS_POS_ERROR)
{
ret = -ENOENT;
goto end1;
}
result = rt_mutex_take(&ci->lock, RT_WAITING_FOREVER);
if (result != RT_EOK)
{
ret = -EINTR;
goto end;
}
fi = get_file_info(ci, file_pos, 1);
if (!fi)
{
fi = inset_file_info(ci, file_pos, file_type, size, osize);
}
rt_mutex_release(&ci->lock);
if (!fi)
{
ret = -ENOENT;
goto end;
}
if (len > 0)
{
RT_ASSERT(fi->size != 0);
RT_ASSERT(fi->buff);
int fill_ret = 0;
fill_ret = fill_file_data(fi);
if (fill_ret < 0)
{
ret = -ENOENT;
deref_file_info(ci, fi->partition_pos);
goto end;
}
len = len - 1;
osize = osize < len ? osize : len;
memcpy(buf, fi->buff, osize);
}
if (ret == 0)
{
buf[osize] = '\0';
ret = osize;
}
deref_file_info(ci, fi->partition_pos);
end:
rt_mutex_release(&ci->lock);
end1:
return ret;
}
#ifdef RT_USING_PAGECACHE
static ssize_t dfs_cromfs_page_read(struct dfs_file *file, struct dfs_page *page)
{
int ret = -EINVAL;
if (page->page)
{
off_t fpos = page->fpos;
ret = dfs_cromfs_read(file, page->page, page->size, &fpos);
}
return ret;
}
#endif
static int dfs_cromfs_readlink(struct dfs_dentry *dentry, char *buf, int len)
{
cromfs_info *ci = NULL;
if (dentry && buf)
{
ci = (cromfs_info *)dentry->mnt->data;
return cromfs_readlink(ci, dentry->pathname, buf, len);
}
return -EBADF;
}
static const struct dfs_file_ops _crom_fops =
{
.open = dfs_cromfs_open,
.close = dfs_cromfs_close,
.lseek = generic_dfs_lseek,
.read = dfs_cromfs_read,
.getdents = dfs_cromfs_getdents,
};
static const struct dfs_filesystem_ops _cromfs_ops =
{
.name = "crom",
.flags = 0,
.default_fops = &_crom_fops,
.mount = dfs_cromfs_mount,
.umount = dfs_cromfs_unmount,
.readlink = dfs_cromfs_readlink,
.stat = dfs_cromfs_stat,
.lookup = dfs_cromfs_lookup,
.free_vnode = dfs_cromfs_free_vnode
};
static struct dfs_filesystem_type _cromfs =
{
.fs_ops = &_cromfs_ops,
};
int dfs_cromfs_init(void)
{
/* register crom file system */
dfs_register(&_cromfs);
return 0;
}
INIT_COMPONENT_EXPORT(dfs_cromfs_init);