1683e75edc
If a vnode is released asynchronously through areleasef(), it is possible for the user process to reuse the file descriptor before areleasef is called. When this happens, getf() will return a stale reference, any operations in the kernel on that file descriptor will fail (as it is closed) and the operations meant for that fd will never occur from userspace's perspective. We correct this by detecting this condition in getf(), doing a putf on the old file handle, updating the file descriptor and proceeding as if everything was fine. When the areleasef() is done, it will harmlessly decrement the reference counter on the Illumos file handle. Signed-off-by: Richard Yao <ryao@gentoo.org> Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov> Closes #492
43 lines
1.7 KiB
C
43 lines
1.7 KiB
C
/*****************************************************************************\
|
|
* Copyright (C) 2015 Cluster Inc.
|
|
* Produced at ClusterHQ Inc (cf, DISCLAIMER).
|
|
* Written by Richard Yao <richard.yao@clusterhq.com>.
|
|
*
|
|
* 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 and we do not do a getf() between
|
|
* the change and using it, 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 */
|