diff options
author | Richard Yao <[email protected]> | 2015-02-17 10:12:56 -0500 |
---|---|---|
committer | Brian Behlendorf <[email protected]> | 2015-04-24 13:02:37 -0700 |
commit | d3c677bcd330423c72cd4d4100727e5c1e8c1f38 (patch) | |
tree | 5c4652b9e6c2ca91c92640361d3867bb5b0d74ba | |
parent | 313b1ea622275e24c3046c3b04a98a933b18f8de (diff) |
Implement areleasef()
Signed-off-by: Richard Yao <[email protected]>
Signed-off-by: Brian Behlendorf <[email protected]>
Closes #449
-rw-r--r-- | include/sys/Makefile.am | 1 | ||||
-rw-r--r-- | include/sys/user.h | 42 | ||||
-rw-r--r-- | include/sys/vnode.h | 3 | ||||
-rw-r--r-- | module/spl/spl-vnode.c | 19 |
4 files changed, 60 insertions, 5 deletions
diff --git a/include/sys/Makefile.am b/include/sys/Makefile.am index f9e883fd4..73c4a8421 100644 --- a/include/sys/Makefile.am +++ b/include/sys/Makefile.am @@ -91,6 +91,7 @@ KERNEL_H = \ $(top_srcdir)/include/sys/u8_textprep.h \ $(top_srcdir)/include/sys/uio.h \ $(top_srcdir)/include/sys/unistd.h \ + $(top_srcdir)/include/sys/user.h \ $(top_srcdir)/include/sys/va_list.h \ $(top_srcdir)/include/sys/varargs.h \ $(top_srcdir)/include/sys/vfs.h \ diff --git a/include/sys/user.h b/include/sys/user.h new file mode 100644 index 000000000..ebbe8f68e --- /dev/null +++ b/include/sys/user.h @@ -0,0 +1,42 @@ +/*****************************************************************************\ + * Copyright (C) 2015 Cluster Inc. + * Produced at ClusterHQ Inc (cf, DISCLAIMER). + * Written by Richard Yao <[email protected]>. + * + * This file is part of the SPL, Solaris Porting Layer. + * For details, see <http://zfsonlinux.org/>. + * + * The SPL 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 SPL 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 the SPL. If not, see <http://www.gnu.org/licenses/>. +\*****************************************************************************/ + +#ifndef _SPL_USER_H +#define _SPL_USER_H + +/* + * We have uf_info_t for areleasef(). We implement areleasef() using a global + * linked list of all open file descriptors with the task structs referenced, + * so accessing the correct descriptor from areleasef() only requires knowing + * about the Linux task_struct. Since this is internal to our compatibility + * layer, we make it an opaque type. + * + * XXX: If the descriptor changes under us, we would get an incorrect + * reference. + */ + +struct uf_info; +typedef struct uf_info uf_info_t; + +#define P_FINFO(x) ((uf_info_t *)x) + +#endif /* SPL_USER_H */ diff --git a/include/sys/vnode.h b/include/sys/vnode.h index 07a344938..0b857d384 100644 --- a/include/sys/vnode.h +++ b/include/sys/vnode.h @@ -40,6 +40,7 @@ #include <sys/types.h> #include <sys/time.h> #include <sys/uio.h> +#include <sys/user.h> #include <sys/sunldi.h> /* @@ -184,6 +185,7 @@ extern int vn_space(vnode_t *vp, int cmd, struct flock *bfp, int flag, offset_t offset, void *x6, void *x7); extern file_t *vn_getf(int fd); extern void vn_releasef(int fd); +extern void vn_areleasef(int fd, uf_info_t *fip); extern int vn_set_pwd(const char *filename); int spl_vn_init(void); @@ -198,6 +200,7 @@ void spl_vn_fini(void); #define vn_is_readonly(vp) 0 #define getf vn_getf #define releasef vn_releasef +#define areleasef vn_areleasef extern vnode_t *rootdir; diff --git a/module/spl/spl-vnode.c b/module/spl/spl-vnode.c index 4c62097dc..ab9830d18 100644 --- a/module/spl/spl-vnode.c +++ b/module/spl/spl-vnode.c @@ -623,14 +623,14 @@ EXPORT_SYMBOL(vn_space); /* Function must be called while holding the vn_file_lock */ static file_t * -file_find(int fd) +file_find(int fd, struct task_struct *task) { file_t *fp; ASSERT(spin_is_locked(&vn_file_lock)); list_for_each_entry(fp, &vn_file_list, f_list) { - if (fd == fp->f_fd && fp->f_task == current) { + if (fd == fp->f_fd && fp->f_task == task) { ASSERT(atomic_read(&fp->f_ref) != 0); return fp; } @@ -654,7 +654,7 @@ vn_getf(int fd) /* Already open just take an extra reference */ spin_lock(&vn_file_lock); - fp = file_find(fd); + fp = file_find(fd, current); if (fp) { atomic_inc(&fp->f_ref); spin_unlock(&vn_file_lock); @@ -734,13 +734,21 @@ static void releasef_locked(file_t *fp) void vn_releasef(int fd) { + areleasef(fd, P_FINFO(current)); +} +EXPORT_SYMBOL(releasef); + +void +vn_areleasef(int fd, uf_info_t *fip) +{ file_t *fp; + struct task_struct *task = (struct task_struct *)fip; if (fd < 0) return; spin_lock(&vn_file_lock); - fp = file_find(fd); + fp = file_find(fd, task); if (fp) { atomic_dec(&fp->f_ref); if (atomic_read(&fp->f_ref) > 0) { @@ -755,7 +763,8 @@ vn_releasef(int fd) return; } /* releasef() */ -EXPORT_SYMBOL(releasef); +EXPORT_SYMBOL(areleasef); + static void #ifdef HAVE_SET_FS_PWD_WITH_CONST |