61 static os_error_t impl_delete_file(
fs_t *fs,
fs_inode_t *node,
const char *name)
87 static os_error_t impl_delete_dir(
fs_t *fs,
fs_inode_t *node,
const char *name)
111 static os_error_t impl_iter_next(
fs_iter_t *iter)
121 static os_error_t impl_iter_end(
fs_iter_t *iter)
141 for (
size_t i = 0; i < len - 1; i++)
144 fs_inode_t *node = impl_query_inode(fs, current, part);
160 return impl_query_inode(fs, current,
vector_tail(parts));
182 for (
size_t i = 0; i < len - 1; i++)
185 fs_inode_t *node = impl_query_inode(fs, current, part);
188 case eOsNodeDir: current = node;
break;
203 for (
size_t i = 0; i < len - 1; i++)
206 fs_inode_t *node = impl_query_inode(fs, current, part);
209 case eOsNodeDir: current = node;
break;
210 default:
return false;
225 for (
size_t i = 0; i < len - 1; i++)
228 fs_inode_t *node = impl_query_inode(fs, current, part);
231 case eOsNodeDir: current = node;
break;
236 return impl_delete_file(fs, current,
vector_tail(parts));
243 io_t *io =
io_new(&kInvalidIo, flags, name, NULL, arena);
250 return make_error_file(name, flags,
eOsNotFound, arena);
262 for (
size_t i = 0; i < len - 1; i++)
265 fs_inode_t *node = impl_query_inode(fs, current, part);
268 case eOsNodeDir: current = node;
break;
269 case eOsNodeNone: current = impl_create_dir(fs, current, part);
break;
270 default:
return make_missing_file(path, flags, fs->
arena);
274 CTASSERTF(current != NULL,
"current is NULL (path = %s)", path);
279 return impl_query_file(fs, file, flags);
287 return make_error_file(path, eOsAccessNone, result.
error, fs->
arena);
290 return impl_query_file(fs, result.
node, flags);
293 return make_missing_file(path, flags, fs->
arena);
303 case eOsNodeDir:
return dir;
304 case eOsNodeNone:
return impl_create_dir(fs, node, name);
306 default:
return NULL;
312 fs_inode_t *file = impl_query_inode(fs, node, name);
315 case eOsNodeFile:
return file;
316 case eOsNodeNone:
return impl_create_file(fs, node, name).
node;
318 default:
return NULL;
326 case eOsNodeDir:
return get_dir_or_create(fs, node, name);
327 case eOsNodeFile:
return get_file_or_create(fs, node, name);
328 default:
CT_NEVER(
"invalid inode type for %s", name);
343 for (
size_t i = 0; i < len; i++)
346 fs_inode_t *node = impl_query_inode(fs, current, part);
353 current = impl_create_dir(fs, current, part);
374 for (
size_t i = 0; i < len; i++)
377 fs_inode_t *node = impl_query_inode(fs, current, part);
393 return inode_is(current, eOsNodeDir);
405 for (
size_t i = 0; i < len - 1; i++)
408 fs_inode_t *node = impl_query_inode(fs, current, part);
425 return impl_delete_dir(fs, current,
vector_tail(parts));
436 io_t *src_io = impl_query_file(src_fs, src_node, eOsAccessRead);
437 io_t *dst_io = impl_query_file(dst_fs, dst_node, eOsAccessWrite | eOsAccessTruncate);
452 const void *data =
io_map(src_io, eOsProtectRead);
453 CTASSERTF(data != NULL,
"failed to map file during sync (path = %s)",
io_name(src_io));
486 if (child->
type == eOsNodeDir)
488 sync_dir(dst, src, other, child);
490 else if (child->
type == eOsNodeFile)
492 if (sync_file(dst, src, other, child) !=
eOsSuccess)
500 CT_NEVER(
"invalid inode type (type = %d)", child->
type);
516 return sync_dir(dst, src, dst->
root, src->
root);
549 return find_inode(fs, fs->
root, path);
576 data->current = NULL;
578 os_error_t err = impl_iter_begin(fs, node, data);
595 os_error_t err = impl_iter_end(iter);
607 os_error_t err = impl_iter_next(iter);
bool inode_is(const fs_inode_t *inode, os_dirent_t type)
CT_NODISCARD CT_CONSTFN STA_RET_STRING CT_OS_API const char * os_dirent_string(os_dirent_t type)
get the string representation of a directory entry type
os_dirent_t
directory entry type
os_access_t
file access mode
STA_DECL void fs_delete(fs_t *fs)
STA_DECL os_error_t fs_iter_next(fs_iter_t *iter, fs_inode_t **inode)
STA_DECL os_error_t fs_iter_begin(fs_t *fs, const fs_inode_t *node, fs_iter_t **iter)
#define STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
CT_NODISCARD CT_PUREFN CT_BASE_API size_t ctu_strlen(const char *str)
get the length of a string not including the null terminator equivalent to strlen but with safety che...
STA_DECL os_dirent_t fs_inode_type(const fs_inode_t *inode)
get the type of an inode
STA_DECL bool fs_dir_exists(fs_t *fs, const char *path)
query if a directory exists
STA_DECL os_error_t fs_file_delete(fs_t *fs, const char *path)
delete a file
STA_DECL bool fs_file_exists(fs_t *fs, const char *path)
query if a file exists
STA_DECL os_error_t fs_dir_delete(fs_t *fs, const char *path)
delete a directory delete a directory and all contained files and folders
STA_DECL os_error_t fs_dir_create(fs_t *fs, const char *path)
create a directory create a directory and all child directories inside a filesystem
STA_DECL fs_inode_t * fs_find_inode(fs_t *fs, const char *path)
find an inode in a filesystem
STA_DECL io_t * fs_open(fs_t *fs, const char *path, os_access_t flags)
open a file at a given location in the filesystem
STA_DECL os_error_t fs_iter_end(fs_iter_t *iter)
end an iteration
STA_DECL void fs_file_create(fs_t *fs, const char *path)
create a file
STA_DECL bool fs_inode_is(const fs_inode_t *inode, os_dirent_t type)
check if a given inode is of a certain type
STA_DECL const char * fs_inode_name(const fs_inode_t *inode)
get the name of an inode
STA_DECL sync_result_t fs_sync(fs_t *dst, fs_t *src)
synchronize 2 filesystems copies all folders and files from src to dst
STA_DECL fs_inode_t * fs_root_inode(fs_t *fs)
get the root inode of a filesystem
CT_IO_API io_t * io_new(const io_callbacks_t *cb, os_access_t flags, const char *name, STA_READS(cb->size) const void *data, arena_t *arena)
create a new IO object for a given interface
CT_IO_API size_t io_write(io_t *io, STA_READS(size) const void *src, size_t size)
write to an io object
CT_NODISCARD CT_IO_API os_error_t io_error(const io_t *io)
get the last error from the io object
CT_IO_API os_error_t io_close(INOUT_NOTNULL io_t *io)
destroy an IO object
CT_NODISCARD CT_IO_API size_t io_size(io_t *io)
get the total size of an io objects contents
CT_NODISCARD CT_IO_API const char * io_name(const io_t *io)
get the name of an io object
CT_NODISCARD CT_IO_API void * io_map(io_t *io, os_protect_t protect)
map an io object into memory maps an io objects entire contents into memory.
#define CT_ALLOC_SIZE_UNKNOWN
unknown allocation size constant when freeing or reallocating memory, this can be used as the size to...
#define ARENA_MALLOC(size, name, parent, arena)
allocate memory from a custom allocator
CT_ARENA_API void arena_free(STA_RELEASE void *ptr, size_t size, arena_t *arena)
release memory from a custom allocator
#define CT_ASSERT_RANGE(value, min, max)
assert that a value is in a range inclusive bounds check
#define CT_NEVER(...)
assert that a code path is never reached
#define CTASSERT(expr)
assert a condition, prints the condition as a message
#define CTASSERTF(expr,...)
assert a condition with a message and optional format arguments
#define CT_PARANOID_ASSERTF(expr,...)
assert a condition with a message and optional format arguments
CT_NODISCARD CT_STD_API vector_t * str_split(const char *str, const char *sep, arena_t *arena)
split a string into a vector by a separator
CT_NODISCARD CT_PUREFN CT_STD_API void * vector_get(const vector_t *vector, size_t index)
get a value from a vector
CT_NODISCARD CT_PUREFN CT_STD_API void * vector_tail(const vector_t *vector)
get the last element of a vector
CT_NODISCARD CT_PUREFN CT_STD_API size_t vector_len(const vector_t *vector)
get the length of a vector
CT_NODISCARD OUT_NOTNULL os_inode_t * dir
fs_dir_create_t pfn_create_dir
fs_iter_next_t pfn_iter_next
fs_file_create_t pfn_create_file
fs_iter_begin_t pfn_iter_begin
fs_query_file_t pfn_query_file
fs_dir_delete_t pfn_delete_dir
fs_file_delete_t pfn_delete_file
fs_iter_end_t pfn_iter_end
fs_query_node_t pfn_query_node
const fs_callbacks_t * cb
callbacks
fs_inode_t * root
root inode
os_error_t error
the last error set on this object
the result of a fs_sync call this is here because we cant use Compiler message notification in the fs...
a generic vector of pointers