diff options
author | alfred <alfred@FreeBSD.org> | 2002-01-13 11:58:06 +0000 |
---|---|---|
committer | alfred <alfred@FreeBSD.org> | 2002-01-13 11:58:06 +0000 |
commit | 844237b3960bfbf49070d6371a84f67f9e3366f6 (patch) | |
tree | 598e20df363e602313c7ad93de8f8c4b4240d61d /sys/compat/linux/linux_file.c | |
parent | 8cd61193307ff459ae72eb7aa6a734eb5e3b427e (diff) | |
download | FreeBSD-src-844237b3960bfbf49070d6371a84f67f9e3366f6.zip FreeBSD-src-844237b3960bfbf49070d6371a84f67f9e3366f6.tar.gz |
SMP Lock struct file, filedesc and the global file list.
Seigo Tanimura (tanimura) posted the initial delta.
I've polished it quite a bit reducing the need for locking and
adapting it for KSE.
Locks:
1 mutex in each filedesc
protects all the fields.
protects "struct file" initialization, while a struct file
is being changed from &badfileops -> &pipeops or something
the filedesc should be locked.
1 mutex in each struct file
protects the refcount fields.
doesn't protect anything else.
the flags used for garbage collection have been moved to
f_gcflag which was the FILLER short, this doesn't need
locking because the garbage collection is a single threaded
container.
could likely be made to use a pool mutex.
1 sx lock for the global filelist.
struct file * fhold(struct file *fp);
/* increments reference count on a file */
struct file * fhold_locked(struct file *fp);
/* like fhold but expects file to locked */
struct file * ffind_hold(struct thread *, int fd);
/* finds the struct file in thread, adds one reference and
returns it unlocked */
struct file * ffind_lock(struct thread *, int fd);
/* ffind_hold, but returns file locked */
I still have to smp-safe the fget cruft, I'll get to that asap.
Diffstat (limited to 'sys/compat/linux/linux_file.c')
-rw-r--r-- | sys/compat/linux/linux_file.c | 34 |
1 files changed, 23 insertions, 11 deletions
diff --git a/sys/compat/linux/linux_file.c b/sys/compat/linux/linux_file.c index 4609a1b..9e1156c 100644 --- a/sys/compat/linux/linux_file.c +++ b/sys/compat/linux/linux_file.c @@ -135,12 +135,13 @@ linux_open(struct thread *td, struct linux_open_args *args) PROC_LOCK(p); if (!error && !(bsd_open_args.flags & O_NOCTTY) && SESS_LEADER(p) && !(p->p_flag & P_CONTROLT)) { - struct filedesc *fdp = p->p_fd; - struct file *fp = fdp->fd_ofiles[td->td_retval[0]]; + struct file *fp; + fp = ffind_hold(td, td->td_retval[0]); PROC_UNLOCK(p); if (fp->f_type == DTYPE_VNODE) fo_ioctl(fp, TIOCSCTTY, (caddr_t) 0, td); + fdrop(fp, td); } else PROC_UNLOCK(p); #ifdef DEBUG @@ -270,21 +271,29 @@ getdents_common(struct thread *td, struct linux_getdents64_args *args, if ((error = getvnode(td->td_proc->p_fd, args->fd, &fp)) != 0) return (error); - if ((fp->f_flag & FREAD) == 0) + if ((fp->f_flag & FREAD) == 0) { + fdrop(fp, td); return (EBADF); + } vp = (struct vnode *) fp->f_data; - if (vp->v_type != VDIR) + if (vp->v_type != VDIR) { + fdrop(fp, td); return (EINVAL); + } - if ((error = VOP_GETATTR(vp, &va, td->td_proc->p_ucred, td))) + if ((error = VOP_GETATTR(vp, &va, td->td_proc->p_ucred, td))) { + fdrop(fp, td); return (error); + } nbytes = args->count; if (nbytes == 1) { /* readdir(2) case. Always struct dirent. */ - if (is64bit) + if (is64bit) { + fdrop(fp, td); return (EINVAL); + } nbytes = sizeof(linux_dirent); justone = 1; } else @@ -435,6 +444,7 @@ out: free(cookies, M_TEMP); VOP_UNLOCK(vp, 0, td); + fdrop(fp, td); free(buf, M_TEMP); return (error); } @@ -987,12 +997,14 @@ fcntl_common(struct thread *td, struct linux_fcntl64_args *args) * significant effect for pipes (SIGIO is not delivered for * pipes under Linux-2.2.35 at least). */ - fdp = td->td_proc->p_fd; - if ((u_int)args->fd >= fdp->fd_nfiles || - (fp = fdp->fd_ofiles[args->fd]) == NULL) - return (EBADF); - if (fp->f_type == DTYPE_PIPE) + fp = ffind_hold(td, args->fd); + if (fp == NULL) + return EBADF; + if (fp->f_type == DTYPE_PIPE) { + fdrop(fp, td); return (EINVAL); + } + fdrop(fp, td); fcntl_args.cmd = F_SETOWN; fcntl_args.arg = args->arg; |