#include <stdio.h> #include <stdbool.h> #include <string.h> #include <dprintf.h> #include <fcntl.h> #include "fs.h" #include "cache.h" /* * Convert a relative pathname to an absolute pathname * In the future this might also resolve symlinks... */ void pm_realpath(com32sys_t *regs) { const char *src = MK_PTR(regs->ds, regs->esi.w[0]); char *dst = MK_PTR(regs->es, regs->edi.w[0]); realpath(dst, src, FILENAME_MAX); } static size_t copy_string(char *buf, size_t ix, size_t bufsize, const char *src) { char c; while ((c = *src++)) { if (ix+1 < bufsize) buf[ix] = c; ix++; } if (ix < bufsize) buf[ix] = '\0'; return ix; } static size_t generic_inode_to_path(struct inode *inode, char *dst, size_t bufsize) { size_t s = 0; dprintf("inode %p name %s\n", inode, inode->name); if (inode->parent) { if (!inode->name) /* Only the root should have no name */ return -1; s = generic_inode_to_path(inode->parent, dst, bufsize); if (s == (size_t)-1) return s; /* Error! */ s = copy_string(dst, s, bufsize, "/"); s = copy_string(dst, s, bufsize, inode->name); } return s; } __export size_t realpath(char *dst, const char *src, size_t bufsize) { int rv; struct file *file; size_t s; dprintf("realpath: input: %s\n", src); if (this_fs->fs_ops->realpath) { s = this_fs->fs_ops->realpath(this_fs, dst, src, bufsize); } else { rv = searchdir(src, O_RDONLY); if (rv < 0) { dprintf("realpath: searchpath failure\n"); return -1; } file = handle_to_file(rv); s = generic_inode_to_path(file->inode, dst, bufsize); if (s == 0) s = copy_string(dst, 0, bufsize, "/"); _close_file(file); } dprintf("realpath: output: %s\n", dst); return s; } __export int chdir(const char *src) { int rv; struct file *file; char cwd_buf[CURRENTDIR_MAX]; size_t s; dprintf("chdir: from %s (inode %p) add %s\n", this_fs->cwd_name, this_fs->cwd, src); if (this_fs->fs_ops->chdir) return this_fs->fs_ops->chdir(this_fs, src); /* Otherwise it is a "conventional filesystem" */ rv = searchdir(src, O_RDONLY|O_DIRECTORY); if (rv < 0) return rv; file = handle_to_file(rv); if (file->inode->mode != DT_DIR) { _close_file(file); return -1; } put_inode(this_fs->cwd); this_fs->cwd = get_inode(file->inode); _close_file(file); /* Save the current working directory */ s = generic_inode_to_path(this_fs->cwd, cwd_buf, CURRENTDIR_MAX-1); /* Make sure the cwd_name ends in a slash, it's supposed to be a prefix */ if (s < 1 || cwd_buf[s-1] != '/') cwd_buf[s++] = '/'; if (s >= CURRENTDIR_MAX) s = CURRENTDIR_MAX - 1; cwd_buf[s++] = '\0'; memcpy(this_fs->cwd_name, cwd_buf, s); dprintf("chdir: final %s (inode %p)\n", this_fs->cwd_name, this_fs->cwd); return 0; }