/* $Id: dirops.c 77492 2019-02-28 01:55:50Z vboxsync $ */ /** @file * vboxsf - VBox Linux Shared Folders VFS, directory inode and file operations. */ /* * Copyright (C) 2006-2019 Oracle Corporation * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation * files (the "Software"), to deal in the Software without * restriction, including without limitation the rights to use, * copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following * conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR * OTHER DEALINGS IN THE SOFTWARE. */ #include "vfsmod.h" #include /** * Reads or re-reads a directory. * * @note As suggested a couple of other places, we should probably stop * reading in the whole directory on open. */ static int sf_dir_open_worker(struct sf_glob_info *sf_g, struct sf_dir_info *sf_d, struct sf_inode_info *sf_i, const char *pszCaller) { int rc; int err; union SfDirOpenCloseReq { VBOXSFCREATEREQ Create; VBOXSFCLOSEREQ Close; } *pReq; SHFLCREATEPARMS *pCreateParms; pReq = (union SfDirOpenCloseReq *)VbglR0PhysHeapAlloc(RT_UOFFSETOF(VBOXSFCREATEREQ, StrPath.String) + sf_i->path->u16Size); if (pReq) { memcpy(&pReq->Create.StrPath, sf_i->path, SHFLSTRING_HEADER_SIZE + sf_i->path->u16Size); pCreateParms = &pReq->Create.CreateParms; } else { LogRelMaxFunc(64, ("failed to allocate %zu bytes for '%s' [caller: %s]\n", RT_UOFFSETOF(VBOXSFCREATEREQ, StrPath.String) + sf_i->path->u16Size, sf_i->path->String.ach, pszCaller)); return -ENOMEM; } RT_ZERO(*pCreateParms); pCreateParms->Handle = SHFL_HANDLE_NIL; pCreateParms->CreateFlags = SHFL_CF_DIRECTORY | SHFL_CF_ACT_OPEN_IF_EXISTS | SHFL_CF_ACT_FAIL_IF_NEW | SHFL_CF_ACCESS_READ; LogFunc(("calling VbglR0SfHostReqCreate on folder %s, flags %#x [caller: %s]\n", sf_i->path->String.utf8, pCreateParms->CreateFlags, pszCaller)); rc = VbglR0SfHostReqCreate(sf_g->map.root, &pReq->Create); if (RT_SUCCESS(rc)) { if (pCreateParms->Result == SHFL_FILE_EXISTS) { /** @todo We could refresh the inode information here since SHFL_FN_CREATE * returns updated object information. */ /** @todo Touch the dentries from here to the mount root since a successful * open means that the whole path is valid. I believe CIFS does this. */ /** @todo Reading all entries upon opening the directory doesn't seem like * a good idea. */ sf_dir_info_empty(sf_d); err = sf_dir_read_all(sf_g, sf_i, sf_d, pCreateParms->Handle); } else err = -ENOENT; AssertCompile(RTASSERT_OFFSET_OF(VBOXSFCREATEREQ, CreateParms.Handle) > sizeof(VBOXSFCLOSEREQ)); /* no aliasing issues */ if (pCreateParms->Handle != SHFL_HANDLE_NIL) { rc = VbglR0SfHostReqClose(sf_g->map.root, &pReq->Close, pCreateParms->Handle); if (RT_FAILURE(rc)) LogFunc(("VbglR0SfHostReqCloseSimple(%s) after err=%d failed rc=%Rrc caller=%s\n", sf_i->path->String.utf8, err, rc, pszCaller)); } } else err = -EPERM; VbglR0PhysHeapFree(pReq); return err; } /** * Open a directory. Read the complete content into a buffer. * * @param inode inode * @param file file * @returns 0 on success, Linux error code otherwise */ static int sf_dir_open(struct inode *inode, struct file *file) { int err; struct sf_glob_info *sf_g = GET_GLOB_INFO(inode->i_sb); struct sf_dir_info *sf_d; struct sf_inode_info *sf_i = GET_INODE_INFO(inode); TRACE(); BUG_ON(!sf_g); BUG_ON(!sf_i); if (file->private_data) { LogFunc(("called on already opened directory '%s'!\n", sf_i->path->String.ach)); return 0; } sf_d = sf_dir_info_alloc(); if (!sf_d) { LogRelFunc(("could not allocate directory info for '%s'\n", sf_i->path->String.ach)); return -ENOMEM; } err = sf_dir_open_worker(sf_g, sf_d, sf_i, "sf_dir_open"); if (!err) file->private_data = sf_d; else sf_dir_info_free(sf_d); return err; } /** * This is called when reference count of [file] goes to zero. Notify * the host that it can free whatever is associated with this directory * and deallocate our own internal buffers * * @param inode inode * @param file file * returns 0 on success, Linux error code otherwise */ static int sf_dir_release(struct inode *inode, struct file *file) { TRACE(); if (file->private_data) sf_dir_info_free(file->private_data); return 0; } /** * Translate RTFMODE into DT_xxx (in conjunction to rtDirType()) * @param fMode file mode * returns d_type */ static int sf_get_d_type(RTFMODE fMode) { int d_type; switch (fMode & RTFS_TYPE_MASK) { case RTFS_TYPE_FIFO: d_type = DT_FIFO; break; case RTFS_TYPE_DEV_CHAR: d_type = DT_CHR; break; case RTFS_TYPE_DIRECTORY: d_type = DT_DIR; break; case RTFS_TYPE_DEV_BLOCK: d_type = DT_BLK; break; case RTFS_TYPE_FILE: d_type = DT_REG; break; case RTFS_TYPE_SYMLINK: d_type = DT_LNK; break; case RTFS_TYPE_SOCKET: d_type = DT_SOCK; break; case RTFS_TYPE_WHITEOUT: d_type = DT_WHT; break; default: d_type = DT_UNKNOWN; break; } return d_type; } /** * Extract element ([dir]->f_pos) from the directory [dir] into [d_name]. * * @returns 0 for success, 1 for end reached, Linux error code otherwise. */ static int sf_getdent(struct file *dir, char d_name[NAME_MAX], int *d_type) { loff_t cur; struct sf_glob_info *sf_g; struct sf_dir_info *sf_d; struct sf_inode_info *sf_i; struct inode *inode; struct list_head *pos, *list; TRACE(); inode = GET_F_DENTRY(dir)->d_inode; sf_i = GET_INODE_INFO(inode); sf_g = GET_GLOB_INFO(inode->i_sb); sf_d = dir->private_data; BUG_ON(!sf_g); BUG_ON(!sf_d); BUG_ON(!sf_i); if (sf_i->force_reread) { int err = sf_dir_open_worker(sf_g, sf_d, sf_i, "sf_getdent"); if (!err) { sf_i->force_reread = 0; } else { if (err == -ENOENT) { sf_dir_info_free(sf_d); dir->private_data = NULL; } return err; } } cur = 0; list = &sf_d->info_list; list_for_each(pos, list) { struct sf_dir_buf *b; SHFLDIRINFO *info; loff_t i; b = list_entry(pos, struct sf_dir_buf, head); if (dir->f_pos >= cur + b->cEntries) { cur += b->cEntries; continue; } for (i = 0, info = b->buf; i < dir->f_pos - cur; ++i) { size_t size; size = offsetof(SHFLDIRINFO, name.String) + info->name.u16Size; info = (SHFLDIRINFO *)((uintptr_t)info + size); } *d_type = sf_get_d_type(info->Info.Attr.fMode); return sf_nlscpy(sf_g, d_name, NAME_MAX, info->name.String.utf8, info->name.u16Length); } return 1; } /** * This is called when vfs wants to populate internal buffers with * directory [dir]s contents. [opaque] is an argument to the * [filldir]. [filldir] magically modifies it's argument - [opaque] * and takes following additional arguments (which i in turn get from * the host via sf_getdent): * * name : name of the entry (i must also supply it's length huh?) * type : type of the entry (FILE | DIR | etc) (i ellect to use DT_UNKNOWN) * pos : position/index of the entry * ino : inode number of the entry (i fake those) * * [dir] contains: * f_pos : cursor into the directory listing * private_data : mean of communication with the host side * * Extract elements from the directory listing (incrementing f_pos * along the way) and feed them to [filldir] until: * * a. there are no more entries (i.e. sf_getdent set done to 1) * b. failure to compute fake inode number * c. filldir returns an error (see comment on that) */ #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) static int sf_dir_iterate(struct file *dir, struct dir_context *ctx) #else static int sf_dir_read(struct file *dir, void *opaque, filldir_t filldir) #endif { TRACE(); for (;;) { int err; ino_t fake_ino; loff_t sanity; char d_name[NAME_MAX]; int d_type = DT_UNKNOWN; err = sf_getdent(dir, d_name, &d_type); switch (err) { case 1: return 0; case 0: break; case -1: default: /* skip erroneous entry and proceed */ LogFunc(("sf_getdent error %d\n", err)); dir->f_pos += 1; #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) ctx->pos += 1; #endif continue; } /* d_name now contains a valid entry name */ #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) sanity = ctx->pos + 0xbeef; #else sanity = dir->f_pos + 0xbeef; #endif fake_ino = sanity; if (sanity - fake_ino) { LogRelFunc(("can not compute ino\n")); return -EINVAL; } #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) if (!dir_emit(ctx, d_name, strlen(d_name), fake_ino, d_type)) { LogFunc(("dir_emit failed\n")); return 0; } #else err = filldir(opaque, d_name, strlen(d_name), dir->f_pos, fake_ino, d_type); if (err) { LogFunc(("filldir returned error %d\n", err)); /* Rely on the fact that filldir returns error only when it runs out of space in opaque */ return 0; } #endif dir->f_pos += 1; #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) ctx->pos += 1; #endif } BUG(); } struct file_operations sf_dir_fops = { .open = sf_dir_open, #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) .iterate = sf_dir_iterate, #else .readdir = sf_dir_read, #endif .release = sf_dir_release, .read = generic_read_dir #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 37) , .llseek = generic_file_llseek #endif }; /* iops */ /** * This is called when vfs failed to locate dentry in the cache. The * job of this function is to allocate inode and link it to dentry. * [dentry] contains the name to be looked in the [parent] directory. * Failure to locate the name is not a "hard" error, in this case NULL * inode is added to [dentry] and vfs should proceed trying to create * the entry via other means. NULL(or "positive" pointer) ought to be * returned in case of success and "negative" pointer on error */ static struct dentry *sf_lookup(struct inode *parent, struct dentry *dentry #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 6, 0) , unsigned int flags #elif LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0) , struct nameidata *nd #endif ) { int err; struct sf_inode_info *sf_i, *sf_new_i; struct sf_glob_info *sf_g; SHFLSTRING *path; struct inode *inode; ino_t ino; SHFLFSOBJINFO fsinfo; TRACE(); sf_g = GET_GLOB_INFO(parent->i_sb); sf_i = GET_INODE_INFO(parent); BUG_ON(!sf_g); BUG_ON(!sf_i); err = sf_path_from_dentry(__func__, sf_g, sf_i, dentry, &path); if (err) goto fail0; /** @todo call host directly and avoid unnecessary copying here. */ err = sf_stat(__func__, sf_g, path, &fsinfo, 1); if (err) { if (err == -ENOENT) { /* -ENOENT: add NULL inode to dentry so it later can be created via call to create/mkdir/open */ kfree(path); inode = NULL; } else goto fail1; } else { sf_new_i = kmalloc(sizeof(*sf_new_i), GFP_KERNEL); if (!sf_new_i) { LogRelFunc(("could not allocate memory for new inode info\n")); err = -ENOMEM; goto fail1; } sf_new_i->handle = SHFL_HANDLE_NIL; sf_new_i->force_reread = 0; RTListInit(&sf_new_i->HandleList); #ifdef VBOX_STRICT sf_new_i->u32Magic = SF_INODE_INFO_MAGIC; #endif ino = iunique(parent->i_sb, 1); #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 4, 25) inode = iget_locked(parent->i_sb, ino); #else inode = iget(parent->i_sb, ino); #endif if (!inode) { LogFunc(("iget failed\n")); err = -ENOMEM; /* XXX: ??? */ goto fail2; } sf_new_i->path = path; SET_INODE_INFO(inode, sf_new_i); sf_init_inode(inode, sf_new_i, &fsinfo, sf_g); #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 4, 25) unlock_new_inode(inode); #endif sf_dentry_chain_increase_parent_ttl(dentry); } //sf_i->force_restat = 0; /** @todo r=bird: This looks like confusion. */ sf_dentry_set_update_jiffies(dentry, jiffies); #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 38) Assert(dentry->d_op == &sf_dentry_ops); /* (taken from the superblock) */ #else dentry->d_op = &sf_dentry_ops; #endif d_add(dentry, inode); return NULL; fail2: kfree(sf_new_i); fail1: kfree(path); fail0: return ERR_PTR(err); } /** * This should allocate memory for sf_inode_info, compute a unique inode * number, get an inode from vfs, initialize inode info, instantiate * dentry. * * @param parent inode entry of the directory * @param dentry directory cache entry * @param path path name * @param info file information * @param handle handle * @returns 0 on success, Linux error code otherwise */ static int sf_instantiate(struct inode *parent, struct dentry *dentry, SHFLSTRING * path, PSHFLFSOBJINFO info, SHFLHANDLE handle) { int err; ino_t ino; struct inode *inode; struct sf_inode_info *sf_new_i; struct sf_glob_info *sf_g = GET_GLOB_INFO(parent->i_sb); TRACE(); BUG_ON(!sf_g); sf_new_i = kmalloc(sizeof(*sf_new_i), GFP_KERNEL); if (!sf_new_i) { LogRelFunc(("could not allocate inode info.\n")); err = -ENOMEM; goto fail0; } ino = iunique(parent->i_sb, 1); #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 4, 25) inode = iget_locked(parent->i_sb, ino); #else inode = iget(parent->i_sb, ino); #endif if (!inode) { LogFunc(("iget failed\n")); err = -ENOMEM; goto fail1; } sf_init_inode(inode, sf_new_i, info, sf_g); sf_new_i->path = path; RTListInit(&sf_new_i->HandleList); sf_new_i->force_restat = 1; sf_new_i->force_reread = 0; sf_new_i->ts_up_to_date = jiffies - INT_MAX / 2; #ifdef VBOX_STRICT sf_new_i->u32Magic = SF_INODE_INFO_MAGIC; #endif SET_INODE_INFO(inode, sf_new_i); d_instantiate(dentry, inode); #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 4, 25) unlock_new_inode(inode); #endif /* Store this handle if we leave the handle open. */ sf_new_i->handle = handle; return 0; fail1: kfree(sf_new_i); fail0: return err; } /** * Create a new regular file / directory. * * @param parent inode of the directory * @param dentry directory cache entry * @param mode file mode * @param fDirectory true if directory, false otherwise * @returns 0 on success, Linux error code otherwise */ static int sf_create_aux(struct inode *parent, struct dentry *dentry, umode_t mode, int fDirectory) { int rc, err; struct sf_inode_info *sf_i = GET_INODE_INFO(parent); struct sf_glob_info *sf_g = GET_GLOB_INFO(parent->i_sb); SHFLSTRING *path; union CreateAuxReq { VBOXSFCREATEREQ Create; VBOXSFCLOSEREQ Close; } *pReq; SHFLCREATEPARMS *pCreateParms; TRACE(); BUG_ON(!sf_i); BUG_ON(!sf_g); err = sf_path_from_dentry(__func__, sf_g, sf_i, dentry, &path); if (err) goto fail0; /** @todo combine with sf_path_from_dentry? */ pReq = (union CreateAuxReq *)VbglR0PhysHeapAlloc(RT_UOFFSETOF(VBOXSFCREATEREQ, StrPath.String) + path->u16Size); if (pReq) { memcpy(&pReq->Create.StrPath, path, SHFLSTRING_HEADER_SIZE + path->u16Size); pCreateParms = &pReq->Create.CreateParms; } else { err = -ENOMEM; goto fail1; } RT_ZERO(*pCreateParms); pCreateParms->Handle = SHFL_HANDLE_NIL; pCreateParms->CreateFlags = SHFL_CF_ACT_CREATE_IF_NEW | SHFL_CF_ACT_FAIL_IF_EXISTS | SHFL_CF_ACCESS_READWRITE | (fDirectory ? SHFL_CF_DIRECTORY : 0); pCreateParms->Info.Attr.fMode = (fDirectory ? RTFS_TYPE_DIRECTORY : RTFS_TYPE_FILE) | (mode & S_IRWXUGO); pCreateParms->Info.Attr.enmAdditional = RTFSOBJATTRADD_NOTHING; LogFunc(("calling VbglR0SfHostReqCreate, folder %s, flags %#x\n", path->String.ach, pCreateParms->CreateFlags)); rc = VbglR0SfHostReqCreate(sf_g->map.root, &pReq->Create); if (RT_FAILURE(rc)) { if (rc == VERR_WRITE_PROTECT) { err = -EROFS; goto fail2; } err = -EPROTO; LogFunc(("(%d): SHFL_FN_CREATE(%s) failed rc=%Rrc\n", fDirectory, sf_i->path->String.utf8, rc)); goto fail2; } if (pCreateParms->Result != SHFL_FILE_CREATED) { err = -EPERM; LogFunc(("(%d): could not create file %s result=%d\n", fDirectory, sf_i->path->String.utf8, pCreateParms->Result)); goto fail2; } sf_dentry_chain_increase_parent_ttl(dentry); err = sf_instantiate(parent, dentry, path, &pCreateParms->Info, fDirectory ? SHFL_HANDLE_NIL : pCreateParms->Handle); if (err) { LogFunc(("(%d): could not instantiate dentry for %s err=%d\n", fDirectory, sf_i->path->String.utf8, err)); goto fail3; } /* * Don't close this handle right now. We assume that the same file is * opened with sf_reg_open() and later closed with sf_reg_close(). Save * the handle in between. Does not apply to directories. True? */ if (fDirectory) { AssertCompile(RTASSERT_OFFSET_OF(VBOXSFCREATEREQ, CreateParms.Handle) > sizeof(VBOXSFCLOSEREQ)); /* no aliasing issues */ rc = VbglR0SfHostReqClose(sf_g->map.root, &pReq->Close, pCreateParms->Handle); if (RT_FAILURE(rc)) LogFunc(("(%d): VbglR0SfHostReqClose failed rc=%Rrc\n", fDirectory, rc)); } sf_i->force_restat = 1; /**< @todo r=bird: Why?!? */ VbglR0PhysHeapFree(pReq); return 0; fail3: rc = VbglR0SfHostReqClose(sf_g->map.root, &pReq->Close, pCreateParms->Handle); if (RT_FAILURE(rc)) LogFunc(("(%d): VbglR0SfHostReqCloseSimple failed rc=%Rrc\n", fDirectory, rc)); fail2: VbglR0PhysHeapFree(pReq); fail1: kfree(path); fail0: return err; } /** * Create a new regular file. * * @param parent inode of the directory * @param dentry directory cache entry * @param mode file mode * @param excl Possible O_EXCL... * @returns 0 on success, Linux error code otherwise */ #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 6, 0) || defined(DOXYGEN_RUNNING) static int sf_create(struct inode *parent, struct dentry *dentry, umode_t mode, bool excl) #elif LINUX_VERSION_CODE >= KERNEL_VERSION(3, 3, 0) static int sf_create(struct inode *parent, struct dentry *dentry, umode_t mode, struct nameidata *nd) #elif LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0) static int sf_create(struct inode *parent, struct dentry *dentry, int mode, struct nameidata *nd) #else static int sf_create(struct inode *parent, struct dentry *dentry, int mode) #endif { /** @todo @a nd (struct nameidata) contains intent with partial open flags for * 2.6.0-3.5.999. In 3.6.0 atomic_open was introduced and stuff * changed (investigate)... */ TRACE(); return sf_create_aux(parent, dentry, mode, 0 /*fDirectory*/); } /** * Create a new directory. * * @param parent inode of the directory * @param dentry directory cache entry * @param mode file mode * @returns 0 on success, Linux error code otherwise */ #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 3, 0) static int sf_mkdir(struct inode *parent, struct dentry *dentry, umode_t mode) #else static int sf_mkdir(struct inode *parent, struct dentry *dentry, int mode) #endif { TRACE(); return sf_create_aux(parent, dentry, mode, 1 /*fDirectory*/); } /** * Remove a regular file / directory. * * @param parent inode of the directory * @param dentry directory cache entry * @param fDirectory true if directory, false otherwise * @returns 0 on success, Linux error code otherwise */ static int sf_unlink_aux(struct inode *parent, struct dentry *dentry, int fDirectory) { int rc, err; struct sf_glob_info *sf_g = GET_GLOB_INFO(parent->i_sb); struct sf_inode_info *sf_parent_i = GET_INODE_INFO(parent); SHFLSTRING *path; TRACE(); BUG_ON(!sf_g); err = sf_path_from_dentry(__func__, sf_g, sf_parent_i, dentry, &path); if (!err) { VBOXSFREMOVEREQ *pReq = (VBOXSFREMOVEREQ *)VbglR0PhysHeapAlloc(RT_UOFFSETOF(VBOXSFREMOVEREQ, StrPath.String) + path->u16Size); if (pReq) { memcpy(&pReq->StrPath, path, SHFLSTRING_HEADER_SIZE + path->u16Size); uint32_t fFlags = fDirectory ? SHFL_REMOVE_DIR : SHFL_REMOVE_FILE; if (dentry->d_inode && ((dentry->d_inode->i_mode & S_IFLNK) == S_IFLNK)) fFlags |= SHFL_REMOVE_SYMLINK; rc = VbglR0SfHostReqRemove(sf_g->map.root, pReq, fFlags); if (dentry->d_inode) { struct sf_inode_info *sf_i = GET_INODE_INFO(dentry->d_inode); sf_i->force_restat = 1; sf_i->force_reread = 1; } if (RT_SUCCESS(rc)) { /* directory access/change time changed */ sf_parent_i->force_restat = 1; /* directory content changed */ sf_parent_i->force_reread = 1; err = 0; } else if (rc == VERR_FILE_NOT_FOUND || rc == VERR_PATH_NOT_FOUND) { LogFunc(("(%d): VbglR0SfRemove(%s) failed rc=%Rrc; calling d_drop on %p\n", fDirectory, path->String.utf8, rc, dentry)); d_drop(dentry); } else { LogFunc(("(%d): VbglR0SfRemove(%s) failed rc=%Rrc\n", fDirectory, path->String.utf8, rc)); err = -RTErrConvertToErrno(rc); } VbglR0PhysHeapFree(pReq); } else err = -ENOMEM; kfree(path); } return err; } /** * Remove a regular file. * * @param parent inode of the directory * @param dentry directory cache entry * @returns 0 on success, Linux error code otherwise */ static int sf_unlink(struct inode *parent, struct dentry *dentry) { TRACE(); return sf_unlink_aux(parent, dentry, 0 /*fDirectory*/); } /** * Remove a directory. * * @param parent inode of the directory * @param dentry directory cache entry * @returns 0 on success, Linux error code otherwise */ static int sf_rmdir(struct inode *parent, struct dentry *dentry) { TRACE(); return sf_unlink_aux(parent, dentry, 1 /*fDirectory*/); } /** * Rename a regular file / directory. * * @param old_parent inode of the old parent directory * @param old_dentry old directory cache entry * @param new_parent inode of the new parent directory * @param new_dentry new directory cache entry * @param flags flags * @returns 0 on success, Linux error code otherwise */ static int sf_rename(struct inode *old_parent, struct dentry *old_dentry, struct inode *new_parent, struct dentry *new_dentry #if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 9, 0) , unsigned flags #endif ) { int err = 0, rc = VINF_SUCCESS; struct sf_glob_info *sf_g = GET_GLOB_INFO(old_parent->i_sb); TRACE(); #if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 9, 0) if (flags) { LogFunc(("rename with flags=%x\n", flags)); return -EINVAL; } #endif if (sf_g != GET_GLOB_INFO(new_parent->i_sb)) { LogFunc(("rename with different roots\n")); err = -EINVAL; } else { struct sf_inode_info *sf_old_i = GET_INODE_INFO(old_parent); struct sf_inode_info *sf_new_i = GET_INODE_INFO(new_parent); /* As we save the relative path inside the inode structure, we need to change this if the rename is successful. */ struct sf_inode_info *sf_file_i = GET_INODE_INFO(old_dentry->d_inode); SHFLSTRING *old_path; SHFLSTRING *new_path; BUG_ON(!sf_old_i); BUG_ON(!sf_new_i); BUG_ON(!sf_file_i); old_path = sf_file_i->path; err = sf_path_from_dentry(__func__, sf_g, sf_new_i, new_dentry, &new_path); if (err) LogFunc(("failed to create new path\n")); else { VBOXSFRENAMEWITHSRCBUFREQ *pReq; pReq = (VBOXSFRENAMEWITHSRCBUFREQ *)VbglR0PhysHeapAlloc(RT_UOFFSETOF(VBOXSFRENAMEWITHSRCBUFREQ, StrDstPath.String) + new_path->u16Size); if (pReq) { memcpy(&pReq->StrDstPath, new_path, SHFLSTRING_HEADER_SIZE + new_path->u16Size); rc = VbglR0SfHostReqRenameWithSrcContig(sf_g->map.root, pReq, old_path, virt_to_phys(old_path), old_dentry->d_inode->i_mode & S_IFDIR ? SHFL_RENAME_DIR : SHFL_RENAME_FILE | SHFL_RENAME_REPLACE_IF_EXISTS); VbglR0PhysHeapFree(pReq); } else rc = VERR_NO_MEMORY; if (RT_SUCCESS(rc)) { kfree(old_path); sf_new_i->force_restat = 1; sf_old_i->force_restat = 1; /* XXX: needed? */ /* Set the new relative path in the inode. */ sf_file_i->path = new_path; } else { LogFunc(("VbglR0SfRename failed rc=%Rrc\n", rc)); err = -RTErrConvertToErrno(rc); kfree(new_path); } } } return err; } #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0) static int sf_symlink(struct inode *parent, struct dentry *dentry, const char *symname) { int err; int rc; struct sf_inode_info *sf_i; struct sf_glob_info *sf_g; SHFLSTRING *path, *ssymname; SHFLFSOBJINFO info; int symname_len = strlen(symname) + 1; TRACE(); sf_g = GET_GLOB_INFO(parent->i_sb); sf_i = GET_INODE_INFO(parent); BUG_ON(!sf_g); BUG_ON(!sf_i); err = sf_path_from_dentry(__func__, sf_g, sf_i, dentry, &path); if (err) goto fail0; ssymname = kmalloc(offsetof(SHFLSTRING, String.utf8) + symname_len, GFP_KERNEL); if (!ssymname) { LogRelFunc(("kmalloc failed, caller=sf_symlink\n")); err = -ENOMEM; goto fail1; } ssymname->u16Length = symname_len - 1; ssymname->u16Size = symname_len; memcpy(ssymname->String.utf8, symname, symname_len); rc = VbglR0SfSymlink(&client_handle, &sf_g->map, path, ssymname, &info); kfree(ssymname); if (RT_FAILURE(rc)) { if (rc == VERR_WRITE_PROTECT) { err = -EROFS; goto fail1; } LogFunc(("VbglR0SfSymlink(%s) failed rc=%Rrc\n", sf_i->path->String.utf8, rc)); err = -EPROTO; goto fail1; } err = sf_instantiate(parent, dentry, path, &info, SHFL_HANDLE_NIL); if (err) { LogFunc(("could not instantiate dentry for %s err=%d\n", sf_i->path->String.utf8, err)); goto fail1; } sf_i->force_restat = 1; return 0; fail1: kfree(path); fail0: return err; } #endif /* LINUX_VERSION_CODE >= 2.6.0 */ struct inode_operations sf_dir_iops = { .lookup = sf_lookup, .create = sf_create, .mkdir = sf_mkdir, .rmdir = sf_rmdir, .unlink = sf_unlink, .rename = sf_rename, #if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 0) .revalidate = sf_inode_revalidate #else .getattr = sf_getattr, .setattr = sf_setattr, .symlink = sf_symlink #endif };