/* vim: shiftwidth=4 tabstop=4 noexpandtab * * Virtual File System * */ #include #include #include #include #include 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) */ /** * 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->read) { uint32_t ret = node->read(node, offset, size, buffer); return ret; } else { return 0; } } /** * 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->write) { uint32_t ret = node->write(node, offset, size, buffer); return ret; } else { return 0; } } /** * 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->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->close) { node->close(node); } } /** * 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->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->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->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); free(parent); return 0; } fs_node_t *clone_fs(fs_node_t *source) { if (!source) { return NULL; } fs_node_t *n = malloc(sizeof(fs_node_t)); memcpy(n, source, sizeof(fs_node_t)); return n; } /** * 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; } struct vfs_entry { char * name; fs_node_t * file; /* Or null */ }; 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); } /** * 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. */ int 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 1; } if (!path || path[0] != '/') { debug_print(ERROR, "Path must be absolute for mountpoint."); return 2; } int ret_val = 0; 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); ret_val = 3; goto _vfs_cleanup; } root->file = local_root; /* We also keep a legacy shortcut around for that */ fs_root = local_root; } else { tree_node_t * node = root_node; char * at = i; while (1) { if (at >= p + path_len) { 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; break; } } if (!found) { debug_print(INFO, "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); ret_val = 3; goto _vfs_cleanup; } ent->file = local_root; } _vfs_cleanup: free(p); return ret_val; } void debug_print_vfs_tree_node(tree_node_t * node, size_t height) { /* End recursion on a blank entry */ if (!node) return; /* Indent output */ for (uint32_t i = 0; i < height; ++i) { kprintf(" "); } /* Get the current process */ struct vfs_entry * fnode = (struct vfs_entry *)node->value; /* Print the process name */ if (fnode->file) { kprintf("%s → 0x%x (%s)", fnode->name, fnode->file, fnode->file->name); } else { kprintf("%s → (empty)", fnode->name); } /* Linefeed */ kprintf("\n"); 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 (!fs_root || !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); /* 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 */ uint32_t depth = 0; fs_node_t *node_ptr = malloc(sizeof(fs_node_t)); /* Find the mountpoint for this file */ fs_node_t *mount_point = get_mount_point(path, path_depth, &path_offset, &depth); if (path_offset >= path+path_len) { free(path); return mount_point; } /* Set the active directory to the mountpoint */ memcpy(node_ptr, mount_point, sizeof(fs_node_t)); 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; }