/* Virtual File System: interface functions Copyright (C) 2011 Free Software Foundation, Inc. Written by: Slava Zanko , 2011 This file is part of the Midnight Commander. The Midnight Commander is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. The Midnight Commander is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ /** * \file * \brief Source: Virtual File System: path handlers * \author Slava Zanko * \date 2011 */ #include #include #include /* For atol() */ #include #include #include #include #include #include /* is_digit() */ #include #include #include #include #include "lib/global.h" #include "lib/widget.h" /* message() */ #include "lib/strutil.h" /* str_crt_conv_from() */ #include "vfs.h" #include "utilvfs.h" #include "path.h" #include "gc.h" extern GString *vfs_str_buffer; extern struct vfs_class *current_vfs; /*** global variables ****************************************************************************/ struct dirent *mc_readdir_result = NULL; /*** file scope macro definitions ****************************************************************/ /*** file scope type declarations ****************************************************************/ /*** file scope variables ************************************************************************/ /*** file scope functions ************************************************************************/ /* --------------------------------------------------------------------------------------------- */ static char * mc_def_getlocalcopy (const char *filename) { char *tmp; int fdin, fdout; ssize_t i; char buffer[8192]; struct stat mystat; fdin = mc_open (filename, O_RDONLY | O_LINEAR); if (fdin == -1) return NULL; fdout = vfs_mkstemps (&tmp, "vfs", filename); if (fdout == -1) goto fail; while ((i = mc_read (fdin, buffer, sizeof (buffer))) > 0) { if (write (fdout, buffer, i) != i) goto fail; } if (i == -1) goto fail; i = mc_close (fdin); fdin = -1; if (i == -1) goto fail; if (close (fdout) == -1) { fdout = -1; goto fail; } if (mc_stat (filename, &mystat) != -1) chmod (tmp, mystat.st_mode); return tmp; fail: if (fdout != -1) close (fdout); if (fdin != -1) mc_close (fdin); g_free (tmp); return NULL; } /* --------------------------------------------------------------------------------------------- */ static int mc_def_ungetlocalcopy (struct vfs_class *vfs, const char *filename, const char *local, int has_changed) { int fdin = -1, fdout = -1; if (has_changed) { char buffer[8192]; ssize_t i; if (!vfs->write) goto failed; fdin = open (local, O_RDONLY); if (fdin == -1) goto failed; fdout = mc_open (filename, O_WRONLY | O_TRUNC); if (fdout == -1) goto failed; while ((i = read (fdin, buffer, sizeof (buffer))) > 0) if (mc_write (fdout, buffer, (size_t) i) != i) goto failed; if (i == -1) goto failed; if (close (fdin) == -1) { fdin = -1; goto failed; } fdin = -1; if (mc_close (fdout) == -1) { fdout = -1; goto failed; } } unlink (local); return 0; failed: message (D_ERROR, _("Changes to file lost"), "%s", filename); if (fdout != -1) mc_close (fdout); if (fdin != -1) close (fdin); unlink (local); return -1; } /* --------------------------------------------------------------------------------------------- */ /*** public functions ****************************************************************************/ /* --------------------------------------------------------------------------------------------- */ int mc_open (const char *filename, int flags, ...) { int mode = 0, result = -1; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (filename); if (vpath == NULL) return -1; /* Get the mode flag */ if (flags & O_CREAT) { va_list ap; va_start (ap, flags); mode = va_arg (ap, int); va_end (ap); } path_element = vfs_path_get_by_index (vpath, -1); if (path_element != NULL && path_element->class->open != NULL) { void *info; /* open must be supported */ info = path_element->class->open (vpath, flags, mode); if (info == NULL) errno = vfs_ferrno (path_element->class); else result = vfs_new_handle (path_element->class, info); } else errno = -EOPNOTSUPP; vfs_path_free (vpath); return result; } /* --------------------------------------------------------------------------------------------- */ /* *INDENT-OFF* */ #define MC_NAMEOP(name, inarg, callarg) \ int mc_##name inarg \ { \ int result; \ vfs_path_t *vpath; \ vfs_path_element_t *path_element; \ \ vpath = vfs_path_from_str (path); \ if (vpath == NULL) \ return -1; \ \ path_element = vfs_path_get_by_index (vpath, -1); \ if (path_element == NULL) \ { \ vfs_path_free(vpath); \ return -1; \ } \ \ if (path_element->class == NULL) \ { \ vfs_path_free(vpath); \ return -1; \ } \ \ result = path_element->class->name != NULL ? path_element->class->name callarg : -1; \ if (result == -1) \ errno = path_element->class->name != NULL ? vfs_ferrno (path_element->class) : E_NOTSUPP; \ vfs_path_free(vpath); \ return result; \ } MC_NAMEOP (chmod, (const char *path, mode_t mode), (vpath, mode)) MC_NAMEOP (chown, (const char *path, uid_t owner, gid_t group), (vpath, owner, group)) MC_NAMEOP (utime, (const char *path, struct utimbuf * times), (vpath, times)) MC_NAMEOP (readlink, (const char *path, char *buf, size_t bufsiz), (vpath, buf, bufsiz)) MC_NAMEOP (unlink, (const char *path), (vpath)) MC_NAMEOP (mkdir, (const char *path, mode_t mode), (vpath, mode)) MC_NAMEOP (rmdir, (const char *path), (vpath)) MC_NAMEOP (mknod, (const char *path, mode_t mode, dev_t dev), (vpath, mode, dev)) /* *INDENT-ON* */ /* --------------------------------------------------------------------------------------------- */ int mc_symlink (const char *name1, const char *path) { int result = -1; vfs_path_t *vpath1, *vpath2; vpath1 = vfs_path_from_str (path); if (vpath1 == NULL) return -1; vpath2 = vfs_path_from_str (name1); if (vpath2 != NULL) { vfs_path_element_t *path_element = vfs_path_get_by_index (vpath1, vfs_path_elements_count (vpath1) - 1); if (path_element != NULL) { result = path_element->class->symlink != NULL ? path_element->class->symlink (vpath2, vpath1) : -1; if (result == -1) errno = path_element->class->symlink != NULL ? vfs_ferrno (path_element->class) : E_NOTSUPP; } } vfs_path_free (vpath1); vfs_path_free (vpath2); return result; } /* --------------------------------------------------------------------------------------------- */ /* *INDENT-OFF* */ #define MC_HANDLEOP(name, inarg, callarg) \ ssize_t mc_##name inarg \ { \ struct vfs_class *vfs; \ int result; \ if (handle == -1) \ return -1; \ vfs = vfs_class_find_by_handle (handle); \ if (vfs == NULL) \ return -1; \ result = vfs->name != NULL ? vfs->name callarg : -1; \ if (result == -1) \ errno = vfs->name != NULL ? vfs_ferrno (vfs) : E_NOTSUPP; \ return result; \ } MC_HANDLEOP (read, (int handle, void *buffer, size_t count), (vfs_class_data_find_by_handle (handle), buffer, count)) MC_HANDLEOP (write, (int handle, const void *buf, size_t nbyte), (vfs_class_data_find_by_handle (handle), buf, nbyte)) /* --------------------------------------------------------------------------------------------- */ #define MC_RENAMEOP(name) \ int mc_##name (const char *fname1, const char *fname2) \ { \ int result; \ vfs_path_t *vpath1, *vpath2; \ vfs_path_element_t *path_element1, *path_element2; \ \ vpath1 = vfs_path_from_str (fname1); \ if (vpath1 == NULL) \ return -1; \ \ vpath2 = vfs_path_from_str (fname2); \ if (vpath2 == NULL) \ { \ vfs_path_free(vpath1); \ return -1; \ }\ path_element1 = vfs_path_get_by_index (vpath1, vfs_path_elements_count (vpath1) - 1); \ path_element2 = vfs_path_get_by_index (vpath2, vfs_path_elements_count (vpath2) - 1); \ \ if (path_element1->class != path_element2->class) \ { \ errno = EXDEV; \ vfs_path_free(vpath1); \ vfs_path_free(vpath2); \ return -1; \ }\ \ result = path_element1->class->name != NULL \ ? path_element1->class->name (vpath1, vpath2) \ : -1; \ if (result == -1) \ errno = path_element1->class->name != NULL ? vfs_ferrno (path_element1->class) : E_NOTSUPP; \ vfs_path_free(vpath1); \ vfs_path_free(vpath2); \ return result; \ } MC_RENAMEOP (link) MC_RENAMEOP (rename) /* *INDENT-ON* */ /* --------------------------------------------------------------------------------------------- */ int mc_ctl (int handle, int ctlop, void *arg) { struct vfs_class *vfs = vfs_class_find_by_handle (handle); if (vfs == NULL) return 0; return vfs->ctl ? (*vfs->ctl) (vfs_class_data_find_by_handle (handle), ctlop, arg) : 0; } /* --------------------------------------------------------------------------------------------- */ int mc_setctl (const char *path, int ctlop, void *arg) { int result = -1; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (path); if (vpath == NULL) vfs_die ("You don't want to pass NULL to mc_setctl."); path_element = vfs_path_get_by_index (vpath, -1); if (path_element != NULL && path_element->class != NULL) result = path_element->class->setctl != NULL ? path_element->class->setctl (vpath, ctlop, arg) : 0; vfs_path_free (vpath); return result; } /* --------------------------------------------------------------------------------------------- */ int mc_close (int handle) { struct vfs_class *vfs; int result; if (handle == -1 || !vfs_class_data_find_by_handle (handle)) return -1; vfs = vfs_class_find_by_handle (handle); if (vfs == NULL) return -1; if (handle < 3) return close (handle); if (!vfs->close) vfs_die ("VFS must support close.\n"); result = (*vfs->close) (vfs_class_data_find_by_handle (handle)); vfs_free_handle (handle); if (result == -1) errno = vfs_ferrno (vfs); return result; } /* --------------------------------------------------------------------------------------------- */ DIR * mc_opendir (const char *dirname) { int handle, *handlep; void *info; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (dirname); if (vpath == NULL) return NULL; path_element = vfs_path_get_by_index (vpath, -1); info = path_element->class->opendir ? (*path_element->class->opendir) (vpath) : NULL; if (info == NULL) { errno = path_element->class->opendir ? vfs_ferrno (path_element->class) : E_NOTSUPP; return NULL; } path_element->dir.info = info; path_element->dir.converter = (path_element->encoding != NULL) ? str_crt_conv_from (path_element->encoding) : str_cnv_from_term; if (path_element->dir.converter == INVALID_CONV) path_element->dir.converter = str_cnv_from_term; handle = vfs_new_handle (path_element->class, path_element); handlep = g_new (int, 1); *handlep = handle; return (DIR *) handlep; } /* --------------------------------------------------------------------------------------------- */ struct dirent * mc_readdir (DIR * dirp) { int handle; struct vfs_class *vfs; struct dirent *entry = NULL; vfs_path_element_t *vfs_path_element; estr_t state; if (!mc_readdir_result) { /* We can't just allocate struct dirent as (see man dirent.h) * struct dirent has VERY nonnaive semantics of allocating * d_name in it. Moreover, linux's glibc-2.9 allocates dirents _less_, * than 'sizeof (struct dirent)' making full bitwise (sizeof dirent) copy * heap corrupter. So, allocate longliving dirent with at least * (MAXNAMLEN + 1) for d_name in it. * Strictly saying resulting dirent is unusable as we don't adjust internal * structures, holding dirent size. But we don't use it in libc infrastructure. * TODO: to make simpler homemade dirent-alike structure. */ mc_readdir_result = (struct dirent *) g_malloc (sizeof (struct dirent) + MAXNAMLEN + 1); } if (!dirp) { errno = EFAULT; return NULL; } handle = *(int *) dirp; vfs = vfs_class_find_by_handle (handle); if (vfs == NULL) return NULL; vfs_path_element = vfs_class_data_find_by_handle (handle); if (vfs->readdir) { entry = (*vfs->readdir) (vfs_path_element->dir.info); if (entry == NULL) return NULL; g_string_set_size (vfs_str_buffer, 0); state = str_vfs_convert_from (vfs_path_element->dir.converter, entry->d_name, vfs_str_buffer); mc_readdir_result->d_ino = entry->d_ino; g_strlcpy (mc_readdir_result->d_name, vfs_str_buffer->str, MAXNAMLEN + 1); } if (entry == NULL) errno = vfs->readdir ? vfs_ferrno (vfs) : E_NOTSUPP; return (entry != NULL) ? mc_readdir_result : NULL; } /* --------------------------------------------------------------------------------------------- */ int mc_closedir (DIR * dirp) { int handle = *(int *) dirp; struct vfs_class *vfs; int result = -1; vfs = vfs_class_find_by_handle (handle); if (vfs != NULL) { vfs_path_element_t *vfs_path_element; vfs_path_element = vfs_class_data_find_by_handle (handle); if (vfs_path_element->dir.converter != str_cnv_from_term) { str_close_conv (vfs_path_element->dir.converter); vfs_path_element->dir.converter = INVALID_CONV; } result = vfs->closedir ? (*vfs->closedir) (vfs_path_element->dir.info) : -1; vfs_free_handle (handle); vfs_path_element_free (vfs_path_element); } g_free (dirp); return result; } /* --------------------------------------------------------------------------------------------- */ int mc_stat (const char *filename, struct stat *buf) { int result = -1; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (filename); if (vpath == NULL) return -1; path_element = vfs_path_get_by_index (vpath, -1); if (path_element != NULL && path_element->class != NULL) { result = path_element->class->stat ? (*path_element->class->stat) (vpath, buf) : -1; if (result == -1) errno = path_element->class->name ? vfs_ferrno (path_element->class) : E_NOTSUPP; } vfs_path_free (vpath); return result; } /* --------------------------------------------------------------------------------------------- */ int mc_lstat (const char *filename, struct stat *buf) { int result = -1; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (filename); if (vpath == NULL) return -1; path_element = vfs_path_get_by_index (vpath, -1); if (path_element != NULL && path_element->class != NULL) { result = path_element->class->lstat ? (*path_element->class->lstat) (vpath, buf) : -1; if (result == -1) errno = path_element->class->name ? vfs_ferrno (path_element->class) : E_NOTSUPP; } vfs_path_free (vpath); return result; } /* --------------------------------------------------------------------------------------------- */ int mc_fstat (int handle, struct stat *buf) { struct vfs_class *vfs; int result; if (handle == -1) return -1; vfs = vfs_class_find_by_handle (handle); if (vfs == NULL) return -1; result = vfs->fstat ? (*vfs->fstat) (vfs_class_data_find_by_handle (handle), buf) : -1; if (result == -1) errno = vfs->name ? vfs_ferrno (vfs) : E_NOTSUPP; return result; } /* --------------------------------------------------------------------------------------------- */ /** * Return current directory. If it's local, reread the current directory * from the OS. Put directory to the provided buffer. */ char * mc_get_current_wd (char *buffer, size_t size) { char *cwd = _vfs_get_cwd (); g_strlcpy (buffer, cwd, size); g_free (cwd); return buffer; } /* --------------------------------------------------------------------------------------------- */ char * mc_getlocalcopy (const char *pathname) { char *result = NULL; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (pathname); if (vpath == NULL) return NULL; path_element = vfs_path_get_by_index (vpath, -1); if (path_element != NULL) { result = path_element->class->getlocalcopy != NULL ? path_element->class->getlocalcopy (vpath) : mc_def_getlocalcopy (pathname); if (result == NULL) errno = vfs_ferrno (path_element->class); } vfs_path_free (vpath); return result; } /* --------------------------------------------------------------------------------------------- */ int mc_ungetlocalcopy (const char *pathname, const char *local, int has_changed) { int return_value = -1; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (pathname); if (vpath == NULL) return -1; path_element = vfs_path_get_by_index (vpath, -1); if (path_element != NULL) { return_value = path_element->class->ungetlocalcopy != NULL ? path_element->class->ungetlocalcopy (vpath, local, has_changed) : mc_def_ungetlocalcopy (path_element->class, path_element->path, local, has_changed); } vfs_path_free (vpath); return return_value; } /* --------------------------------------------------------------------------------------------- */ /** * VFS chdir. * Return 0 on success, -1 on failure. */ int mc_chdir (const char *path) { struct vfs_class *old_vfs; vfsid old_vfsid; int result; vfs_path_t *vpath; vfs_path_element_t *path_element; vpath = vfs_path_from_str (path); if (vpath == NULL) return -1; path_element = vfs_path_get_by_index (vpath, -1); if (!path_element->class->chdir) return -1; result = (*path_element->class->chdir) (vpath); if (result == -1) { errno = vfs_ferrno (path_element->class); return -1; } old_vfsid = vfs_getid (vfs_get_raw_current_dir ()); old_vfs = current_vfs; /* Actually change directory */ vfs_set_raw_current_dir (vpath); current_vfs = path_element->class; /* This function uses the new current_dir implicitly */ vfs_stamp_create (old_vfs, old_vfsid); /* Sometimes we assume no trailing slash on cwd */ path_element = vfs_path_get_by_index (vfs_get_raw_current_dir (), -1); if ((path_element != NULL) && (*path_element->path != '\0')) { char *p; p = strchr (path_element->path, 0) - 1; if (*p == PATH_SEP && p > path_element->path) *p = 0; } return 0; } /* --------------------------------------------------------------------------------------------- */ off_t mc_lseek (int fd, off_t offset, int whence) { struct vfs_class *vfs; off_t result; if (fd == -1) return -1; vfs = vfs_class_find_by_handle (fd); if (vfs == NULL) return -1; result = vfs->lseek ? (*vfs->lseek) (vfs_class_data_find_by_handle (fd), offset, whence) : -1; if (result == -1) errno = vfs->lseek ? vfs_ferrno (vfs) : E_NOTSUPP; return result; } /* --------------------------------------------------------------------------------------------- */