toaruos/kernel/fs/vfs.c
2015-05-21 04:28:14 -04:00

828 lines
19 KiB
C

/* vim: tabstop=4 shiftwidth=4 noexpandtab
* This file is part of ToaruOS and is released under the terms
* of the NCSA / University of Illinois License - see LICENSE.md
* Copyright (C) 2011-2014 Kevin Lange
* Copyright (C) 2014 Lioncash
* Copyright (C) 2012 Tianyi Wang
*
* Virtual File System
*
*/
#include <system.h>
#include <fs.h>
#include <printf.h>
#include <list.h>
#include <process.h>
#include <logging.h>
#include <hashmap.h>
tree_t * fs_tree = NULL; /* File system mountpoint tree */
fs_node_t * fs_root = NULL; /* Pointer to the root mount fs_node (must be some form of filesystem, even ramdisk) */
hashmap_t * fs_types = NULL;
static struct dirent * readdir_mapper(fs_node_t *node, uint32_t index) {
tree_node_t * d = (tree_node_t *)node->device;
if (!d) return NULL;
if (index == 0) {
struct dirent * dir = malloc(sizeof(struct dirent));
strcpy(dir->name, ".");
dir->ino = 0;
return dir;
} else if (index == 1) {
struct dirent * dir = malloc(sizeof(struct dirent));
strcpy(dir->name, "..");
dir->ino = 1;
return dir;
}
index -= 2;
unsigned int i = 0;
foreach(child, d->children) {
if (i == index) {
/* Recursively print the children */
tree_node_t * tchild = (tree_node_t *)child->value;
struct vfs_entry * n = (struct vfs_entry *)tchild->value;
struct dirent * dir = malloc(sizeof(struct dirent));
size_t len = strlen(n->name) + 1;
memcpy(&dir->name, n->name, MIN(256, len));
dir->ino = i;
return dir;
}
++i;
}
return NULL;
}
static fs_node_t * vfs_mapper(void) {
fs_node_t * fnode = malloc(sizeof(fs_node_t));
memset(fnode, 0x00, sizeof(fs_node_t));
fnode->mask = 0666;
fnode->flags = FS_DIRECTORY;
fnode->readdir = readdir_mapper;
return fnode;
}
/**
* read_fs: Read a file system node based on its underlying type.
*
* @param node Node to read
* @param offset Offset into the node data to read from
* @param size How much data to read (in bytes)
* @param buffer A buffer to copy of the read data into
* @returns Bytes read
*/
uint32_t read_fs(fs_node_t *node, uint32_t offset, uint32_t size, uint8_t *buffer) {
if (!node) return -1;
if (node->read) {
uint32_t ret = node->read(node, offset, size, buffer);
return ret;
} else {
return -1;
}
}
/**
* write_fs: Write a file system node based on its underlying type.
*
* @param node Node to write to
* @param offset Offset into the node data to write to
* @param size How much data to write (in bytes)
* @param buffer A buffer to copy from
* @returns Bytes written
*/
uint32_t write_fs(fs_node_t *node, uint32_t offset, uint32_t size, uint8_t *buffer) {
if (!node) return -1;
if (node->write) {
uint32_t ret = node->write(node, offset, size, buffer);
return ret;
} else {
return -1;
}
}
//volatile uint8_t tmp_refcount_lock = 0;
static spin_lock_t tmp_refcount_lock = { 0 };
void vfs_lock(fs_node_t * node) {
spin_lock(tmp_refcount_lock);
node->refcount = -1;
spin_unlock(tmp_refcount_lock);
}
/**
* open_fs: Open a file system node.
*
* @param node Node to open
* @param flags Same as open, specifies read/write/append/truncate
*/
void open_fs(fs_node_t *node, unsigned int flags) {
if (!node) return;
if (node->refcount >= 0) {
spin_lock(tmp_refcount_lock);
node->refcount++;
spin_unlock(tmp_refcount_lock);
}
if (node->open) {
node->open(node, flags);
}
}
/**
* close_fs: Close a file system node
*
* @param node Node to close
*/
void close_fs(fs_node_t *node) {
assert(node != fs_root && "Attempted to close the filesystem root. kablooey");
if (!node) {
debug_print(WARNING, "Double close? This isn't an fs_node.");
return;
}
if (node->refcount == -1) return;
spin_lock(tmp_refcount_lock);
node->refcount--;
if (node->refcount == 0) {
debug_print(NOTICE, "Node refcount [%s] is now 0: %d", node->name, node->refcount);
if (node->close) {
node->close(node);
}
free(node);
}
spin_unlock(tmp_refcount_lock);
}
/**
* chmod_fs
*/
int chmod_fs(fs_node_t *node, int mode) {
if (node->chmod) {
return node->chmod(node, mode);
}
return 0;
}
/**
* readdir_fs: Read a directory for the requested index
*
* @param node Directory to read
* @param index Offset to look for
* @returns A dirent object.
*/
struct dirent *readdir_fs(fs_node_t *node, uint32_t index) {
if (!node) return NULL;
if ((node->flags & FS_DIRECTORY) && node->readdir) {
struct dirent *ret = node->readdir(node, index);
return ret;
} else {
return (struct dirent *)NULL;
}
}
/**
* finddir_fs: Find the requested file in the directory and return an fs_node for it
*
* @param node Directory to search
* @param name File to look for
* @returns An fs_node that the caller can free
*/
fs_node_t *finddir_fs(fs_node_t *node, char *name) {
if (!node) return NULL;
if ((node->flags & FS_DIRECTORY) && node->finddir) {
fs_node_t *ret = node->finddir(node, name);
return ret;
} else {
debug_print(WARNING, "Node passed to finddir_fs isn't a directory!");
debug_print(WARNING, "node = 0x%x, name = %s", node, name);
return (fs_node_t *)NULL;
}
}
/**
* ioctl_fs: Control Device
*
* @param node Device node to control
* @param request Device-specific request code
* @param argp Depends on `request`
* @returns Depends on `request`
*/
int ioctl_fs(fs_node_t *node, int request, void * argp) {
if (!node) return -1;
if (node->ioctl) {
return node->ioctl(node, request, argp);
} else {
return -1; /* TODO Should actually be ENOTTY, but we're bad at error numbers */
}
}
/*
* XXX: The following two function should be replaced with
* one function to create children of directory nodes.
* There is no fundamental difference between a directory
* and a file, thus, the use of flag sets should suffice
*/
int create_file_fs(char *name, uint16_t permission) {
fs_node_t * parent;
char *cwd = (char *)(current_process->wd_name);
char *path = canonicalize_path(cwd, name);
char * parent_path = malloc(strlen(path) + 4);
sprintf(parent_path, "%s/..", path);
char * f_path = path + strlen(path) - 1;
while (f_path > path) {
if (*f_path == '/') {
f_path += 1;
break;
}
f_path--;
}
debug_print(WARNING, "creating file %s within %s (hope these strings are good)", f_path, parent_path);
parent = kopen(parent_path, 0);
free(parent_path);
if (!parent) {
free(path);
return -1;
}
if (parent->create) {
parent->create(parent, f_path, permission);
}
free(path);
free(parent);
return 0;
}
int unlink_fs(char * name) {
fs_node_t * parent;
char *cwd = (char *)(current_process->wd_name);
char *path = canonicalize_path(cwd, name);
char * parent_path = malloc(strlen(path) + 4);
sprintf(parent_path, "%s/..", path);
char * f_path = path + strlen(path) - 1;
while (f_path > path) {
if (*f_path == '/') {
f_path += 1;
break;
}
f_path--;
}
debug_print(WARNING, "unlinking file %s within %s (hope these strings are good)", f_path, parent_path);
parent = kopen(parent_path, 0);
free(parent_path);
if (!parent) {
free(path);
return -1;
}
if (parent->unlink) {
parent->unlink(parent, f_path);
}
free(path);
free(parent);
return 0;
}
int mkdir_fs(char *name, uint16_t permission) {
fs_node_t * parent;
char *cwd = (char *)(current_process->wd_name);
char *path = canonicalize_path(cwd, name);
char * parent_path = malloc(strlen(path) + 4);
sprintf(parent_path, "%s/..", path);
char * f_path = path + strlen(path) - 1;
while (f_path > path) {
if (*f_path == '/') {
f_path += 1;
break;
}
f_path--;
}
debug_print(WARNING, "creating directory %s within %s (hope these strings are good)", f_path, parent_path);
parent = kopen(parent_path, 0);
free(parent_path);
if (!parent) {
free(path);
return -1;
}
if (parent->mkdir) {
parent->mkdir(parent, f_path, permission);
}
free(path);
close_fs(parent);
return 0;
}
fs_node_t *clone_fs(fs_node_t *source) {
if (!source) return NULL;
if (source->refcount >= 0) {
spin_lock(tmp_refcount_lock);
source->refcount++;
spin_unlock(tmp_refcount_lock);
}
return source;
}
/**
* canonicalize_path: Canonicalize a path.
*
* @param cwd Current working directory
* @param input Path to append or canonicalize on
* @returns An absolute path string
*/
char *canonicalize_path(char *cwd, char *input) {
/* This is a stack-based canonicalizer; we use a list as a stack */
list_t *out = list_create();
/*
* If we have a relative path, we need to canonicalize
* the working directory and insert it into the stack.
*/
if (strlen(input) && input[0] != PATH_SEPARATOR) {
/* Make a copy of the working directory */
char *path = malloc((strlen(cwd) + 1) * sizeof(char));
memcpy(path, cwd, strlen(cwd) + 1);
/* Setup tokenizer */
char *pch;
char *save;
pch = strtok_r(path,PATH_SEPARATOR_STRING,&save);
/* Start tokenizing */
while (pch != NULL) {
/* Make copies of the path elements */
char *s = malloc(sizeof(char) * (strlen(pch) + 1));
memcpy(s, pch, strlen(pch) + 1);
/* And push them */
list_insert(out, s);
pch = strtok_r(NULL,PATH_SEPARATOR_STRING,&save);
}
free(path);
}
/* Similarly, we need to push the elements from the new path */
char *path = malloc((strlen(input) + 1) * sizeof(char));
memcpy(path, input, strlen(input) + 1);
/* Initialize the tokenizer... */
char *pch;
char *save;
pch = strtok_r(path,PATH_SEPARATOR_STRING,&save);
/*
* Tokenize the path, this time, taking care to properly
* handle .. and . to represent up (stack pop) and current
* (do nothing)
*/
while (pch != NULL) {
if (!strcmp(pch,PATH_UP)) {
/*
* Path = ..
* Pop the stack to move up a directory
*/
node_t * n = list_pop(out);
if (n) {
free(n->value);
free(n);
}
} else if (!strcmp(pch,PATH_DOT)) {
/*
* Path = .
* Do nothing
*/
} else {
/*
* Regular path, push it
* XXX: Path elements should be checked for existence!
*/
char * s = malloc(sizeof(char) * (strlen(pch) + 1));
memcpy(s, pch, strlen(pch) + 1);
list_insert(out, s);
}
pch = strtok_r(NULL, PATH_SEPARATOR_STRING, &save);
}
free(path);
/* Calculate the size of the path string */
size_t size = 0;
foreach(item, out) {
/* Helpful use of our foreach macro. */
size += strlen(item->value) + 1;
}
/* join() the list */
char *output = malloc(sizeof(char) * (size + 1));
char *output_offset = output;
if (size == 0) {
/*
* If the path is empty, we take this to mean the root
* thus we synthesize a path of "/" to return.
*/
output = realloc(output, sizeof(char) * 2);
output[0] = PATH_SEPARATOR;
output[1] = '\0';
} else {
/* Otherwise, append each element together */
foreach(item, out) {
output_offset[0] = PATH_SEPARATOR;
output_offset++;
memcpy(output_offset, item->value, strlen(item->value) + 1);
output_offset += strlen(item->value);
}
}
/* Clean up the various things we used to get here */
list_destroy(out);
list_free(out);
free(out);
/* And return a working, absolute path */
return output;
}
void vfs_install(void) {
/* Initialize the mountpoint tree */
fs_tree = tree_create();
struct vfs_entry * root = malloc(sizeof(struct vfs_entry));
root->name = strdup("[root]");
root->file = NULL; /* Nothing mounted as root */
tree_set_root(fs_tree, root);
fs_types = hashmap_create(5);
}
int vfs_register(char * name, vfs_mount_callback callback) {
if (hashmap_get(fs_types, name)) return 1;
hashmap_set(fs_types, name, (void *)(uintptr_t)callback);
return 0;
}
int vfs_mount_type(char * type, char * arg, char * mountpoint) {
vfs_mount_callback t = (vfs_mount_callback)(uintptr_t)hashmap_get(fs_types, type);
if (!t) {
debug_print(WARNING, "Unknown filesystem type: %s", type);
return -ENODEV;
}
fs_node_t * n = t(arg, mountpoint);
if (!n) return -EINVAL;
vfs_mount(mountpoint, n);
debug_print(NOTICE, "Mounted %s[%s] to %s: 0x%x", type, arg, mountpoint, n);
debug_print_vfs_tree();
return 0;
}
//volatile uint8_t tmp_vfs_lock = 0;
static spin_lock_t tmp_vfs_lock = { 0 };
/**
* vfs_mount - Mount a file system to the specified path.
*
* For example, if we have an EXT2 filesystem with a root node
* of ext2_root and we want to mount it to /, we would run
* vfs_mount("/", ext2_root); - or, if we have a procfs node,
* we could mount that to /dev/procfs. Individual files can also
* be mounted.
*
* Paths here must be absolute.
*/
void * vfs_mount(char * path, fs_node_t * local_root) {
if (!fs_tree) {
debug_print(ERROR, "VFS hasn't been initialized, you can't mount things yet!");
return NULL;
}
if (!path || path[0] != '/') {
debug_print(ERROR, "Path must be absolute for mountpoint.");
return NULL;
}
spin_lock(tmp_vfs_lock);
local_root->refcount = -1;
tree_node_t * ret_val = NULL;
char * p = strdup(path);
char * i = p;
int path_len = strlen(p);
/* Chop the path up */
while (i < p + path_len) {
if (*i == PATH_SEPARATOR) {
*i = '\0';
}
i++;
}
/* Clean up */
p[path_len] = '\0';
i = p + 1;
/* Root */
tree_node_t * root_node = fs_tree->root;
if (*i == '\0') {
/* Special case, we're trying to set the root node */
struct vfs_entry * root = (struct vfs_entry *)root_node->value;
if (root->file) {
debug_print(WARNING, "Path %s already mounted, unmount before trying to mount something else.", path);
}
root->file = local_root;
/* We also keep a legacy shortcut around for that */
fs_root = local_root;
ret_val = root_node;
} else {
tree_node_t * node = root_node;
char * at = i;
while (1) {
if (at >= p + path_len) {
break;
}
int found = 0;
debug_print(NOTICE, "Searching for %s", at);
foreach(child, node->children) {
tree_node_t * tchild = (tree_node_t *)child->value;
struct vfs_entry * ent = (struct vfs_entry *)tchild->value;
if (!strcmp(ent->name, at)) {
found = 1;
node = tchild;
ret_val = node;
break;
}
}
if (!found) {
debug_print(NOTICE, "Did not find %s, making it.", at);
struct vfs_entry * ent = malloc(sizeof(struct vfs_entry));
ent->name = strdup(at);
ent->file = NULL;
node = tree_node_insert_child(fs_tree, node, ent);
}
at = at + strlen(at) + 1;
}
struct vfs_entry * ent = (struct vfs_entry *)node->value;
if (ent->file) {
debug_print(WARNING, "Path %s already mounted, unmount before trying to mount something else.", path);
}
ent->file = local_root;
ret_val = node;
}
free(p);
spin_unlock(tmp_vfs_lock);
return ret_val;
}
void map_vfs_directory(char * c) {
fs_node_t * f = vfs_mapper();
struct vfs_entry * e = vfs_mount(c, f);
if (!strcmp(c, "/")) {
f->device = fs_tree->root;
} else {
f->device = e;
}
}
void debug_print_vfs_tree_node(tree_node_t * node, size_t height) {
/* End recursion on a blank entry */
if (!node) return;
char * tmp = malloc(512);
memset(tmp, 0, 512);
char * c = tmp;
/* Indent output */
for (uint32_t i = 0; i < height; ++i) {
c += sprintf(c, " ");
}
/* Get the current process */
struct vfs_entry * fnode = (struct vfs_entry *)node->value;
/* Print the process name */
if (fnode->file) {
c += sprintf(c, "%s → 0x%x (%s)", fnode->name, fnode->file, fnode->file->name);
} else {
c += sprintf(c, "%s → (empty)", fnode->name);
}
/* Linefeed */
debug_print(NOTICE, "%s", tmp);
free(tmp);
foreach(child, node->children) {
/* Recursively print the children */
debug_print_vfs_tree_node(child->value, height + 1);
}
}
void debug_print_vfs_tree(void) {
debug_print_vfs_tree_node(fs_tree->root, 0);
}
/**
* get_mount_point
*
*/
fs_node_t *get_mount_point(char * path, unsigned int path_depth, char **outpath, unsigned int * outdepth) {
size_t depth;
for (depth = 0; depth <= path_depth; ++depth) {
path += strlen(path) + 1;
}
/* Last available node */
fs_node_t * last = fs_root;
tree_node_t * node = fs_tree->root;
char * at = *outpath;
int _depth = 1;
int _tree_depth = 0;
while (1) {
if (at >= path) {
break;
}
int found = 0;
debug_print(INFO, "Searching for %s", at);
foreach(child, node->children) {
tree_node_t * tchild = (tree_node_t *)child->value;
struct vfs_entry * ent = (struct vfs_entry *)tchild->value;
if (!strcmp(ent->name, at)) {
found = 1;
node = tchild;
at = at + strlen(at) + 1;
if (ent->file) {
_tree_depth = _depth;
last = ent->file;
*outpath = at;
}
break;
}
}
if (!found) {
break;
}
_depth++;
}
*outdepth = _tree_depth;
if (last) {
fs_node_t * last_clone = malloc(sizeof(fs_node_t));
memcpy(last_clone, last, sizeof(fs_node_t));
return last_clone;
}
return last;
}
/**
* kopen: Open a file by name.
*
* Explore the file system tree to find the appropriate node for
* for a given path. The path can be relative to the working directory
* and will be canonicalized by the kernel.
*
* @param filename Filename to open
* @param flags Flag bits for read/write mode.
* @returns A file system node element that the caller can free.
*/
fs_node_t *kopen(char *filename, uint32_t flags) {
/* Simple sanity checks that we actually have a file system */
if (!filename) {
return NULL;
}
debug_print(INFO, "kopen(%s)", filename);
/* Reference the current working directory */
char *cwd = (char *)(current_process->wd_name);
/* Canonicalize the (potentially relative) path... */
char *path = canonicalize_path(cwd, filename);
/* And store the length once to save recalculations */
size_t path_len = strlen(path);
/* If strlen(path) == 1, then path = "/"; return root */
if (path_len == 1) {
/* Clone the root file system node */
fs_node_t *root_clone = malloc(sizeof(fs_node_t));
memcpy(root_clone, fs_root, sizeof(fs_node_t));
/* Free the path */
free(path);
open_fs(root_clone, flags);
/* And return the clone */
return root_clone;
}
/* Otherwise, we need to break the path up and start searching */
char *path_offset = path;
uint32_t path_depth = 0;
while (path_offset < path + path_len) {
/* Find each PATH_SEPARATOR */
if (*path_offset == PATH_SEPARATOR) {
*path_offset = '\0';
path_depth++;
}
path_offset++;
}
/* Clean up */
path[path_len] = '\0';
path_offset = path + 1;
/*
* At this point, the path is tokenized and path_offset points
* to the first token (directory) and path_depth is the number
* of directories in the path
*/
/*
* Dig through the (real) tree to find the file
*/
unsigned int depth = 0;
/* Find the mountpoint for this file */
fs_node_t *node_ptr = get_mount_point(path, path_depth, &path_offset, &depth);
if (!node_ptr) return NULL;
if (path_offset >= path+path_len) {
free(path);
open_fs(node_ptr, flags);
return node_ptr;
}
fs_node_t *node_next = NULL;
for (; depth < path_depth; ++depth) {
/* Search the active directory for the requested directory */
debug_print(INFO, "... Searching for %s", path_offset);
node_next = finddir_fs(node_ptr, path_offset);
free(node_ptr);
node_ptr = node_next;
if (!node_ptr) {
/* We failed to find the requested directory */
free((void *)path);
return NULL;
} else if (depth == path_depth - 1) {
/* We found the file and are done, open the node */
open_fs(node_ptr, flags);
free((void *)path);
return node_ptr;
}
/* We are still searching... */
path_offset += strlen(path_offset) + 1;
}
debug_print(INFO, "- Not found.");
/* We failed to find the requested file, but our loop terminated. */
free((void *)path);
return NULL;
}