CloverBootloader/FileSystems/VBoxFsDxe/fsw_core.c

1140 lines
35 KiB
C

/* $Id: fsw_core.c 29125 2010-05-06 09:43:05Z vboxsync $ */
/** @file
* fsw_core.c - Core file system wrapper abstraction layer code.
*/
/*-
* This code is based on:
*
* Copyright (c) 2006 Christoph Pfisterer
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* * 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.
*
* * Neither the name of Christoph Pfisterer nor the names of the
* 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.
*/
// nms42 - caching support
#include "fsw_core.h"
//#include "Version.h"
//CONST CHAR8* CloverRevision = REVISION_STR;
#define DEBUG_CR 0
#if DEBUG_CR==2
#define DBG(...) AsciiPrint(__VA_ARGS__)
#elif DEBUG_CR==1
#define DBG(...) BootLog(__VA_ARGS__)
#else
#define DBG(...)
#endif
const char* fsw_errors[] = {
"SUCCESS", "OUT_OF_MEMORY", "IO_ERROR", "UNSUPPORTED", "NOT_FOUND",
"VOLUME_CORRUPTED", "UNKNOWN_ERROR"
};
// functions
static void fsw_blockcache_free(struct fsw_volume *vol);
#define MAX_CACHE_LEVEL (5)
/**
* Mount a volume with a given file system driver. This function is called by the
* host driver to make a volume accessible. The file system driver to use is specified
* by a pointer to its dispatch table. The file system driver will look at the
* data on the volume to determine if it can read the format. If the volume is found
* unsuitable, FSW_UNSUPPORTED is returned.
*
* If this function returns FSW_SUCCESS, *vol_out points at a valid volume data
* structure. The caller must release it later by calling fsw_unmount.
*
* If this function returns an error status, the caller only needs to clean up its
* own buffers that may have been allocated through the read_block interface.
*/
fsw_status_t fsw_mount(void *host_data,
struct fsw_host_table *host_table,
struct fsw_fstype_table *fstype_table,
struct fsw_volume **vol_out)
{
fsw_status_t status;
struct fsw_volume *vol;
// allocate memory for the structure
status = fsw_alloc_zero(fstype_table->volume_struct_size, (void **)&vol);
if (status)
return status;
// initialize fields
vol->phys_blocksize = 512;
vol->log_blocksize = 512;
vol->label.type = FSW_STRING_TYPE_EMPTY;
vol->host_data = host_data;
vol->host_table = host_table;
vol->fstype_table = fstype_table;
vol->host_string_type = host_table->native_string_type;
// let the fs driver mount the file system
status = vol->fstype_table->volume_mount(vol);
if (status)
goto errorexit;
// TODO: anything else?
*vol_out = vol;
return FSW_SUCCESS;
errorexit:
fsw_unmount(vol);
return status;
}
/**
* Unmount a volume by releasing all memory associated with it. This function is
* called by the host driver when a volume is no longer needed. It is also called
* by the core after a failed mount to clean up any allocated memory.
*
* Note that all dnodes must have been released before calling this function.
*/
void fsw_unmount(struct fsw_volume *vol)
{
if (!vol) {
return;
}
if (vol->root)
fsw_dnode_release(vol->root);
// TODO: check that no other dnodes are still around
vol->fstype_table->volume_free(vol);
fsw_blockcache_free(vol);
fsw_strfree(&vol->label);
fsw_free(vol);
}
/**
* Get in-depth information on the volume. This function can be called by the host
* driver to get additional information on the volume.
*/
fsw_status_t fsw_volume_stat(struct fsw_volume *vol, struct fsw_volume_stat *sb)
{
if (!vol) {
return FSW_UNSUPPORTED;
}
return vol->fstype_table->volume_stat(vol, sb);
}
/**
* Set the physical and logical block sizes of the volume. This functions is called by
* the file system driver to announce the block sizes it wants to use for accessing
* the disk (physical) and for addressing file contents (logical).
* Usually both sizes will be the same but there may be file systems that need to access
* metadata at a smaller block size than the allocation unit for files.
*
* Calling this function causes the block cache to be dropped. All pointers returned
* from fsw_block_get become invalid. This function should only be called while
* mounting the file system, not as a part of file access operations.
*
* Both sizes are measured in bytes, must be powers of 2, and must not be smaller
* than 512 bytes. The logical block size cannot be smaller than the physical block size.
*/
void fsw_set_blocksize_(struct fsw_volume *vol, fsw_u32 phys_blocksize, fsw_u32 log_blocksize)
{
// TODO: Check the sizes. Both must be powers of 2. log_blocksize must not be smaller than
// phys_blocksize.
// drop core block cache if present
fsw_blockcache_free(vol);
// signal host driver to drop caches etc.
vol->host_table->change_blocksize(vol,
vol->phys_blocksize, vol->log_blocksize,
phys_blocksize, log_blocksize);
vol->phys_blocksize = phys_blocksize;
vol->log_blocksize = log_blocksize;
}
/**
* Get a block of data from the disk. This function is called by the file system driver
* or by core functions. It calls through to the host driver's device access routine.
* Given a physical block number, it reads the block into memory (or fetches it from the
* block cache) and returns the address of the memory buffer. The caller should provide
* an indication of how important the block is in the cache_level parameter. Blocks with
* a low level are purged first. Some suggestions for cache levels:
*
* - 0: File data
* - 1: Directory data, symlink data
* - 2: File system metadata
* - 3..5: File system metadata with a high rate of access
*
* If this function returns successfully, the returned data pointer is valid until the
* caller calls fsw_block_release.
*/
fsw_status_t fsw_block_get_(struct VOLSTRUCTNAME *vol, fsw_u32 phys_bno, fsw_u32 cache_level, void **buffer_out)
{
fsw_status_t status;
fsw_u32 i, discard_level, new_bcache_size;
struct fsw_blockcache *new_bcache = NULL;
// TODO: allow the host driver to do its own caching; just call through if
// the appropriate function pointers are set
if (cache_level > MAX_CACHE_LEVEL)
cache_level = MAX_CACHE_LEVEL;
// check block cache
for (i = 0; i < vol->bcache_size; i++) {
if (vol->bcache[i].phys_bno == phys_bno) {
// cache hit!
if (vol->bcache[i].cache_level < cache_level)
vol->bcache[i].cache_level = cache_level; // promote the entry
vol->bcache[i].refcount++;
*buffer_out = vol->bcache[i].data;
return FSW_SUCCESS;
}
}
// find a free entry in the cache table
for (i = 0; i < vol->bcache_size; i++) {
if (vol->bcache[i].phys_bno == (fsw_u32)FSW_INVALID_BNO)
break;
}
if (i >= vol->bcache_size) {
for (discard_level = 0; discard_level <= MAX_CACHE_LEVEL; discard_level++) {
for (i = 0; i < vol->bcache_size; i++) {
if (vol->bcache[i].refcount == 0 && vol->bcache[i].cache_level <= discard_level)
break;
}
if (i < vol->bcache_size)
break;
}
}
if (i >= vol->bcache_size) {
// enlarge / create the cache
if (vol->bcache_size < 16)
new_bcache_size = 16;
else
new_bcache_size = vol->bcache_size << 1;
status = fsw_alloc(new_bcache_size * sizeof(struct fsw_blockcache), &new_bcache);
if (status != FSW_SUCCESS) {
if (new_bcache) {
fsw_free(new_bcache);
}
return status;
}
if (vol->bcache_size > 0) {
fsw_memcpy(new_bcache, vol->bcache, vol->bcache_size * sizeof(struct fsw_blockcache));
}
for (i = vol->bcache_size; i < new_bcache_size; i++) {
new_bcache[i].refcount = 0;
new_bcache[i].cache_level = 0;
new_bcache[i].phys_bno = (fsw_u32)FSW_INVALID_BNO;
new_bcache[i].data = NULL;
}
i = vol->bcache_size;
// switch caches
if (vol->bcache != NULL)
fsw_free(vol->bcache);
vol->bcache = new_bcache;
vol->bcache_size = new_bcache_size;
}
vol->bcache[i].phys_bno = (fsw_u32)FSW_INVALID_BNO;
// read the data
if (vol->bcache[i].data == NULL) {
status = fsw_alloc(vol->phys_blocksize, &vol->bcache[i].data);
if (status)
return status;
}
status = vol->host_table->read_block(vol, phys_bno, vol->bcache[i].data);
if (status)
return status;
vol->bcache[i].phys_bno = phys_bno;
vol->bcache[i].cache_level = cache_level;
vol->bcache[i].refcount = 1;
*buffer_out = vol->bcache[i].data;
return FSW_SUCCESS;
}
/**
* Releases a disk block. This function must be called to release disk blocks returned
* from fsw_block_get.
*/
void fsw_block_release_(struct VOLSTRUCTNAME *vol, fsw_u32 phys_bno, void *buffer)
{
fsw_u32 i;
if (!vol) {
return;
}
// TODO: allow the host driver to do its own caching; just call through if
// the appropriate function pointers are set
// update block cache
for (i = 0; i < vol->bcache_size; i++) {
if (vol->bcache[i].phys_bno == phys_bno && vol->bcache[i].refcount > 0)
vol->bcache[i].refcount--;
}
}
/**
* Release the block cache. Called internally when changing block sizes and when
* unmounting the volume. It frees all data occupied by the generic block cache.
*/
static void fsw_blockcache_free(struct fsw_volume *vol)
{
fsw_u32 i;
if (!vol) {
return;
}
for (i = 0; i < vol->bcache_size; i++) {
if (vol->bcache[i].data != NULL)
fsw_free(vol->bcache[i].data);
}
if (vol->bcache != NULL) {
fsw_free(vol->bcache);
vol->bcache = NULL;
}
vol->bcache_size = 0;
}
/**
* Add a new dnode to the list of known dnodes. This internal function is used when a
* dnode is created to add it to the dnode list that is used to search for existing
* dnodes by id.
*/
static void fsw_dnode_register(struct fsw_volume *vol, struct fsw_dnode *dno)
{
if (!vol || !dno) {
return;
}
dno->next = vol->dnode_head;
if (vol->dnode_head != NULL)
vol->dnode_head->prev = dno;
dno->prev = NULL;
vol->dnode_head = dno;
}
/**
* Create a dnode representing the root directory. This function is called by the file system
* driver while mounting the file system. The root directory is special because it has no parent
* dnode, its name is defined to be empty, and its type is also fixed. Otherwise, this functions
* behaves in the same way as fsw_dnode_create.
*/
fsw_status_t fsw_dnode_create_root_(struct fsw_volume *vol, fsw_u32 dnode_id, struct fsw_dnode **dno_out)
{
fsw_status_t status;
struct fsw_dnode *dno;
// allocate memory for the structure
status = fsw_alloc_zero(vol->fstype_table->dnode_struct_size, (void **)&dno);
if (status)
return status;
// fill the structure
dno->vol = vol;
dno->parent = NULL;
dno->dnode_id = dnode_id;
dno->type = FSW_DNODE_TYPE_DIR;
dno->refcount = 1;
dno->name.type = FSW_STRING_TYPE_EMPTY;
// TODO: instead, call a function to create an empty string in the native string type
fsw_dnode_register(vol, dno);
*dno_out = dno;
return FSW_SUCCESS;
}
/**
* Create a new dnode representing a file system object. This function is called by
* the file system driver in response to directory lookup or read requests. Note that
* if there already is a dnode with the given dnode_id on record, then no new object
* is created. Instead, the existing dnode is returned and its reference count
* increased. All other parameters are ignored in this case.
*
* The type passed into this function may be FSW_DNODE_TYPE_UNKNOWN. It is sufficient
* to fill the type field during the dnode_fill call.
*
* The name parameter must describe a string with the object's name. A copy will be
* stored in the dnode structure for future reference. The name will not be used to
* shortcut directory lookups, but may be used to reconstruct paths.
*
* If the function returns successfully, *dno_out contains a pointer to the dnode
* that must be released by the caller with fsw_dnode_release.
*/
fsw_status_t fsw_dnode_create_(struct fsw_dnode *parent_dno, fsw_u32 dnode_id, int type,
struct fsw_string *name, struct fsw_dnode **dno_out)
{
fsw_status_t status;
struct fsw_volume *vol;
struct fsw_dnode *dno;
if (!parent_dno) {
return FSW_NOT_FOUND;
}
vol = parent_dno->vol;
// check if we already have a dnode with the same id
for (dno = vol->dnode_head; dno; dno = dno->next) {
if (dno->dnode_id == dnode_id) {
fsw_dnode_retain(dno);
*dno_out = dno;
return FSW_SUCCESS;
}
}
// allocate memory for the structure
status = fsw_alloc_zero(vol->fstype_table->dnode_struct_size, (void **)&dno);
if (status)
return status;
// fill the structure
dno->vol = vol;
dno->parent = parent_dno;
fsw_dnode_retain(dno->parent);
dno->dnode_id = dnode_id;
dno->type = type;
dno->refcount = 1;
status = fsw_strdup_coerce(&dno->name, vol->host_table->native_string_type, name);
if (status) {
fsw_free(dno);
return status;
}
fsw_dnode_register(vol, dno);
*dno_out = dno;
return FSW_SUCCESS;
}
/**
* Increases the reference count of a dnode. This must be balanced with
* fsw_dnode_release calls. Note that some dnode functions return a retained
* dnode pointer to their caller.
*/
void fsw_dnode_retain(struct fsw_dnode *dno)
{
dno->refcount++;
}
/**
* Release a dnode pointer, deallocating it if this was the last reference.
* This function decrements the reference counter of the dnode. If the counter
* reaches zero, the dnode is freed. Since the parent dnode is released
* during that process, this function may cause it to be freed, too.
*/
void fsw_dnode_release(struct fsw_dnode *dno)
{
struct fsw_volume *vol;
struct fsw_dnode *parent_dno;
if (!dno) {
return;
}
vol = dno->vol;
dno->refcount--;
#if defined(FSW_DNODE_CACHE_SIZE) && FSW_DNODE_CACHE_SIZE > 0 // numcslots always zero for non dir dnodes
if (dno->refcount != dno->numcslots)
return;
#else
if (dno->refcount > 0)
return;
#endif
parent_dno = dno->parent;
// de-register from volume's list
if (dno->next)
dno->next->prev = dno->prev;
if (dno->prev)
dno->prev->next = dno->next;
if (vol->dnode_head == dno)
vol->dnode_head = dno->next;
#if defined(FSW_DNODE_CACHE_SIZE) && FSW_DNODE_CACHE_SIZE > 0
if (dno->type == FSW_DNODE_TYPE_DIR) {
int i;
for (i = 0; i < FSW_DNODE_CACHE_SIZE; i++) {
struct fsw_dnode *cache_entry = dno->cache[i];
if (cache_entry == NULL)
continue;
// numcslots not decremented on purpose
fsw_dnode_release(cache_entry);
}
}
#endif
// run fstype-specific cleanup
vol->fstype_table->dnode_free(vol, dno);
fsw_strfree(&dno->name);
fsw_free(dno);
// release our pointer to the parent, possibly deallocating it, too
if (parent_dno)
fsw_dnode_release(parent_dno);
}
/**
* Get full information about a dnode from disk. This function is called by the host
* driver as well as by the core functions. Some file systems defer reading full
* information on a dnode until it is actually needed (i.e. separation between
* directory and inode information). This function makes sure that all information
* is available in the dnode structure. The following fields may not have a correct
* value until fsw_dnode_fill has been called:
*
* type, size
*/
fsw_status_t fsw_dnode_fill(struct fsw_dnode *dno)
{
// TODO: check a flag right here, call fstype's dnode_fill only once per dnode
return dno->vol->fstype_table->dnode_fill(dno->vol, dno);
}
/**
* Get extended information about a dnode. This function can be called by the host
* driver to get a full compliment of information about a dnode in addition to the
* fields of the fsw_dnode structure itself.
*
* Some data requires host-specific conversion to be useful (i.e. timestamps) and
* will be passed to callback functions instead of being written into the structure.
* These callbacks must be filled in by the caller.
*/
fsw_status_t fsw_dnode_stat(struct fsw_dnode *dno, struct fsw_dnode_stat_str *sb)
{
fsw_status_t status;
status = fsw_dnode_fill(dno);
if (status)
return status;
sb->used_bytes = 0;
status = dno->vol->fstype_table->dnode_stat(dno->vol, dno, sb);
if (!status && !sb->used_bytes)
sb->used_bytes = FSW_U64_DIV(dno->size + dno->vol->log_blocksize - 1, dno->vol->log_blocksize);
return status;
}
/**
* Lookup a directory entry by name in directory cache.
* Given a directory dnode and a file name, it looks up the named entry in the
* directory cache. If miss, the function calls fstype lookup and cache positive results.
*
* If the dnode is not a directory, the call will fail.
*
* If the function returns FSW_SUCCESS, *child_dno_out points to the requested directory
* entry. The caller must call fsw_dnode_release on it.
*/
fsw_status_t fsw_dnode_lookup_cache(struct fsw_dnode *dno,
struct fsw_string *lookup_name, struct fsw_dnode **child_dno_out)
{
fsw_status_t status;
struct fsw_volume *vol = dno->vol;
struct fsw_dnode *cache_dno = NULL;
#if defined(FSW_DNODE_CACHE_SIZE) && FSW_DNODE_CACHE_SIZE > 0
int i;
#endif
fsw_dnode_retain(dno);
// ensure we have full information
status = fsw_dnode_fill(dno);
if (status)
goto errorexit;
// make sure we operate on a directory
if (dno->type != FSW_DNODE_TYPE_DIR) {
status = FSW_UNSUPPORTED;
goto errorexit;
}
#if defined(FSW_DNODE_CACHE_SIZE) && FSW_DNODE_CACHE_SIZE > 0
for (i = 0; i < FSW_DNODE_CACHE_SIZE; i++) {
struct fsw_dnode *cache_entry = dno->cache[i];
if (cache_entry == NULL)
continue;
if (fsw_streq(&cache_entry->name, lookup_name)) {
cache_dno = cache_entry;
break;
}
}
if (cache_dno != NULL) {
// move found entry to first slot
while (i > 0) {
dno->cache[i] = dno->cache[i - 1];
i--;
}
dno->cache[0] = cache_dno;
fsw_dnode_retain(cache_dno);
goto goodexit;
}
#endif
// Cache miss (or no cache at all). Do real lookup
status = vol->fstype_table->dir_lookup(vol, dno, lookup_name, &cache_dno);
if (status)
goto errorexit;
#if defined(FSW_DNODE_CACHE_SIZE) && FSW_DNODE_CACHE_SIZE > 0
// release dnode pushed out of cache
i = FSW_DNODE_CACHE_SIZE - 1;
if (dno->cache[i] != NULL) {
dno->numcslots--;
fsw_dnode_release(dno->cache[i]);
}
// cache found entry at first slot
while (i > 0) {
dno->cache[i] = dno->cache[i - 1];
i--;
}
dno->cache[0] = cache_dno;
dno->numcslots++;
fsw_dnode_retain(cache_dno);
goodexit:
#endif
fsw_dnode_release(dno);
*child_dno_out = cache_dno;
return FSW_SUCCESS;
errorexit:
fsw_dnode_release(dno);
if (cache_dno != NULL)
fsw_dnode_release(cache_dno);
return status;
}
/**
* Lookup a directory entry by name. This function is called by the host driver.
* Given a directory dnode and a file name, it looks up the named entry in the
* directory.
*
* If the dnode is symlink it resolved.
*
* If the dnode is not a directory, the call will fail.
*
* If the function returns FSW_SUCCESS, *child_dno_out points to the requested directory
* entry. The caller must call fsw_dnode_release on it.
*/
fsw_status_t fsw_dnode_lookup(struct fsw_dnode *dno,
struct fsw_string *lookup_name,
struct fsw_dnode **child_dno_out)
{
fsw_status_t status;
struct fsw_dnode *child_dno = NULL;
fsw_dnode_retain(dno);
DBG(" lookup name at dnode type=%d\n", dno->type);
// ensure we have full information
status = fsw_dnode_fill(dno);
if (status)
goto errorexit;
// resolve symlink if necessary
if (dno->type == FSW_DNODE_TYPE_SYMLINK) {
DBG("resolve link at core\n");
status = fsw_dnode_resolve(dno, &child_dno);
if (status)
goto errorexit;
// symlink target becomes the new dno
fsw_dnode_release(dno);
dno = child_dno; // is already retained
child_dno = NULL;
// ensure we have full information
status = fsw_dnode_fill(dno);
if (status)
goto errorexit;
}
// make sure we operate on a directory
if (dno->type != FSW_DNODE_TYPE_DIR) {
status = FSW_UNSUPPORTED;
goto errorexit;
}
// check special paths
if (fsw_streq_cstr(lookup_name, ".")) { // self directory
child_dno = dno;
fsw_dnode_retain(child_dno);
} else if (fsw_streq_cstr(lookup_name, "..")) { // parent directory
if (dno->parent == NULL) {
// We cannot go up from the root directory. Caution: Certain apps like the EFI shell
// rely on this behaviour!
status = FSW_NOT_FOUND;
DBG("go up root\n");
goto errorexit;
}
child_dno = dno->parent;
fsw_dnode_retain(child_dno);
} else {
// do an cached actual lookup
DBG(" dnode_lookup_cache\n");
status = fsw_dnode_lookup_cache(dno, lookup_name, &child_dno);
if (status)
goto errorexit;
}
fsw_dnode_release(dno);
*child_dno_out = child_dno;
return FSW_SUCCESS;
errorexit:
DBG(" lookup failed status=%a\n", fsw_errors[status]);
fsw_dnode_release(dno);
if (child_dno != NULL)
fsw_dnode_release(child_dno);
return status;
}
/**
* Find a file system object by path. This function is called by the host driver.
* Given a directory dnode and a relative or absolute path, it walks the directory
* tree until it finds the target dnode. If an intermediate node turns out to be
* a symlink, it is resolved automatically. If the target node is a symlink, it
* is not resolved.
*
* If the function returns FSW_SUCCESS, *child_dno_out points to the requested directory
* entry. The caller must call fsw_dnode_release on it.
*/
fsw_status_t fsw_dnode_lookup_path(struct fsw_dnode *dno,
struct fsw_string *lookup_path,
char separator,
struct fsw_dnode **child_dno_out)
{
fsw_status_t status;
struct fsw_volume *vol;
struct fsw_dnode *child_dno = NULL;
struct fsw_string lookup_name;
struct fsw_string remaining_path;
int root_if_empty;
if (!dno) {
return FSW_UNSUPPORTED;
}
vol = dno->vol;
#if DEBUG_CR
{
int i;
fsw_u16 ch;
if (lookup_path->type == FSW_STRING_TYPE_ISO88591) {
DBG("dnode %a lookup ASCII\n", lookup_path->data);
} else if (lookup_path->type == FSW_STRING_TYPE_UTF16) {
DBG("dnode lookup UNI:");
for (i = 0; i < lookup_path->len; i++) {
ch = ((fsw_u16 *) lookup_path->data)[i];
DBG("%c", ch?ch:L'@');
}
DBG("\n");
}
}
#endif
// remaining_path = *lookup_path;
fsw_memcpy(&remaining_path, lookup_path, sizeof(struct fsw_string));
fsw_dnode_retain(dno);
DBG("len of remaining_path=%d\n", fsw_strlen(&remaining_path));
// loop over the path
for (root_if_empty = 1; fsw_strlen(&remaining_path) > 0; root_if_empty = 0) {
// parse next path component
fsw_strsplit(&lookup_name, &remaining_path, separator);
DBG(" len of lookup_name=%d\n", fsw_strlen(&lookup_name));
if (fsw_strlen(&lookup_name) == 0) { // empty path component
if (root_if_empty) {
child_dno = vol->root;
DBG("set lookup root\n");
} else {
child_dno = dno;
DBG("set lookup current\n");
}
fsw_dnode_retain(child_dno);
} else {
// do an actual directory lookup
status = fsw_dnode_lookup(dno, &lookup_name, &child_dno);
if (status) {
DBG("dnode lookup failed %a\n", fsw_errors[status]);
goto errorexit;
}
}
// child_dno becomes the new dno
fsw_dnode_release(dno);
dno = child_dno; // is already retained
child_dno = NULL;
}
*child_dno_out = dno;
return FSW_SUCCESS;
errorexit:
fsw_dnode_release(dno);
if (child_dno != NULL)
fsw_dnode_release(child_dno);
return status;
}
/**
* Get the next directory item in sequential order. This function is called by the
* host driver to read the complete contents of a directory in sequential (file system
* defined) order. Calling this function returns the next entry. Iteration state is
* kept by a shandle on the directory's dnode. The caller must set up the shandle
* when starting the iteration.
*
* When the end of the directory is reached, this function returns FSW_NOT_FOUND.
* If the function returns FSW_SUCCESS, *child_dno_out points to the next directory
* entry. The caller must call fsw_dnode_release on it.
*/
fsw_status_t fsw_dnode_dir_read(struct fsw_shandle *shand, struct fsw_dnode **child_dno_out)
{
fsw_status_t status;
struct fsw_dnode *dno;
fsw_u64 saved_pos;
if (!shand) {
return FSW_UNSUPPORTED;
}
dno = shand->dnode;
if (!dno || dno->type != FSW_DNODE_TYPE_DIR)
return FSW_UNSUPPORTED;
saved_pos = shand->pos;
DBG("core dir_read at id=%d\n", dno->dnode_id);
status = dno->vol->fstype_table->dir_read(dno->vol, dno, shand, child_dno_out);
if (status)
shand->pos = saved_pos;
return status;
}
/**
* Read the target path of a symbolic link. This function is called by the host driver
* to read the "content" of a symbolic link, that is the relative or absolute path
* it points to.
*
* If the function returns FSW_SUCCESS, the string handle provided by the caller is
* filled with a string in the host's preferred encoding. The caller is responsible
* for calling fsw_strfree on the string.
*/
fsw_status_t fsw_dnode_readlink(struct fsw_dnode *dno, struct fsw_string *target_name)
{
fsw_status_t status;
status = fsw_dnode_fill(dno);
if (status)
return status;
if (dno->type != FSW_DNODE_TYPE_SYMLINK)
return FSW_UNSUPPORTED;
DBG("core readlink at id=%d\n", dno->dnode_id);
return dno->vol->fstype_table->readlink(dno->vol, dno, target_name);
}
/**
* Read the target path of a symbolic link by accessing file data. This function can
* be called by the file system driver if the file system stores the target path
* as normal file data. This function will open an shandle, read the whole content
* of the file into a buffer, and build a string from that. Currently the encoding
* for the string is fixed as FSW_STRING_TYPE_ISO88591.
*
* If the function returns FSW_SUCCESS, the string handle provided by the caller is
* filled with a string in the host's preferred encoding. The caller is responsible
* for calling fsw_strfree on the string.
*/
fsw_status_t fsw_dnode_readlink_data_(struct fsw_dnode *dno, struct fsw_string *link_target)
{
fsw_status_t status;
struct fsw_shandle shand;
fsw_u32 buffer_size;
char buffer[FSW_PATH_MAX];
struct fsw_string s;
if (dno == NULL || dno->size > FSW_PATH_MAX)
return FSW_VOLUME_CORRUPTED;
s.type = FSW_STRING_TYPE_ISO88591;
s.size = s.len = (int)dno->size;
s.data = buffer;
// open shandle and read the data
status = fsw_shandle_open(dno, &shand);
if (status)
return status;
buffer_size = (fsw_u32)s.size;
status = fsw_shandle_read(&shand, &buffer_size, buffer);
fsw_shandle_close(&shand);
if (status)
return status;
if ((int)buffer_size < s.size)
return FSW_VOLUME_CORRUPTED;
if (dno->vol == NULL)
return FSW_VOLUME_CORRUPTED;
status = fsw_strdup_coerce(link_target, dno->vol->host_string_type, &s);
return status;
}
/**
* Resolve a symbolic link. This function can be called by the host driver to make
* sure the a dnode is fully resolved instead of pointing at a symlink. If the dnode
* passed in is not a symlink, it is returned unmodified.
*
* Note that absolute paths will be resolved relative to the root directory of the
* volume. If the host is an operating system with its own VFS layer, it should
* resolve symlinks on its own.
*
* If the function returns FSW_SUCCESS, *target_dno_out points at a dnode that is
* not a symlink. The caller is responsible for calling fsw_dnode_release on it.
*/
fsw_status_t fsw_dnode_resolve(struct fsw_dnode *dno, struct fsw_dnode **target_dno_out)
{
fsw_status_t status;
struct fsw_string target_name;
struct fsw_dnode *target_dno;
fsw_dnode_retain(dno);
while (1) {
// get full information
status = fsw_dnode_fill(dno);
if (status)
goto errorexit;
if (dno->type != FSW_DNODE_TYPE_SYMLINK) {
// found a non-symlink target, return it
*target_dno_out = dno;
return FSW_SUCCESS;
}
if (dno->parent == NULL) { // safety measure, cannot happen in theory
status = FSW_NOT_FOUND;
goto errorexit;
}
// read the link's target
status = fsw_dnode_readlink(dno, &target_name);
if (status)
goto errorexit;
// resolve it
status = fsw_dnode_lookup_path(dno->parent, &target_name, '/', &target_dno);
// status = fsw_dnode_lookup_path(dno->parent, &target_name, '\\', &target_dno);
fsw_strfree(&target_name);
if (status)
goto errorexit;
// target_dno becomes the new dno
fsw_dnode_release(dno);
dno = target_dno; // is already retained
}
errorexit:
fsw_dnode_release(dno);
return status;
}
/**
* Set up a shandle (storage handle) to access a file's data. This function is called
* by the host driver and by the core when they need to access a file's data. It is also
* used in accessing the raw data of directories and symlinks if the file system uses
* the same mechanisms for storing the data of those items.
*
* The storage for the fsw_shandle structure is provided by the caller. The dnode and pos
* fields may be accessed, pos may also be written to to set the file pointer. The file's
* data size is available as shand->dnode->size.
*
* If this function returns FSW_SUCCESS, the caller must call fsw_shandle_close to release
* the dnode reference held by the shandle.
*/
fsw_status_t fsw_shandle_open_(struct fsw_dnode *dno, struct fsw_shandle *shand)
{
fsw_status_t status;
struct fsw_volume *vol;
if (!dno || !dno->vol) {
return FSW_VOLUME_CORRUPTED;
}
vol = dno->vol;
// read full dnode information into memory
status = vol->fstype_table->dnode_fill(vol, dno);
if (status)
return status;
// setup shandle
fsw_dnode_retain(dno);
shand->dnode = dno;
shand->pos = 0;
shand->extent.type = FSW_EXTENT_TYPE_INVALID;
return FSW_SUCCESS;
}
/**
* Close a shandle after accessing the dnode's data. This function is called by the host
* driver or core functions when they are finished with accessing a file's data. It
* releases the dnode reference and frees any buffers associated with the shandle itself.
* The dnode is only released if this was the last reference using it.
*/
void fsw_shandle_close(struct fsw_shandle *shand)
{
if (shand->extent.type == FSW_EXTENT_TYPE_BUFFER) {
fsw_free(shand->extent.buffer);
}
fsw_dnode_release(shand->dnode);
}
/**
* Read data from a shandle (storage handle for a dnode). This function is called by the
* host driver or internally when data is read from a file. TODO: more
*/
fsw_status_t fsw_shandle_read(struct fsw_shandle *shand, fsw_u32 *buffer_size_inout, void *buffer_in)
{
fsw_status_t status;
struct fsw_dnode *dno = shand->dnode;
struct fsw_volume *vol = dno->vol;
fsw_u8 *buffer, *block_buffer;
fsw_u32 buflen, copylen, pos;
fsw_u32 log_bno, pos_in_extent, phys_bno, pos_in_physblock;
fsw_u32 cache_level;
if (shand->pos >= dno->size) { // already at EOF
*buffer_size_inout = 0;
return FSW_SUCCESS;
}
// initialize vars
buffer = buffer_in;
buflen = *buffer_size_inout;
pos = (fsw_u32)shand->pos;
cache_level = (dno->type != FSW_DNODE_TYPE_FILE) ? 1 : 0;
// restrict read to file size
if (buflen > dno->size - pos)
buflen = (fsw_u32)(dno->size - pos);
while (buflen > 0) {
// get extent for the current logical block
log_bno = pos / vol->log_blocksize;
if (shand->extent.type == FSW_EXTENT_TYPE_INVALID ||
log_bno < shand->extent.log_start ||
log_bno >= shand->extent.log_start + shand->extent.log_count) {
if (shand->extent.type == FSW_EXTENT_TYPE_BUFFER)
fsw_free(shand->extent.buffer);
// ask the file system for the proper extent
shand->extent.log_start = log_bno;
status = vol->fstype_table->get_extent(vol, dno, &shand->extent);
if (status) {
shand->extent.type = FSW_EXTENT_TYPE_INVALID;
return status;
}
}
pos_in_extent = pos - shand->extent.log_start * vol->log_blocksize;
// dispatch by extent type
if (shand->extent.type == FSW_EXTENT_TYPE_PHYSBLOCK) {
// convert to physical block number and offset
phys_bno = shand->extent.phys_start + pos_in_extent / vol->phys_blocksize;
pos_in_physblock = pos_in_extent & (vol->phys_blocksize - 1);
copylen = vol->phys_blocksize - pos_in_physblock;
if (copylen > buflen)
copylen = buflen;
// get one physical block
status = fsw_block_get(vol, phys_bno, cache_level, (void **)&block_buffer);
if (status)
return status;
// copy data from it
fsw_memcpy(buffer, block_buffer + pos_in_physblock, copylen);
fsw_block_release(vol, phys_bno, block_buffer);
} else if (shand->extent.type == FSW_EXTENT_TYPE_BUFFER) {
copylen = shand->extent.log_count * vol->log_blocksize - pos_in_extent;
if (copylen > buflen)
copylen = buflen;
fsw_memcpy(buffer, (fsw_u8 *)shand->extent.buffer + pos_in_extent, copylen);
} else { // _SPARSE or _INVALID
copylen = shand->extent.log_count * vol->log_blocksize - pos_in_extent;
if (copylen > buflen)
copylen = buflen;
fsw_memzero(buffer, copylen);
}
buffer += copylen;
buflen -= copylen;
pos += copylen;
}
*buffer_size_inout = (fsw_u32)(pos - shand->pos);
shand->pos = pos;
return FSW_SUCCESS;
}
// EOF