2005-01-06 23:35:40 +00:00
|
|
|
/*-
|
2017-11-20 19:43:44 +00:00
|
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
*
|
1994-05-24 10:09:53 +00:00
|
|
|
* Copyright (c) 1982, 1986, 1989, 1993
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
* (c) UNIX System Laboratories, Inc.
|
|
|
|
* All or some portions of this file are derived from material licensed
|
|
|
|
* to the University of California by American Telephone and Telegraph
|
|
|
|
* Co. or Unix System Laboratories, Inc. and are reproduced herein with
|
|
|
|
* the permission of UNIX System Laboratories, Inc.
|
|
|
|
*
|
2013-03-15 12:57:30 +00:00
|
|
|
* Copyright (c) 2012 Konstantin Belousov <kib@FreeBSD.org>
|
2014-06-15 04:51:53 +00:00
|
|
|
* Copyright (c) 2013, 2014 The FreeBSD Foundation
|
2013-03-15 14:01:37 +00:00
|
|
|
*
|
2013-03-15 12:57:30 +00:00
|
|
|
* Portions of this software were developed by Konstantin Belousov
|
|
|
|
* under sponsorship from the FreeBSD Foundation.
|
|
|
|
*
|
1994-05-24 10:09:53 +00:00
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
2016-09-15 13:16:20 +00:00
|
|
|
* 3. Neither the name of the University nor the names of its contributors
|
1994-05-24 10:09:53 +00:00
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* @(#)vfs_vnops.c 8.2 (Berkeley) 1/21/94
|
|
|
|
*/
|
|
|
|
|
2003-06-11 00:56:59 +00:00
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
2015-10-08 09:54:33 +00:00
|
|
|
#include "opt_hwpmc_hooks.h"
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
2013-08-21 17:36:01 +00:00
|
|
|
#include <sys/disk.h>
|
Detect badly behaved coredump note helpers
Coredump notes depend on being able to invoke dump routines twice; once
in a dry-run mode to get the size of the note, and another to actually
emit the note to the corefile.
When a note helper emits a different length section the second time
around than the length it requested the first time, the kernel produces
a corrupt coredump.
NT_PROCSTAT_FILES output length, when packing kinfo structs, is tied to
the length of filenames corresponding to vnodes in the process' fd table
via vn_fullpath. As vnodes may move around during dump, this is racy.
So:
- Detect badly behaved notes in putnote() and pad underfilled notes.
- Add a fail point, debug.fail_point.fill_kinfo_vnode__random_path to
exercise the NT_PROCSTAT_FILES corruption. It simply picks random
lengths to expand or truncate paths to in fo_fill_kinfo_vnode().
- Add a sysctl, kern.coredump_pack_fileinfo, to allow users to
disable kinfo packing for PROCSTAT_FILES notes. This should avoid
both FILES note corruption and truncation, even if filenames change,
at the cost of about 1 kiB in padding bloat per open fd. Document
the new sysctl in core.5.
- Fix note_procstat_files to self-limit in the 2nd pass. Since
sometimes this will result in a short write, pad up to our advertised
size. This addresses note corruption, at the risk of sometimes
truncating the last several fd info entries.
- Fix NT_PROCSTAT_FILES consumers libutil and libprocstat to grok the
zero padding.
With suggestions from: bjk, jhb, kib, wblock
Approved by: markj (mentor)
Relnotes: yes
Sponsored by: EMC / Isilon Storage Division
Differential Revision: https://reviews.freebsd.org/D3548
2015-09-03 20:32:10 +00:00
|
|
|
#include <sys/fail.h>
|
1997-03-23 03:37:54 +00:00
|
|
|
#include <sys/fcntl.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/file.h>
|
2004-07-10 21:47:53 +00:00
|
|
|
#include <sys/kdb.h>
|
2019-05-21 20:38:48 +00:00
|
|
|
#include <sys/ktr.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/stat.h>
|
2006-11-06 13:42:10 +00:00
|
|
|
#include <sys/priv.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/proc.h>
|
2003-04-29 13:36:06 +00:00
|
|
|
#include <sys/limits.h>
|
2001-10-11 17:52:20 +00:00
|
|
|
#include <sys/lock.h>
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
#include <sys/mman.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/mount.h>
|
2000-10-20 07:58:15 +00:00
|
|
|
#include <sys/mutex.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/namei.h>
|
|
|
|
#include <sys/vnode.h>
|
2000-05-05 09:59:14 +00:00
|
|
|
#include <sys/bio.h>
|
1999-08-04 18:53:50 +00:00
|
|
|
#include <sys/buf.h>
|
1997-03-24 11:52:29 +00:00
|
|
|
#include <sys/filio.h>
|
2010-05-05 16:44:25 +00:00
|
|
|
#include <sys/resourcevar.h>
|
2013-03-09 02:32:23 +00:00
|
|
|
#include <sys/rwlock.h>
|
2002-03-26 01:09:51 +00:00
|
|
|
#include <sys/sx.h>
|
2012-06-03 16:06:56 +00:00
|
|
|
#include <sys/sysctl.h>
|
1997-03-24 11:52:29 +00:00
|
|
|
#include <sys/ttycom.h>
|
1999-08-13 11:22:48 +00:00
|
|
|
#include <sys/conf.h>
|
2001-12-18 20:48:54 +00:00
|
|
|
#include <sys/syslog.h>
|
2004-06-01 18:03:20 +00:00
|
|
|
#include <sys/unistd.h>
|
2014-09-22 16:20:47 +00:00
|
|
|
#include <sys/user.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2011-08-16 20:07:47 +00:00
|
|
|
#include <security/audit/audit.h>
|
2006-10-22 11:52:19 +00:00
|
|
|
#include <security/mac/mac_framework.h>
|
|
|
|
|
2011-08-25 08:17:39 +00:00
|
|
|
#include <vm/vm.h>
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
#include <vm/vm_extern.h>
|
|
|
|
#include <vm/pmap.h>
|
|
|
|
#include <vm/vm_map.h>
|
2011-08-25 08:17:39 +00:00
|
|
|
#include <vm/vm_object.h>
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
#include <vm/vm_page.h>
|
2019-09-03 20:31:48 +00:00
|
|
|
#include <vm/vm_pager.h>
|
2011-08-25 08:17:39 +00:00
|
|
|
|
2015-10-08 09:54:33 +00:00
|
|
|
#ifdef HWPMC_HOOKS
|
|
|
|
#include <sys/pmckern.h>
|
|
|
|
#endif
|
|
|
|
|
2002-12-24 09:44:51 +00:00
|
|
|
static fo_rdwr_t vn_read;
|
|
|
|
static fo_rdwr_t vn_write;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
static fo_rdwr_t vn_io_fault;
|
2008-01-07 20:05:19 +00:00
|
|
|
static fo_truncate_t vn_truncate;
|
2002-12-24 09:44:51 +00:00
|
|
|
static fo_ioctl_t vn_ioctl;
|
|
|
|
static fo_poll_t vn_poll;
|
|
|
|
static fo_kqfilter_t vn_kqfilter;
|
|
|
|
static fo_stat_t vn_statfile;
|
|
|
|
static fo_close_t vn_closefile;
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
static fo_mmap_t vn_mmap;
|
2020-01-08 19:05:32 +00:00
|
|
|
static fo_fallocate_t vn_fallocate;
|
1995-12-17 21:23:44 +00:00
|
|
|
|
2001-02-15 16:34:11 +00:00
|
|
|
struct fileops vnops = {
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
.fo_read = vn_io_fault,
|
|
|
|
.fo_write = vn_io_fault,
|
2008-01-07 20:05:19 +00:00
|
|
|
.fo_truncate = vn_truncate,
|
2003-06-18 18:16:40 +00:00
|
|
|
.fo_ioctl = vn_ioctl,
|
|
|
|
.fo_poll = vn_poll,
|
|
|
|
.fo_kqfilter = vn_kqfilter,
|
|
|
|
.fo_stat = vn_statfile,
|
|
|
|
.fo_close = vn_closefile,
|
2011-08-16 20:07:47 +00:00
|
|
|
.fo_chmod = vn_chmod,
|
|
|
|
.fo_chown = vn_chown,
|
2013-08-15 07:54:31 +00:00
|
|
|
.fo_sendfile = vn_sendfile,
|
2013-08-21 17:36:01 +00:00
|
|
|
.fo_seek = vn_seek,
|
2014-09-22 16:20:47 +00:00
|
|
|
.fo_fill_kinfo = vn_fill_kinfo,
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
.fo_mmap = vn_mmap,
|
2020-01-08 19:05:32 +00:00
|
|
|
.fo_fallocate = vn_fallocate,
|
2003-06-18 19:53:59 +00:00
|
|
|
.fo_flags = DFLAG_PASSABLE | DFLAG_SEEKABLE
|
2000-04-16 18:53:38 +00:00
|
|
|
};
|
|
|
|
|
2014-06-15 04:51:53 +00:00
|
|
|
static const int io_hold_cnt = 16;
|
|
|
|
static int vn_io_fault_enable = 1;
|
|
|
|
SYSCTL_INT(_debug, OID_AUTO, vn_io_fault_enable, CTLFLAG_RW,
|
|
|
|
&vn_io_fault_enable, 0, "Enable vn_io_fault lock avoidance");
|
2015-07-31 04:12:51 +00:00
|
|
|
static int vn_io_fault_prefault = 0;
|
|
|
|
SYSCTL_INT(_debug, OID_AUTO, vn_io_fault_prefault, CTLFLAG_RW,
|
|
|
|
&vn_io_fault_prefault, 0, "Enable vn_io_fault prefaulting");
|
2014-06-15 04:51:53 +00:00
|
|
|
static u_long vn_io_faults_cnt;
|
|
|
|
SYSCTL_ULONG(_debug, OID_AUTO, vn_io_faults, CTLFLAG_RD,
|
|
|
|
&vn_io_faults_cnt, 0, "Count of vn_io_fault lock avoidance triggers");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns true if vn_io_fault mode of handling the i/o request should
|
|
|
|
* be used.
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
do_vn_io_fault(struct vnode *vp, struct uio *uio)
|
|
|
|
{
|
|
|
|
struct mount *mp;
|
|
|
|
|
|
|
|
return (uio->uio_segflg == UIO_USERSPACE && vp->v_type == VREG &&
|
|
|
|
(mp = vp->v_mount) != NULL &&
|
|
|
|
(mp->mnt_kern_flag & MNTK_NO_IOPF) != 0 && vn_io_fault_enable);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Structure used to pass arguments to vn_io_fault1(), to do either
|
|
|
|
* file- or vnode-based I/O calls.
|
|
|
|
*/
|
|
|
|
struct vn_io_fault_args {
|
|
|
|
enum {
|
|
|
|
VN_IO_FAULT_FOP,
|
|
|
|
VN_IO_FAULT_VOP
|
|
|
|
} kind;
|
|
|
|
struct ucred *cred;
|
|
|
|
int flags;
|
|
|
|
union {
|
|
|
|
struct fop_args_tag {
|
|
|
|
struct file *fp;
|
|
|
|
fo_rdwr_t *doio;
|
|
|
|
} fop_args;
|
|
|
|
struct vop_args_tag {
|
|
|
|
struct vnode *vp;
|
|
|
|
} vop_args;
|
|
|
|
} args;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int vn_io_fault1(struct vnode *vp, struct uio *uio,
|
|
|
|
struct vn_io_fault_args *args, struct thread *td);
|
|
|
|
|
2001-11-11 22:39:07 +00:00
|
|
|
int
|
2018-06-01 13:26:45 +00:00
|
|
|
vn_open(struct nameidata *ndp, int *flagp, int cmode, struct file *fp)
|
2001-11-11 22:39:07 +00:00
|
|
|
{
|
|
|
|
struct thread *td = ndp->ni_cnd.cn_thread;
|
|
|
|
|
2009-06-21 13:41:32 +00:00
|
|
|
return (vn_open_cred(ndp, flagp, cmode, 0, td->td_ucred, fp));
|
2001-11-11 22:39:07 +00:00
|
|
|
}
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
2012-06-08 18:32:09 +00:00
|
|
|
* Common code for vnode open operations via a name lookup.
|
|
|
|
* Lookup the vnode and invoke VOP_CREATE if needed.
|
1994-05-24 10:09:53 +00:00
|
|
|
* Check permissions, and call the VOP_OPEN or VOP_CREATE routine.
|
1999-12-15 23:02:35 +00:00
|
|
|
*
|
2000-05-12 16:06:49 +00:00
|
|
|
* Note that this does NOT free nameidata for the successful case,
|
1999-12-15 23:02:35 +00:00
|
|
|
* due to the NDINIT being done elsewhere.
|
1994-05-24 10:09:53 +00:00
|
|
|
*/
|
1994-05-25 09:21:21 +00:00
|
|
|
int
|
2009-06-21 13:41:32 +00:00
|
|
|
vn_open_cred(struct nameidata *ndp, int *flagp, int cmode, u_int vn_open_flags,
|
|
|
|
struct ucred *cred, struct file *fp)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
2000-07-04 03:34:11 +00:00
|
|
|
struct vnode *vp;
|
2000-07-11 22:07:57 +00:00
|
|
|
struct mount *mp;
|
2001-09-12 08:38:13 +00:00
|
|
|
struct thread *td = ndp->ni_cnd.cn_thread;
|
1994-05-24 10:09:53 +00:00
|
|
|
struct vattr vat;
|
|
|
|
struct vattr *vap = &vat;
|
2008-10-28 13:44:11 +00:00
|
|
|
int fmode, error;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2000-07-11 22:07:57 +00:00
|
|
|
restart:
|
2000-07-04 03:34:11 +00:00
|
|
|
fmode = *flagp;
|
2015-09-09 19:31:08 +00:00
|
|
|
if ((fmode & (O_CREAT | O_EXCL | O_DIRECTORY)) == (O_CREAT |
|
|
|
|
O_EXCL | O_DIRECTORY))
|
|
|
|
return (EINVAL);
|
|
|
|
else if ((fmode & (O_CREAT | O_DIRECTORY)) == O_CREAT) {
|
1994-05-24 10:09:53 +00:00
|
|
|
ndp->ni_cnd.cn_nameiop = CREATE;
|
2014-12-18 10:01:12 +00:00
|
|
|
/*
|
|
|
|
* Set NOCACHE to avoid flushing the cache when
|
|
|
|
* rolling in many files at once.
|
|
|
|
*/
|
|
|
|
ndp->ni_cnd.cn_flags = ISOPEN | LOCKPARENT | LOCKLEAF | NOCACHE;
|
1998-04-06 18:25:21 +00:00
|
|
|
if ((fmode & O_EXCL) == 0 && (fmode & O_NOFOLLOW) == 0)
|
1994-05-24 10:09:53 +00:00
|
|
|
ndp->ni_cnd.cn_flags |= FOLLOW;
|
2018-10-25 22:16:34 +00:00
|
|
|
if ((fmode & O_BENEATH) != 0)
|
|
|
|
ndp->ni_cnd.cn_flags |= BENEATH;
|
2009-06-21 13:41:32 +00:00
|
|
|
if (!(vn_open_flags & VN_OPEN_NOAUDIT))
|
|
|
|
ndp->ni_cnd.cn_flags |= AUDITVNODE1;
|
2012-11-27 10:32:35 +00:00
|
|
|
if (vn_open_flags & VN_OPEN_NOCAPCHECK)
|
|
|
|
ndp->ni_cnd.cn_flags |= NOCAPCHECK;
|
2019-11-29 14:02:32 +00:00
|
|
|
if ((vn_open_flags & VN_OPEN_INVFS) == 0)
|
|
|
|
bwillwrite();
|
2000-07-04 03:34:11 +00:00
|
|
|
if ((error = namei(ndp)) != 0)
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
|
|
|
if (ndp->ni_vp == NULL) {
|
|
|
|
VATTR_NULL(vap);
|
|
|
|
vap->va_type = VREG;
|
|
|
|
vap->va_mode = cmode;
|
1997-02-10 02:22:35 +00:00
|
|
|
if (fmode & O_EXCL)
|
|
|
|
vap->va_vaflags |= VA_EXCLUSIVE;
|
2000-07-11 22:07:57 +00:00
|
|
|
if (vn_start_write(ndp->ni_dvp, &mp, V_NOWAIT) != 0) {
|
|
|
|
NDFREE(ndp, NDF_ONLY_PNBUF);
|
|
|
|
vput(ndp->ni_dvp);
|
|
|
|
if ((error = vn_start_write(NULL, &mp,
|
|
|
|
V_XSLEEP | PCATCH)) != 0)
|
|
|
|
return (error);
|
|
|
|
goto restart;
|
|
|
|
}
|
2014-12-21 13:32:07 +00:00
|
|
|
if ((vn_open_flags & VN_OPEN_NAMECACHE) != 0)
|
|
|
|
ndp->ni_cnd.cn_flags |= MAKEENTRY;
|
2002-10-19 20:56:44 +00:00
|
|
|
#ifdef MAC
|
2007-10-24 19:04:04 +00:00
|
|
|
error = mac_vnode_check_create(cred, ndp->ni_dvp,
|
2002-10-19 20:56:44 +00:00
|
|
|
&ndp->ni_cnd, vap);
|
2009-04-10 10:52:19 +00:00
|
|
|
if (error == 0)
|
2002-10-19 20:56:44 +00:00
|
|
|
#endif
|
|
|
|
error = VOP_CREATE(ndp->ni_dvp, &ndp->ni_vp,
|
|
|
|
&ndp->ni_cnd, vap);
|
2000-07-04 03:34:11 +00:00
|
|
|
vput(ndp->ni_dvp);
|
2000-07-11 22:07:57 +00:00
|
|
|
vn_finished_write(mp);
|
1999-12-15 23:02:35 +00:00
|
|
|
if (error) {
|
|
|
|
NDFREE(ndp, NDF_ONLY_PNBUF);
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
1999-12-15 23:02:35 +00:00
|
|
|
}
|
1994-05-24 10:09:53 +00:00
|
|
|
fmode &= ~O_TRUNC;
|
|
|
|
vp = ndp->ni_vp;
|
|
|
|
} else {
|
|
|
|
if (ndp->ni_dvp == ndp->ni_vp)
|
|
|
|
vrele(ndp->ni_dvp);
|
|
|
|
else
|
|
|
|
vput(ndp->ni_dvp);
|
|
|
|
ndp->ni_dvp = NULL;
|
|
|
|
vp = ndp->ni_vp;
|
|
|
|
if (fmode & O_EXCL) {
|
|
|
|
error = EEXIST;
|
|
|
|
goto bad;
|
|
|
|
}
|
2019-09-17 18:32:18 +00:00
|
|
|
if (vp->v_type == VDIR) {
|
|
|
|
error = EISDIR;
|
|
|
|
goto bad;
|
|
|
|
}
|
1994-05-24 10:09:53 +00:00
|
|
|
fmode &= ~O_CREAT;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ndp->ni_cnd.cn_nameiop = LOOKUP;
|
2005-04-27 09:05:19 +00:00
|
|
|
ndp->ni_cnd.cn_flags = ISOPEN |
|
2012-10-22 17:50:54 +00:00
|
|
|
((fmode & O_NOFOLLOW) ? NOFOLLOW : FOLLOW) | LOCKLEAF;
|
2009-03-11 14:13:47 +00:00
|
|
|
if (!(fmode & FWRITE))
|
|
|
|
ndp->ni_cnd.cn_flags |= LOCKSHARED;
|
2018-10-25 22:16:34 +00:00
|
|
|
if ((fmode & O_BENEATH) != 0)
|
|
|
|
ndp->ni_cnd.cn_flags |= BENEATH;
|
2009-06-21 13:41:32 +00:00
|
|
|
if (!(vn_open_flags & VN_OPEN_NOAUDIT))
|
|
|
|
ndp->ni_cnd.cn_flags |= AUDITVNODE1;
|
2012-11-27 10:32:35 +00:00
|
|
|
if (vn_open_flags & VN_OPEN_NOCAPCHECK)
|
|
|
|
ndp->ni_cnd.cn_flags |= NOCAPCHECK;
|
2000-07-04 03:34:11 +00:00
|
|
|
if ((error = namei(ndp)) != 0)
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
|
|
|
vp = ndp->ni_vp;
|
|
|
|
}
|
2012-06-08 18:32:09 +00:00
|
|
|
error = vn_open_vnode(vp, fmode, cred, td, fp);
|
|
|
|
if (error)
|
2010-03-21 20:43:23 +00:00
|
|
|
goto bad;
|
2012-06-08 18:32:09 +00:00
|
|
|
*flagp = fmode;
|
|
|
|
return (0);
|
|
|
|
bad:
|
|
|
|
NDFREE(ndp, NDF_ONLY_PNBUF);
|
|
|
|
vput(vp);
|
|
|
|
*flagp = fmode;
|
|
|
|
ndp->ni_vp = NULL;
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
static int
|
|
|
|
vn_open_vnode_advlock(struct vnode *vp, int fmode, struct file *fp)
|
|
|
|
{
|
|
|
|
struct flock lf;
|
|
|
|
int error, lock_flags, type;
|
|
|
|
|
|
|
|
ASSERT_VOP_LOCKED(vp, "vn_open_vnode_advlock");
|
|
|
|
if ((fmode & (O_EXLOCK | O_SHLOCK)) == 0)
|
|
|
|
return (0);
|
|
|
|
KASSERT(fp != NULL, ("open with flock requires fp"));
|
|
|
|
if (fp->f_type != DTYPE_NONE && fp->f_type != DTYPE_VNODE)
|
|
|
|
return (EOPNOTSUPP);
|
|
|
|
|
|
|
|
lock_flags = VOP_ISLOCKED(vp);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
|
|
|
|
lf.l_whence = SEEK_SET;
|
|
|
|
lf.l_start = 0;
|
|
|
|
lf.l_len = 0;
|
|
|
|
lf.l_type = (fmode & O_EXLOCK) != 0 ? F_WRLCK : F_RDLCK;
|
|
|
|
type = F_FLOCK;
|
|
|
|
if ((fmode & FNONBLOCK) == 0)
|
|
|
|
type |= F_WAIT;
|
|
|
|
error = VOP_ADVLOCK(vp, (caddr_t)fp, F_SETLK, &lf, type);
|
|
|
|
if (error == 0)
|
|
|
|
fp->f_flag |= FHASLOCK;
|
|
|
|
|
|
|
|
vn_lock(vp, lock_flags | LK_RETRY);
|
2019-12-08 21:30:04 +00:00
|
|
|
if (error == 0 && VN_IS_DOOMED(vp))
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
error = ENOENT;
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2012-06-08 18:32:09 +00:00
|
|
|
/*
|
|
|
|
* Common code for vnode open operations once a vnode is located.
|
|
|
|
* Check permissions, and call the VOP_OPEN routine.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
vn_open_vnode(struct vnode *vp, int fmode, struct ucred *cred,
|
|
|
|
struct thread *td, struct file *fp)
|
|
|
|
{
|
|
|
|
accmode_t accmode;
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
int error;
|
2012-06-08 18:32:09 +00:00
|
|
|
|
|
|
|
if (vp->v_type == VLNK)
|
|
|
|
return (EMLINK);
|
|
|
|
if (vp->v_type == VSOCK)
|
|
|
|
return (EOPNOTSUPP);
|
|
|
|
if (vp->v_type != VDIR && fmode & O_DIRECTORY)
|
|
|
|
return (ENOTDIR);
|
2008-10-28 13:44:11 +00:00
|
|
|
accmode = 0;
|
2002-08-01 17:14:28 +00:00
|
|
|
if (fmode & (FWRITE | O_TRUNC)) {
|
2012-06-08 18:32:09 +00:00
|
|
|
if (vp->v_type == VDIR)
|
|
|
|
return (EISDIR);
|
2008-10-28 13:44:11 +00:00
|
|
|
accmode |= VWRITE;
|
2002-08-01 17:14:28 +00:00
|
|
|
}
|
|
|
|
if (fmode & FREAD)
|
2008-10-28 13:44:11 +00:00
|
|
|
accmode |= VREAD;
|
2008-03-31 11:57:18 +00:00
|
|
|
if (fmode & FEXEC)
|
2008-10-28 13:44:11 +00:00
|
|
|
accmode |= VEXEC;
|
2009-12-08 20:47:10 +00:00
|
|
|
if ((fmode & O_APPEND) && (fmode & FWRITE))
|
2009-11-04 07:14:16 +00:00
|
|
|
accmode |= VAPPEND;
|
2002-08-01 17:14:28 +00:00
|
|
|
#ifdef MAC
|
2015-04-22 01:54:25 +00:00
|
|
|
if (fmode & O_CREAT)
|
|
|
|
accmode |= VCREAT;
|
|
|
|
if (fmode & O_VERIFY)
|
|
|
|
accmode |= VVERIFY;
|
2008-10-28 13:44:11 +00:00
|
|
|
error = mac_vnode_check_open(cred, vp, accmode);
|
2002-08-01 17:14:28 +00:00
|
|
|
if (error)
|
2012-06-08 18:32:09 +00:00
|
|
|
return (error);
|
2015-04-22 01:54:25 +00:00
|
|
|
|
|
|
|
accmode &= ~(VCREAT | VVERIFY);
|
2002-08-01 17:14:28 +00:00
|
|
|
#endif
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
if ((fmode & O_CREAT) == 0 && accmode != 0) {
|
|
|
|
error = VOP_ACCESS(vp, accmode, cred, td);
|
|
|
|
if (error != 0)
|
|
|
|
return (error);
|
1994-05-24 10:09:53 +00:00
|
|
|
}
|
2013-09-13 06:52:23 +00:00
|
|
|
if (vp->v_type == VFIFO && VOP_ISLOCKED(vp) != LK_EXCLUSIVE)
|
|
|
|
vn_lock(vp, LK_UPGRADE | LK_RETRY);
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
error = VOP_OPEN(vp, fmode, cred, td, fp);
|
|
|
|
if (error != 0)
|
2012-06-08 18:32:09 +00:00
|
|
|
return (error);
|
1996-08-21 21:56:23 +00:00
|
|
|
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
error = vn_open_vnode_advlock(vp, fmode, fp);
|
|
|
|
if (error == 0 && (fmode & FWRITE) != 0) {
|
|
|
|
error = VOP_ADD_WRITECOUNT(vp, 1);
|
|
|
|
if (error == 0) {
|
|
|
|
CTR3(KTR_VFS, "%s: vp %p v_writecount increased to %d",
|
|
|
|
__func__, vp, vp->v_writecount);
|
2017-02-09 23:35:57 +00:00
|
|
|
}
|
|
|
|
}
|
2016-01-17 08:40:51 +00:00
|
|
|
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
/*
|
|
|
|
* Error from advlock or VOP_ADD_WRITECOUNT() still requires
|
|
|
|
* calling VOP_CLOSE() to pair with earlier VOP_OPEN().
|
|
|
|
* Arrange for that by having fdrop() to use vn_closefile().
|
|
|
|
*/
|
2017-02-09 23:35:57 +00:00
|
|
|
if (error != 0) {
|
|
|
|
fp->f_flag |= FOPENFAILED;
|
|
|
|
fp->f_vnode = vp;
|
|
|
|
if (fp->f_ops == &badfileops) {
|
|
|
|
fp->f_type = DTYPE_VNODE;
|
|
|
|
fp->f_ops = &vnops;
|
2012-07-31 18:25:00 +00:00
|
|
|
}
|
2017-02-09 23:35:57 +00:00
|
|
|
vref(vp);
|
2012-03-08 20:27:20 +00:00
|
|
|
}
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
|
2012-06-08 18:32:09 +00:00
|
|
|
ASSERT_VOP_LOCKED(vp, "vn_open_vnode");
|
2016-01-17 08:40:51 +00:00
|
|
|
return (error);
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check for write permissions on the specified vnode.
|
1997-02-10 02:22:35 +00:00
|
|
|
* Prototype text segments cannot be written.
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
* It is racy.
|
1994-05-24 10:09:53 +00:00
|
|
|
*/
|
1994-05-25 09:21:21 +00:00
|
|
|
int
|
2017-05-17 00:34:34 +00:00
|
|
|
vn_writechk(struct vnode *vp)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
|
|
|
|
2002-08-04 10:29:36 +00:00
|
|
|
ASSERT_VOP_LOCKED(vp, "vn_writechk");
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* If there's shared text associated with
|
|
|
|
* the vnode, try to free it up once. If
|
|
|
|
* we fail, we can't allow writing.
|
|
|
|
*/
|
2012-09-28 11:25:02 +00:00
|
|
|
if (VOP_IS_TEXT(vp))
|
1994-05-24 10:09:53 +00:00
|
|
|
return (ETXTBSY);
|
2002-08-04 10:29:36 +00:00
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Vnode close call
|
|
|
|
*/
|
2017-02-09 23:36:50 +00:00
|
|
|
static int
|
|
|
|
vn_close1(struct vnode *vp, int flags, struct ucred *file_cred,
|
|
|
|
struct thread *td, bool keep_ref)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
2005-03-13 11:56:28 +00:00
|
|
|
struct mount *mp;
|
2009-03-11 14:13:47 +00:00
|
|
|
int error, lock_flags;
|
|
|
|
|
2013-11-09 20:30:13 +00:00
|
|
|
if (vp->v_type != VFIFO && (flags & FWRITE) == 0 &&
|
|
|
|
MNT_EXTENDED_SHARED(vp->v_mount))
|
2009-03-11 14:13:47 +00:00
|
|
|
lock_flags = LK_SHARED;
|
|
|
|
else
|
|
|
|
lock_flags = LK_EXCLUSIVE;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2005-03-13 11:56:28 +00:00
|
|
|
vn_start_write(vp, &mp, V_WAIT);
|
2009-03-11 14:13:47 +00:00
|
|
|
vn_lock(vp, lock_flags | LK_RETRY);
|
2016-07-05 16:37:01 +00:00
|
|
|
AUDIT_ARG_VNODE1(vp);
|
2016-01-17 08:40:51 +00:00
|
|
|
if ((flags & (FWRITE | FOPENFAILED)) == FWRITE) {
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
VOP_ADD_WRITECOUNT_CHECKED(vp, -1);
|
2012-03-08 20:27:20 +00:00
|
|
|
CTR3(KTR_VFS, "%s: vp %p v_writecount decreased to %d",
|
|
|
|
__func__, vp, vp->v_writecount);
|
2007-02-12 22:53:01 +00:00
|
|
|
}
|
Make similar changes to fo_stat() and fo_poll() as made earlier to
fo_read() and fo_write(): explicitly use the cred argument to fo_poll()
as "active_cred" using the passed file descriptor's f_cred reference
to provide access to the file credential. Add an active_cred
argument to fo_stat() so that implementers have access to the active
credential as well as the file credential. Generally modify callers
of fo_stat() to pass in td->td_ucred rather than fp->f_cred, which
was redundantly provided via the fp argument. This set of modifications
also permits threads to perform these operations on behalf of another
thread without modifying their credential.
Trickle this change down into fo_stat/poll() implementations:
- badfo_poll(), badfo_stat(): modify/add arguments.
- kqueue_poll(), kqueue_stat(): modify arguments.
- pipe_poll(), pipe_stat(): modify/add arguments, pass active_cred to
MAC checks rather than td->td_ucred.
- soo_poll(), soo_stat(): modify/add arguments, pass fp->f_cred rather
than cred to pru_sopoll() to maintain current semantics.
- sopoll(): moidfy arguments.
- vn_poll(), vn_statfile(): modify/add arguments, pass new arguments
to vn_stat(). Pass active_cred to MAC and fp->f_cred to VOP_POLL()
to maintian current semantics.
- vn_close(): rename cred to file_cred to reflect reality while I'm here.
- vn_stat(): Add active_cred and file_cred arguments to vn_stat()
and consumers so that this distinction is maintained at the VFS
as well as 'struct file' layer. Pass active_cred instead of
td->td_ucred to MAC and to VOP_GETATTR() to maintain current semantics.
- fifofs: modify the creation of a "filetemp" so that the file
credential is properly initialized and can be used in the socket
code if desired. Pass ap->a_td->td_ucred as the active
credential to soo_poll(). If we teach the vnop interface about
the distinction between file and active credentials, we would use
the active credential here.
Note that current inconsistent passing of active_cred vs. file_cred to
VOP's is maintained. It's not clear why GETATTR would be authorized
using active_cred while POLL would be authorized using file_cred at
the file system level.
Obtained from: TrustedBSD Project
Sponsored by: DARPA, NAI Labs
2002-08-16 12:52:03 +00:00
|
|
|
error = VOP_CLOSE(vp, flags, file_cred, td);
|
2017-02-09 23:36:50 +00:00
|
|
|
if (keep_ref)
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2017-02-09 23:36:50 +00:00
|
|
|
else
|
|
|
|
vput(vp);
|
2005-03-13 11:56:28 +00:00
|
|
|
vn_finished_write(mp);
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2017-02-09 23:36:50 +00:00
|
|
|
int
|
|
|
|
vn_close(struct vnode *vp, int flags, struct ucred *file_cred,
|
|
|
|
struct thread *td)
|
|
|
|
{
|
|
|
|
|
|
|
|
return (vn_close1(vp, flags, file_cred, td, false));
|
|
|
|
}
|
|
|
|
|
2002-03-05 15:38:49 +00:00
|
|
|
/*
|
2008-01-05 08:54:51 +00:00
|
|
|
* Heuristic to detect sequential operation.
|
2002-03-05 15:38:49 +00:00
|
|
|
*/
|
2008-01-05 08:54:51 +00:00
|
|
|
static int
|
2000-04-02 00:55:28 +00:00
|
|
|
sequential_heuristic(struct uio *uio, struct file *fp)
|
|
|
|
{
|
2002-01-13 11:58:06 +00:00
|
|
|
|
2014-08-26 08:17:22 +00:00
|
|
|
ASSERT_VOP_LOCKED(fp->f_vnode, __func__);
|
|
|
|
if (fp->f_flag & FRDAHEAD)
|
2009-09-28 16:59:47 +00:00
|
|
|
return (fp->f_seqcount << IO_SEQSHIFT);
|
|
|
|
|
2008-01-05 08:54:51 +00:00
|
|
|
/*
|
|
|
|
* Offset 0 is handled specially. open() sets f_seqcount to 1 so
|
|
|
|
* that the first I/O is normally considered to be slightly
|
|
|
|
* sequential. Seeking to offset 0 doesn't change sequentiality
|
|
|
|
* unless previous seeks have reduced f_seqcount to 0, in which
|
|
|
|
* case offset 0 is not special.
|
|
|
|
*/
|
2000-04-02 00:55:28 +00:00
|
|
|
if ((uio->uio_offset == 0 && fp->f_seqcount > 0) ||
|
|
|
|
uio->uio_offset == fp->f_nextoff) {
|
|
|
|
/*
|
2008-01-05 08:54:51 +00:00
|
|
|
* f_seqcount is in units of fixed-size blocks so that it
|
|
|
|
* depends mainly on the amount of sequential I/O and not
|
|
|
|
* much on the number of sequential I/O's. The fixed size
|
|
|
|
* of 16384 is hard-coded here since it is (not quite) just
|
|
|
|
* a magic size that works well here. This size is more
|
|
|
|
* closely related to the best I/O size for real disks than
|
|
|
|
* to any block size used by software.
|
2000-04-02 00:55:28 +00:00
|
|
|
*/
|
2019-07-17 17:01:07 +00:00
|
|
|
if (uio->uio_resid >= IO_SEQMAX * 16384)
|
|
|
|
fp->f_seqcount = IO_SEQMAX;
|
|
|
|
else {
|
|
|
|
fp->f_seqcount += howmany(uio->uio_resid, 16384);
|
|
|
|
if (fp->f_seqcount > IO_SEQMAX)
|
|
|
|
fp->f_seqcount = IO_SEQMAX;
|
|
|
|
}
|
2008-01-05 08:54:51 +00:00
|
|
|
return (fp->f_seqcount << IO_SEQSHIFT);
|
2000-04-02 00:55:28 +00:00
|
|
|
}
|
|
|
|
|
2008-01-05 08:54:51 +00:00
|
|
|
/* Not sequential. Quickly draw-down sequentiality. */
|
2000-04-02 00:55:28 +00:00
|
|
|
if (fp->f_seqcount > 1)
|
|
|
|
fp->f_seqcount = 1;
|
|
|
|
else
|
|
|
|
fp->f_seqcount = 0;
|
2008-01-05 08:54:51 +00:00
|
|
|
return (0);
|
2000-04-02 00:55:28 +00:00
|
|
|
}
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Package up an I/O request on a vnode into a uio and do it.
|
|
|
|
*/
|
1994-05-25 09:21:21 +00:00
|
|
|
int
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
vn_rdwr(enum uio_rw rw, struct vnode *vp, void *base, int len, off_t offset,
|
|
|
|
enum uio_seg segflg, int ioflg, struct ucred *active_cred,
|
|
|
|
struct ucred *file_cred, ssize_t *aresid, struct thread *td)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
|
|
|
struct uio auio;
|
|
|
|
struct iovec aiov;
|
2000-07-11 22:07:57 +00:00
|
|
|
struct mount *mp;
|
In order to better support flexible and extensible access control,
make a series of modifications to the credential arguments relating
to file read and write operations to cliarfy which credential is
used for what:
- Change fo_read() and fo_write() to accept "active_cred" instead of
"cred", and change the semantics of consumers of fo_read() and
fo_write() to pass the active credential of the thread requesting
an operation rather than the cached file cred. The cached file
cred is still available in fo_read() and fo_write() consumers
via fp->f_cred. These changes largely in sys_generic.c.
For each implementation of fo_read() and fo_write(), update cred
usage to reflect this change and maintain current semantics:
- badfo_readwrite() unchanged
- kqueue_read/write() unchanged
pipe_read/write() now authorize MAC using active_cred rather
than td->td_ucred
- soo_read/write() unchanged
- vn_read/write() now authorize MAC using active_cred but
VOP_READ/WRITE() with fp->f_cred
Modify vn_rdwr() to accept two credential arguments instead of a
single credential: active_cred and file_cred. Use active_cred
for MAC authorization, and select a credential for use in
VOP_READ/WRITE() based on whether file_cred is NULL or not. If
file_cred is provided, authorize the VOP using that cred,
otherwise the active credential, matching current semantics.
Modify current vn_rdwr() consumers to pass a file_cred if used
in the context of a struct file, and to always pass active_cred.
When vn_rdwr() is used without a file_cred, pass NOCRED.
These changes should maintain current semantics for read/write,
but avoid a redundant passing of fp->f_cred, as well as making
it more clear what the origin of each credential is in file
descriptor read/write operations.
Follow-up commits will make similar changes to other file descriptor
operations, and modify the MAC framework to pass both credentials
to MAC policy modules so they can implement either semantic for
revocation.
Obtained from: TrustedBSD Project
Sponsored by: DARPA, NAI Labs
2002-08-15 20:55:08 +00:00
|
|
|
struct ucred *cred;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
void *rl_cookie;
|
2014-06-15 04:51:53 +00:00
|
|
|
struct vn_io_fault_args args;
|
2009-06-04 16:18:07 +00:00
|
|
|
int error, lock_flags;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2018-09-12 04:57:34 +00:00
|
|
|
if (offset < 0 && vp->v_type != VCHR)
|
|
|
|
return (EINVAL);
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
auio.uio_iov = &aiov;
|
|
|
|
auio.uio_iovcnt = 1;
|
|
|
|
aiov.iov_base = base;
|
|
|
|
aiov.iov_len = len;
|
|
|
|
auio.uio_resid = len;
|
|
|
|
auio.uio_offset = offset;
|
|
|
|
auio.uio_segflg = segflg;
|
|
|
|
auio.uio_rw = rw;
|
|
|
|
auio.uio_td = td;
|
|
|
|
error = 0;
|
|
|
|
|
2000-07-11 22:07:57 +00:00
|
|
|
if ((ioflg & IO_NODELOCKED) == 0) {
|
2014-10-04 18:28:27 +00:00
|
|
|
if ((ioflg & IO_RANGELOCKED) == 0) {
|
|
|
|
if (rw == UIO_READ) {
|
|
|
|
rl_cookie = vn_rangelock_rlock(vp, offset,
|
|
|
|
offset + len);
|
|
|
|
} else {
|
|
|
|
rl_cookie = vn_rangelock_wlock(vp, offset,
|
|
|
|
offset + len);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
rl_cookie = NULL;
|
2000-07-11 22:07:57 +00:00
|
|
|
mp = NULL;
|
2002-06-28 17:51:11 +00:00
|
|
|
if (rw == UIO_WRITE) {
|
|
|
|
if (vp->v_type != VCHR &&
|
|
|
|
(error = vn_start_write(vp, &mp, V_WAIT | PCATCH))
|
|
|
|
!= 0)
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
goto out;
|
2009-06-08 21:23:54 +00:00
|
|
|
if (MNT_SHARED_WRITES(mp) ||
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
((mp == NULL) && MNT_SHARED_WRITES(vp->v_mount)))
|
2009-06-04 16:18:07 +00:00
|
|
|
lock_flags = LK_SHARED;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
else
|
2009-06-04 16:18:07 +00:00
|
|
|
lock_flags = LK_EXCLUSIVE;
|
2009-04-13 23:09:44 +00:00
|
|
|
} else
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
lock_flags = LK_SHARED;
|
|
|
|
vn_lock(vp, lock_flags | LK_RETRY);
|
|
|
|
} else
|
|
|
|
rl_cookie = NULL;
|
2002-06-28 17:51:11 +00:00
|
|
|
|
2005-04-05 01:11:43 +00:00
|
|
|
ASSERT_VOP_LOCKED(vp, "IO_NODELOCKED with no vp lock held");
|
2002-08-12 16:15:34 +00:00
|
|
|
#ifdef MAC
|
|
|
|
if ((ioflg & IO_NOMACCHECK) == 0) {
|
|
|
|
if (rw == UIO_READ)
|
2007-10-24 19:04:04 +00:00
|
|
|
error = mac_vnode_check_read(active_cred, file_cred,
|
2002-08-19 19:04:53 +00:00
|
|
|
vp);
|
2002-08-12 16:15:34 +00:00
|
|
|
else
|
2007-10-24 19:04:04 +00:00
|
|
|
error = mac_vnode_check_write(active_cred, file_cred,
|
2002-08-19 19:04:53 +00:00
|
|
|
vp);
|
2002-08-12 16:15:34 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (error == 0) {
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
if (file_cred != NULL)
|
In order to better support flexible and extensible access control,
make a series of modifications to the credential arguments relating
to file read and write operations to cliarfy which credential is
used for what:
- Change fo_read() and fo_write() to accept "active_cred" instead of
"cred", and change the semantics of consumers of fo_read() and
fo_write() to pass the active credential of the thread requesting
an operation rather than the cached file cred. The cached file
cred is still available in fo_read() and fo_write() consumers
via fp->f_cred. These changes largely in sys_generic.c.
For each implementation of fo_read() and fo_write(), update cred
usage to reflect this change and maintain current semantics:
- badfo_readwrite() unchanged
- kqueue_read/write() unchanged
pipe_read/write() now authorize MAC using active_cred rather
than td->td_ucred
- soo_read/write() unchanged
- vn_read/write() now authorize MAC using active_cred but
VOP_READ/WRITE() with fp->f_cred
Modify vn_rdwr() to accept two credential arguments instead of a
single credential: active_cred and file_cred. Use active_cred
for MAC authorization, and select a credential for use in
VOP_READ/WRITE() based on whether file_cred is NULL or not. If
file_cred is provided, authorize the VOP using that cred,
otherwise the active credential, matching current semantics.
Modify current vn_rdwr() consumers to pass a file_cred if used
in the context of a struct file, and to always pass active_cred.
When vn_rdwr() is used without a file_cred, pass NOCRED.
These changes should maintain current semantics for read/write,
but avoid a redundant passing of fp->f_cred, as well as making
it more clear what the origin of each credential is in file
descriptor read/write operations.
Follow-up commits will make similar changes to other file descriptor
operations, and modify the MAC framework to pass both credentials
to MAC policy modules so they can implement either semantic for
revocation.
Obtained from: TrustedBSD Project
Sponsored by: DARPA, NAI Labs
2002-08-15 20:55:08 +00:00
|
|
|
cred = file_cred;
|
|
|
|
else
|
|
|
|
cred = active_cred;
|
2014-06-15 04:51:53 +00:00
|
|
|
if (do_vn_io_fault(vp, &auio)) {
|
|
|
|
args.kind = VN_IO_FAULT_VOP;
|
|
|
|
args.cred = cred;
|
|
|
|
args.flags = ioflg;
|
|
|
|
args.args.vop_args.vp = vp;
|
|
|
|
error = vn_io_fault1(vp, &auio, &args, td);
|
|
|
|
} else if (rw == UIO_READ) {
|
2002-08-12 16:15:34 +00:00
|
|
|
error = VOP_READ(vp, &auio, ioflg, cred);
|
2014-06-15 04:51:53 +00:00
|
|
|
} else /* if (rw == UIO_WRITE) */ {
|
2002-08-12 16:15:34 +00:00
|
|
|
error = VOP_WRITE(vp, &auio, ioflg, cred);
|
2014-06-15 04:51:53 +00:00
|
|
|
}
|
2002-08-12 16:15:34 +00:00
|
|
|
}
|
1994-05-24 10:09:53 +00:00
|
|
|
if (aresid)
|
|
|
|
*aresid = auio.uio_resid;
|
|
|
|
else
|
|
|
|
if (auio.uio_resid && error == 0)
|
|
|
|
error = EIO;
|
2000-07-11 22:07:57 +00:00
|
|
|
if ((ioflg & IO_NODELOCKED) == 0) {
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
if (mp != NULL)
|
|
|
|
vn_finished_write(mp);
|
2000-07-11 22:07:57 +00:00
|
|
|
}
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
out:
|
|
|
|
if (rl_cookie != NULL)
|
|
|
|
vn_rangelock_unlock(vp, rl_cookie);
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2001-09-08 20:02:33 +00:00
|
|
|
/*
|
|
|
|
* Package up an I/O request on a vnode into a uio and do it. The I/O
|
|
|
|
* request is split up into smaller chunks and we try to avoid saturating
|
|
|
|
* the buffer cache while potentially holding a vnode locked, so we
|
2011-02-08 00:16:36 +00:00
|
|
|
* check bwillwrite() before calling vn_rdwr(). We also call kern_yield()
|
2001-09-26 06:54:32 +00:00
|
|
|
* to give other processes a chance to lock the vnode (either other processes
|
|
|
|
* core'ing the same binary, or unrelated processes scanning the directory).
|
2001-09-08 20:02:33 +00:00
|
|
|
*/
|
|
|
|
int
|
2018-06-01 13:26:45 +00:00
|
|
|
vn_rdwr_inchunks(enum uio_rw rw, struct vnode *vp, void *base, size_t len,
|
|
|
|
off_t offset, enum uio_seg segflg, int ioflg, struct ucred *active_cred,
|
|
|
|
struct ucred *file_cred, size_t *aresid, struct thread *td)
|
2001-09-08 20:02:33 +00:00
|
|
|
{
|
|
|
|
int error = 0;
|
2012-02-21 01:05:12 +00:00
|
|
|
ssize_t iaresid;
|
2001-09-08 20:02:33 +00:00
|
|
|
|
|
|
|
do {
|
2004-03-13 02:56:27 +00:00
|
|
|
int chunk;
|
2001-09-08 20:02:33 +00:00
|
|
|
|
2004-03-13 02:56:27 +00:00
|
|
|
/*
|
|
|
|
* Force `offset' to a multiple of MAXBSIZE except possibly
|
|
|
|
* for the first chunk, so that filesystems only need to
|
|
|
|
* write full blocks except possibly for the first and last
|
|
|
|
* chunks.
|
|
|
|
*/
|
|
|
|
chunk = MAXBSIZE - (uoff_t)offset % MAXBSIZE;
|
|
|
|
|
|
|
|
if (chunk > len)
|
|
|
|
chunk = len;
|
2001-09-08 20:02:33 +00:00
|
|
|
if (rw != UIO_READ && vp->v_type == VREG)
|
|
|
|
bwillwrite();
|
2004-06-05 02:18:28 +00:00
|
|
|
iaresid = 0;
|
2001-09-08 20:02:33 +00:00
|
|
|
error = vn_rdwr(rw, vp, base, chunk, offset, segflg,
|
2004-06-05 02:18:28 +00:00
|
|
|
ioflg, active_cred, file_cred, &iaresid, td);
|
2001-09-08 20:02:33 +00:00
|
|
|
len -= chunk; /* aresid calc already includes length */
|
|
|
|
if (error)
|
|
|
|
break;
|
|
|
|
offset += chunk;
|
2005-12-14 00:49:52 +00:00
|
|
|
base = (char *)base + chunk;
|
2011-05-13 05:27:58 +00:00
|
|
|
kern_yield(PRI_USER);
|
2001-09-08 20:02:33 +00:00
|
|
|
} while (len);
|
|
|
|
if (aresid)
|
2004-06-05 02:18:28 +00:00
|
|
|
*aresid = len + iaresid;
|
2001-09-08 20:02:33 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2012-07-02 21:01:03 +00:00
|
|
|
off_t
|
|
|
|
foffset_lock(struct file *fp, int flags)
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
{
|
|
|
|
struct mtx *mtxp;
|
2012-07-02 21:01:03 +00:00
|
|
|
off_t res;
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
|
2012-07-02 21:01:03 +00:00
|
|
|
KASSERT((flags & FOF_OFFSET) == 0, ("FOF_OFFSET passed"));
|
|
|
|
|
|
|
|
#if OFF_MAX <= LONG_MAX
|
|
|
|
/*
|
|
|
|
* Caller only wants the current f_offset value. Assume that
|
|
|
|
* the long and shorter integer types reads are atomic.
|
|
|
|
*/
|
|
|
|
if ((flags & FOF_NOLOCK) != 0)
|
|
|
|
return (fp->f_offset);
|
|
|
|
#endif
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* According to McKusick the vn lock was protecting f_offset here.
|
|
|
|
* It is now protected by the FOFFSET_LOCKED flag.
|
|
|
|
*/
|
|
|
|
mtxp = mtx_pool_find(mtxpool_sleep, fp);
|
|
|
|
mtx_lock(mtxp);
|
2012-07-02 21:01:03 +00:00
|
|
|
if ((flags & FOF_NOLOCK) == 0) {
|
|
|
|
while (fp->f_vnread_flags & FOFFSET_LOCKED) {
|
|
|
|
fp->f_vnread_flags |= FOFFSET_LOCK_WAITING;
|
|
|
|
msleep(&fp->f_vnread_flags, mtxp, PUSER -1,
|
|
|
|
"vofflock", 0);
|
|
|
|
}
|
|
|
|
fp->f_vnread_flags |= FOFFSET_LOCKED;
|
|
|
|
}
|
|
|
|
res = fp->f_offset;
|
|
|
|
mtx_unlock(mtxp);
|
|
|
|
return (res);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
foffset_unlock(struct file *fp, off_t val, int flags)
|
|
|
|
{
|
|
|
|
struct mtx *mtxp;
|
|
|
|
|
|
|
|
KASSERT((flags & FOF_OFFSET) == 0, ("FOF_OFFSET passed"));
|
|
|
|
|
|
|
|
#if OFF_MAX <= LONG_MAX
|
|
|
|
if ((flags & FOF_NOLOCK) != 0) {
|
|
|
|
if ((flags & FOF_NOUPDATE) == 0)
|
|
|
|
fp->f_offset = val;
|
|
|
|
if ((flags & FOF_NEXTOFF) != 0)
|
|
|
|
fp->f_nextoff = val;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
mtxp = mtx_pool_find(mtxpool_sleep, fp);
|
|
|
|
mtx_lock(mtxp);
|
|
|
|
if ((flags & FOF_NOUPDATE) == 0)
|
|
|
|
fp->f_offset = val;
|
|
|
|
if ((flags & FOF_NEXTOFF) != 0)
|
|
|
|
fp->f_nextoff = val;
|
|
|
|
if ((flags & FOF_NOLOCK) == 0) {
|
|
|
|
KASSERT((fp->f_vnread_flags & FOFFSET_LOCKED) != 0,
|
|
|
|
("Lost FOFFSET_LOCKED"));
|
|
|
|
if (fp->f_vnread_flags & FOFFSET_LOCK_WAITING)
|
|
|
|
wakeup(&fp->f_vnread_flags);
|
|
|
|
fp->f_vnread_flags = 0;
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
}
|
|
|
|
mtx_unlock(mtxp);
|
|
|
|
}
|
|
|
|
|
2012-07-02 21:01:03 +00:00
|
|
|
void
|
|
|
|
foffset_lock_uio(struct file *fp, struct uio *uio, int flags)
|
|
|
|
{
|
|
|
|
|
|
|
|
if ((flags & FOF_OFFSET) == 0)
|
|
|
|
uio->uio_offset = foffset_lock(fp, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
foffset_unlock_uio(struct file *fp, struct uio *uio, int flags)
|
|
|
|
{
|
|
|
|
|
|
|
|
if ((flags & FOF_OFFSET) == 0)
|
|
|
|
foffset_unlock(fp, uio->uio_offset, flags);
|
|
|
|
}
|
|
|
|
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
static int
|
|
|
|
get_advice(struct file *fp, struct uio *uio)
|
|
|
|
{
|
|
|
|
struct mtx *mtxp;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = POSIX_FADV_NORMAL;
|
2016-01-22 20:35:20 +00:00
|
|
|
if (fp->f_advice == NULL || fp->f_vnode->v_type != VREG)
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
return (ret);
|
|
|
|
|
|
|
|
mtxp = mtx_pool_find(mtxpool_sleep, fp);
|
|
|
|
mtx_lock(mtxp);
|
2016-01-22 20:35:20 +00:00
|
|
|
if (fp->f_advice != NULL &&
|
|
|
|
uio->uio_offset >= fp->f_advice->fa_start &&
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
uio->uio_offset + uio->uio_resid <= fp->f_advice->fa_end)
|
|
|
|
ret = fp->f_advice->fa_advice;
|
|
|
|
mtx_unlock(mtxp);
|
|
|
|
return (ret);
|
|
|
|
}
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* File table vnode read routine.
|
|
|
|
*/
|
1995-12-17 21:23:44 +00:00
|
|
|
static int
|
2018-06-01 13:26:45 +00:00
|
|
|
vn_read(struct file *fp, struct uio *uio, struct ucred *active_cred, int flags,
|
|
|
|
struct thread *td)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
This is what was "fdfix2.patch," a fix for fd sharing. It's pretty
far-reaching in fd-land, so you'll want to consult the code for
changes. The biggest change is that now, you don't use
fp->f_ops->fo_foo(fp, bar)
but instead
fo_foo(fp, bar),
which increments and decrements the fp refcount upon entry and exit.
Two new calls, fhold() and fdrop(), are provided. Each does what it
seems like it should, and if fdrop() brings the refcount to zero, the
fd is freed as well.
Thanks to peter ("to hell with it, it looks ok to me.") for his review.
Thanks to msmith for keeping me from putting locks everywhere :)
Reviewed by: peter
1999-09-19 17:00:25 +00:00
|
|
|
struct vnode *vp;
|
2015-09-30 23:06:29 +00:00
|
|
|
off_t orig_offset;
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
int error, ioflag;
|
2012-10-22 17:50:54 +00:00
|
|
|
int advice;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
KASSERT(uio->uio_td == td, ("uio_td %p is not td %p",
|
|
|
|
uio->uio_td, td));
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
KASSERT(flags & FOF_OFFSET, ("No FOF_OFFSET"));
|
2003-06-22 08:41:43 +00:00
|
|
|
vp = fp->f_vnode;
|
1999-04-21 05:56:45 +00:00
|
|
|
ioflag = 0;
|
|
|
|
if (fp->f_flag & FNONBLOCK)
|
|
|
|
ioflag |= IO_NDELAY;
|
2001-05-24 07:22:27 +00:00
|
|
|
if (fp->f_flag & O_DIRECT)
|
|
|
|
ioflag |= IO_DIRECT;
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
advice = get_advice(fp, uio);
|
2011-11-04 04:02:50 +00:00
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
|
|
|
|
|
|
|
switch (advice) {
|
|
|
|
case POSIX_FADV_NORMAL:
|
|
|
|
case POSIX_FADV_SEQUENTIAL:
|
2012-01-30 19:35:15 +00:00
|
|
|
case POSIX_FADV_NOREUSE:
|
2011-11-04 04:02:50 +00:00
|
|
|
ioflag |= sequential_heuristic(uio, fp);
|
|
|
|
break;
|
|
|
|
case POSIX_FADV_RANDOM:
|
|
|
|
/* Disable read-ahead for random I/O. */
|
|
|
|
break;
|
|
|
|
}
|
2015-09-30 23:06:29 +00:00
|
|
|
orig_offset = uio->uio_offset;
|
2000-04-02 00:55:28 +00:00
|
|
|
|
2002-08-01 17:23:22 +00:00
|
|
|
#ifdef MAC
|
2007-10-24 19:04:04 +00:00
|
|
|
error = mac_vnode_check_read(active_cred, fp->f_cred, vp);
|
2002-08-01 17:23:22 +00:00
|
|
|
if (error == 0)
|
|
|
|
#endif
|
In order to better support flexible and extensible access control,
make a series of modifications to the credential arguments relating
to file read and write operations to cliarfy which credential is
used for what:
- Change fo_read() and fo_write() to accept "active_cred" instead of
"cred", and change the semantics of consumers of fo_read() and
fo_write() to pass the active credential of the thread requesting
an operation rather than the cached file cred. The cached file
cred is still available in fo_read() and fo_write() consumers
via fp->f_cred. These changes largely in sys_generic.c.
For each implementation of fo_read() and fo_write(), update cred
usage to reflect this change and maintain current semantics:
- badfo_readwrite() unchanged
- kqueue_read/write() unchanged
pipe_read/write() now authorize MAC using active_cred rather
than td->td_ucred
- soo_read/write() unchanged
- vn_read/write() now authorize MAC using active_cred but
VOP_READ/WRITE() with fp->f_cred
Modify vn_rdwr() to accept two credential arguments instead of a
single credential: active_cred and file_cred. Use active_cred
for MAC authorization, and select a credential for use in
VOP_READ/WRITE() based on whether file_cred is NULL or not. If
file_cred is provided, authorize the VOP using that cred,
otherwise the active credential, matching current semantics.
Modify current vn_rdwr() consumers to pass a file_cred if used
in the context of a struct file, and to always pass active_cred.
When vn_rdwr() is used without a file_cred, pass NOCRED.
These changes should maintain current semantics for read/write,
but avoid a redundant passing of fp->f_cred, as well as making
it more clear what the origin of each credential is in file
descriptor read/write operations.
Follow-up commits will make similar changes to other file descriptor
operations, and modify the MAC framework to pass both credentials
to MAC policy modules so they can implement either semantic for
revocation.
Obtained from: TrustedBSD Project
Sponsored by: DARPA, NAI Labs
2002-08-15 20:55:08 +00:00
|
|
|
error = VOP_READ(vp, uio, ioflag, fp->f_cred);
|
2000-04-02 00:55:28 +00:00
|
|
|
fp->f_nextoff = uio->uio_offset;
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2012-01-30 19:35:15 +00:00
|
|
|
if (error == 0 && advice == POSIX_FADV_NOREUSE &&
|
2015-09-30 23:06:29 +00:00
|
|
|
orig_offset != uio->uio_offset)
|
2012-06-19 18:42:24 +00:00
|
|
|
/*
|
2015-09-30 23:06:29 +00:00
|
|
|
* Use POSIX_FADV_DONTNEED to flush pages and buffers
|
|
|
|
* for the backing file after a POSIX_FADV_NOREUSE
|
|
|
|
* read(2).
|
2012-06-19 18:42:24 +00:00
|
|
|
*/
|
2015-09-30 23:06:29 +00:00
|
|
|
error = VOP_ADVISE(vp, orig_offset, uio->uio_offset - 1,
|
|
|
|
POSIX_FADV_DONTNEED);
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* File table vnode write routine.
|
|
|
|
*/
|
1995-12-17 21:23:44 +00:00
|
|
|
static int
|
2018-06-01 13:26:45 +00:00
|
|
|
vn_write(struct file *fp, struct uio *uio, struct ucred *active_cred, int flags,
|
|
|
|
struct thread *td)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
1999-07-08 06:06:00 +00:00
|
|
|
struct vnode *vp;
|
2000-07-11 22:07:57 +00:00
|
|
|
struct mount *mp;
|
2015-09-30 23:06:29 +00:00
|
|
|
off_t orig_offset;
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
int error, ioflag, lock_flags;
|
2012-10-22 17:50:54 +00:00
|
|
|
int advice;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2001-09-12 08:38:13 +00:00
|
|
|
KASSERT(uio->uio_td == td, ("uio_td %p is not td %p",
|
|
|
|
uio->uio_td, td));
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
KASSERT(flags & FOF_OFFSET, ("No FOF_OFFSET"));
|
2003-06-22 08:41:43 +00:00
|
|
|
vp = fp->f_vnode;
|
1999-07-08 06:06:00 +00:00
|
|
|
if (vp->v_type == VREG)
|
|
|
|
bwillwrite();
|
1999-04-21 05:56:45 +00:00
|
|
|
ioflag = IO_UNIT;
|
|
|
|
if (vp->v_type == VREG && (fp->f_flag & O_APPEND))
|
1994-05-24 10:09:53 +00:00
|
|
|
ioflag |= IO_APPEND;
|
|
|
|
if (fp->f_flag & FNONBLOCK)
|
|
|
|
ioflag |= IO_NDELAY;
|
2001-05-24 07:22:27 +00:00
|
|
|
if (fp->f_flag & O_DIRECT)
|
|
|
|
ioflag |= IO_DIRECT;
|
1997-02-10 02:22:35 +00:00
|
|
|
if ((fp->f_flag & O_FSYNC) ||
|
|
|
|
(vp->v_mount && (vp->v_mount->mnt_flag & MNT_SYNCHRONOUS)))
|
|
|
|
ioflag |= IO_SYNC;
|
2000-07-11 22:07:57 +00:00
|
|
|
mp = NULL;
|
2000-11-02 21:14:13 +00:00
|
|
|
if (vp->v_type != VCHR &&
|
2005-01-24 10:31:42 +00:00
|
|
|
(error = vn_start_write(vp, &mp, V_WAIT | PCATCH)) != 0)
|
|
|
|
goto unlock;
|
Fix locking for f_offset, vn_read() and vn_write() cases only, for now.
It seems that intended locking protocol for struct file f_offset field
was as follows: f_offset should always be changed under the vnode lock
(except fcntl(2) and lseek(2) did not followed the rules). Since
read(2) uses shared vnode lock, FOFFSET_LOCKED block is additionally
taken to serialize shared vnode lock owners.
This was broken first by enabling shared lock on writes, then by
fadvise changes, which moved f_offset assigned from under vnode lock,
and last by vn_io_fault() doing chunked i/o. More, due to uio_offset
not yet valid in vn_io_fault(), the range lock for reads was taken on
the wrong region.
Change the locking for f_offset to always use FOFFSET_LOCKED block,
which is placed before rangelocks in the lock order.
Extract foffset_lock() and foffset_unlock() functions which implements
FOFFSET_LOCKED lock, and consistently lock f_offset with it in the
vn_io_fault() both for reads and writes, even if MNTK_NO_IOPF flag is
not set for the vnode mount. Indicate that f_offset is already valid
for vn_read() and vn_write() calls from vn_io_fault() with FOF_OFFSET
flag, and assert that all callers of vn_read() and vn_write() follow
this protocol.
Extract get_advice() function to calculate the POSIX_FADV_XXX value
for the i/o region, and use it were appropriate.
Reviewed by: jhb
Tested by: pho
MFC after: 2 weeks
2012-06-21 09:19:41 +00:00
|
|
|
|
|
|
|
advice = get_advice(fp, uio);
|
2012-09-25 21:31:17 +00:00
|
|
|
|
|
|
|
if (MNT_SHARED_WRITES(mp) ||
|
|
|
|
(mp == NULL && MNT_SHARED_WRITES(vp->v_mount))) {
|
2009-06-04 16:18:07 +00:00
|
|
|
lock_flags = LK_SHARED;
|
|
|
|
} else {
|
|
|
|
lock_flags = LK_EXCLUSIVE;
|
|
|
|
}
|
|
|
|
|
|
|
|
vn_lock(vp, lock_flags | LK_RETRY);
|
2011-11-04 04:02:50 +00:00
|
|
|
switch (advice) {
|
|
|
|
case POSIX_FADV_NORMAL:
|
|
|
|
case POSIX_FADV_SEQUENTIAL:
|
2012-06-19 18:42:24 +00:00
|
|
|
case POSIX_FADV_NOREUSE:
|
2011-11-04 04:02:50 +00:00
|
|
|
ioflag |= sequential_heuristic(uio, fp);
|
|
|
|
break;
|
|
|
|
case POSIX_FADV_RANDOM:
|
|
|
|
/* XXX: Is this correct? */
|
|
|
|
break;
|
|
|
|
}
|
2015-09-30 23:06:29 +00:00
|
|
|
orig_offset = uio->uio_offset;
|
2011-11-04 04:02:50 +00:00
|
|
|
|
2002-08-01 17:23:22 +00:00
|
|
|
#ifdef MAC
|
2007-10-24 19:04:04 +00:00
|
|
|
error = mac_vnode_check_write(active_cred, fp->f_cred, vp);
|
2002-08-01 17:23:22 +00:00
|
|
|
if (error == 0)
|
|
|
|
#endif
|
In order to better support flexible and extensible access control,
make a series of modifications to the credential arguments relating
to file read and write operations to cliarfy which credential is
used for what:
- Change fo_read() and fo_write() to accept "active_cred" instead of
"cred", and change the semantics of consumers of fo_read() and
fo_write() to pass the active credential of the thread requesting
an operation rather than the cached file cred. The cached file
cred is still available in fo_read() and fo_write() consumers
via fp->f_cred. These changes largely in sys_generic.c.
For each implementation of fo_read() and fo_write(), update cred
usage to reflect this change and maintain current semantics:
- badfo_readwrite() unchanged
- kqueue_read/write() unchanged
pipe_read/write() now authorize MAC using active_cred rather
than td->td_ucred
- soo_read/write() unchanged
- vn_read/write() now authorize MAC using active_cred but
VOP_READ/WRITE() with fp->f_cred
Modify vn_rdwr() to accept two credential arguments instead of a
single credential: active_cred and file_cred. Use active_cred
for MAC authorization, and select a credential for use in
VOP_READ/WRITE() based on whether file_cred is NULL or not. If
file_cred is provided, authorize the VOP using that cred,
otherwise the active credential, matching current semantics.
Modify current vn_rdwr() consumers to pass a file_cred if used
in the context of a struct file, and to always pass active_cred.
When vn_rdwr() is used without a file_cred, pass NOCRED.
These changes should maintain current semantics for read/write,
but avoid a redundant passing of fp->f_cred, as well as making
it more clear what the origin of each credential is in file
descriptor read/write operations.
Follow-up commits will make similar changes to other file descriptor
operations, and modify the MAC framework to pass both credentials
to MAC policy modules so they can implement either semantic for
revocation.
Obtained from: TrustedBSD Project
Sponsored by: DARPA, NAI Labs
2002-08-15 20:55:08 +00:00
|
|
|
error = VOP_WRITE(vp, uio, ioflag, fp->f_cred);
|
2000-04-02 00:55:28 +00:00
|
|
|
fp->f_nextoff = uio->uio_offset;
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2006-04-28 21:54:05 +00:00
|
|
|
if (vp->v_type != VCHR)
|
|
|
|
vn_finished_write(mp);
|
2012-06-19 18:42:24 +00:00
|
|
|
if (error == 0 && advice == POSIX_FADV_NOREUSE &&
|
2015-09-30 23:06:29 +00:00
|
|
|
orig_offset != uio->uio_offset)
|
2012-06-19 18:42:24 +00:00
|
|
|
/*
|
2015-09-30 23:06:29 +00:00
|
|
|
* Use POSIX_FADV_DONTNEED to flush pages and buffers
|
|
|
|
* for the backing file after a POSIX_FADV_NOREUSE
|
|
|
|
* write(2).
|
2012-06-19 18:42:24 +00:00
|
|
|
*/
|
2015-09-30 23:06:29 +00:00
|
|
|
error = VOP_ADVISE(vp, orig_offset, uio->uio_offset - 1,
|
|
|
|
POSIX_FADV_DONTNEED);
|
2005-01-24 10:31:42 +00:00
|
|
|
unlock:
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2008-01-07 20:05:19 +00:00
|
|
|
/*
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
* The vn_io_fault() is a wrapper around vn_read() and vn_write() to
|
|
|
|
* prevent the following deadlock:
|
|
|
|
*
|
|
|
|
* Assume that the thread A reads from the vnode vp1 into userspace
|
|
|
|
* buffer buf1 backed by the pages of vnode vp2. If a page in buf1 is
|
|
|
|
* currently not resident, then system ends up with the call chain
|
|
|
|
* vn_read() -> VOP_READ(vp1) -> uiomove() -> [Page Fault] ->
|
|
|
|
* vm_fault(buf1) -> vnode_pager_getpages(vp2) -> VOP_GETPAGES(vp2)
|
|
|
|
* which establishes lock order vp1->vn_lock, then vp2->vn_lock.
|
|
|
|
* If, at the same time, thread B reads from vnode vp2 into buffer buf2
|
|
|
|
* backed by the pages of vnode vp1, and some page in buf2 is not
|
|
|
|
* resident, we get a reversed order vp2->vn_lock, then vp1->vn_lock.
|
|
|
|
*
|
|
|
|
* To prevent the lock order reversal and deadlock, vn_io_fault() does
|
|
|
|
* not allow page faults to happen during VOP_READ() or VOP_WRITE().
|
|
|
|
* Instead, it first tries to do the whole range i/o with pagefaults
|
|
|
|
* disabled. If all pages in the i/o buffer are resident and mapped,
|
|
|
|
* VOP will succeed (ignoring the genuine filesystem errors).
|
|
|
|
* Otherwise, we get back EFAULT, and vn_io_fault() falls back to do
|
|
|
|
* i/o in chunks, with all pages in the chunk prefaulted and held
|
|
|
|
* using vm_fault_quick_hold_pages().
|
|
|
|
*
|
|
|
|
* Filesystems using this deadlock avoidance scheme should use the
|
|
|
|
* array of the held pages from uio, saved in the curthread->td_ma,
|
|
|
|
* instead of doing uiomove(). A helper function
|
|
|
|
* vn_io_fault_uiomove() converts uiomove request into
|
|
|
|
* uiomove_fromphys() over td_ma array.
|
|
|
|
*
|
|
|
|
* Since vnode locks do not cover the whole i/o anymore, rangelocks
|
|
|
|
* make the current i/o request atomic with respect to other i/os and
|
|
|
|
* truncations.
|
2008-01-07 20:05:19 +00:00
|
|
|
*/
|
2014-06-15 04:51:53 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Decode vn_io_fault_args and perform the corresponding i/o.
|
|
|
|
*/
|
2008-01-07 20:05:19 +00:00
|
|
|
static int
|
2014-06-15 04:51:53 +00:00
|
|
|
vn_io_fault_doio(struct vn_io_fault_args *args, struct uio *uio,
|
|
|
|
struct thread *td)
|
|
|
|
{
|
2018-03-24 13:13:52 +00:00
|
|
|
int error, save;
|
2014-06-15 04:51:53 +00:00
|
|
|
|
2018-03-24 13:13:52 +00:00
|
|
|
error = 0;
|
|
|
|
save = vm_fault_disable_pagefaults();
|
2014-06-15 04:51:53 +00:00
|
|
|
switch (args->kind) {
|
|
|
|
case VN_IO_FAULT_FOP:
|
2018-03-24 13:13:52 +00:00
|
|
|
error = (args->args.fop_args.doio)(args->args.fop_args.fp,
|
|
|
|
uio, args->cred, args->flags, td);
|
|
|
|
break;
|
2014-06-15 04:51:53 +00:00
|
|
|
case VN_IO_FAULT_VOP:
|
|
|
|
if (uio->uio_rw == UIO_READ) {
|
2018-03-24 13:13:52 +00:00
|
|
|
error = VOP_READ(args->args.vop_args.vp, uio,
|
|
|
|
args->flags, args->cred);
|
2014-06-15 04:51:53 +00:00
|
|
|
} else if (uio->uio_rw == UIO_WRITE) {
|
2018-03-24 13:13:52 +00:00
|
|
|
error = VOP_WRITE(args->args.vop_args.vp, uio,
|
|
|
|
args->flags, args->cred);
|
2014-06-15 04:51:53 +00:00
|
|
|
}
|
|
|
|
break;
|
2018-03-24 13:13:52 +00:00
|
|
|
default:
|
|
|
|
panic("vn_io_fault_doio: unknown kind of io %d %d",
|
|
|
|
args->kind, uio->uio_rw);
|
2014-06-15 04:51:53 +00:00
|
|
|
}
|
2018-03-24 13:13:52 +00:00
|
|
|
vm_fault_enable_pagefaults(save);
|
|
|
|
return (error);
|
2014-06-15 04:51:53 +00:00
|
|
|
}
|
|
|
|
|
2015-07-31 04:12:51 +00:00
|
|
|
static int
|
|
|
|
vn_io_fault_touch(char *base, const struct uio *uio)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
|
|
|
|
r = fubyte(base);
|
|
|
|
if (r == -1 || (uio->uio_rw == UIO_READ && subyte(base, r) == -1))
|
|
|
|
return (EFAULT);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vn_io_fault_prefault_user(const struct uio *uio)
|
|
|
|
{
|
|
|
|
char *base;
|
|
|
|
const struct iovec *iov;
|
|
|
|
size_t len;
|
|
|
|
ssize_t resid;
|
|
|
|
int error, i;
|
|
|
|
|
|
|
|
KASSERT(uio->uio_segflg == UIO_USERSPACE,
|
|
|
|
("vn_io_fault_prefault userspace"));
|
|
|
|
|
|
|
|
error = i = 0;
|
|
|
|
iov = uio->uio_iov;
|
|
|
|
resid = uio->uio_resid;
|
|
|
|
base = iov->iov_base;
|
|
|
|
len = iov->iov_len;
|
|
|
|
while (resid > 0) {
|
|
|
|
error = vn_io_fault_touch(base, uio);
|
|
|
|
if (error != 0)
|
|
|
|
break;
|
|
|
|
if (len < PAGE_SIZE) {
|
|
|
|
if (len != 0) {
|
|
|
|
error = vn_io_fault_touch(base + len - 1, uio);
|
|
|
|
if (error != 0)
|
|
|
|
break;
|
|
|
|
resid -= len;
|
|
|
|
}
|
|
|
|
if (++i >= uio->uio_iovcnt)
|
|
|
|
break;
|
|
|
|
iov = uio->uio_iov + i;
|
|
|
|
base = iov->iov_base;
|
|
|
|
len = iov->iov_len;
|
|
|
|
} else {
|
|
|
|
len -= PAGE_SIZE;
|
|
|
|
base += PAGE_SIZE;
|
|
|
|
resid -= PAGE_SIZE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2014-06-15 04:51:53 +00:00
|
|
|
/*
|
|
|
|
* Common code for vn_io_fault(), agnostic to the kind of i/o request.
|
|
|
|
* Uses vn_io_fault_doio() to make the call to an actual i/o function.
|
|
|
|
* Used from vn_rdwr() and vn_io_fault(), which encode the i/o request
|
|
|
|
* into args and call vn_io_fault1() to handle faults during the user
|
|
|
|
* mode buffer accesses.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
vn_io_fault1(struct vnode *vp, struct uio *uio, struct vn_io_fault_args *args,
|
|
|
|
struct thread *td)
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
{
|
|
|
|
vm_page_t ma[io_hold_cnt + 2];
|
|
|
|
struct uio *uio_clone, short_uio;
|
|
|
|
struct iovec short_iovec[1];
|
|
|
|
vm_page_t *prev_td_ma;
|
|
|
|
vm_prot_t prot;
|
2014-06-15 04:51:53 +00:00
|
|
|
vm_offset_t addr, end;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
size_t len, resid;
|
|
|
|
ssize_t adv;
|
2018-03-24 13:13:52 +00:00
|
|
|
int error, cnt, saveheld, prev_td_ma_cnt;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
|
2015-07-31 04:12:51 +00:00
|
|
|
if (vn_io_fault_prefault) {
|
|
|
|
error = vn_io_fault_prefault_user(uio);
|
|
|
|
if (error != 0)
|
|
|
|
return (error); /* Or ignore ? */
|
|
|
|
}
|
|
|
|
|
2014-06-15 04:51:53 +00:00
|
|
|
prot = uio->uio_rw == UIO_READ ? VM_PROT_WRITE : VM_PROT_READ;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The UFS follows IO_UNIT directive and replays back both
|
|
|
|
* uio_offset and uio_resid if an error is encountered during the
|
|
|
|
* operation. But, since the iovec may be already advanced,
|
|
|
|
* uio is still in an inconsistent state.
|
|
|
|
*
|
|
|
|
* Cache a copy of the original uio, which is advanced to the redo
|
|
|
|
* point using UIO_NOCOPY below.
|
|
|
|
*/
|
|
|
|
uio_clone = cloneuio(uio);
|
|
|
|
resid = uio->uio_resid;
|
|
|
|
|
|
|
|
short_uio.uio_segflg = UIO_USERSPACE;
|
|
|
|
short_uio.uio_rw = uio->uio_rw;
|
|
|
|
short_uio.uio_td = uio->uio_td;
|
|
|
|
|
2014-06-15 04:51:53 +00:00
|
|
|
error = vn_io_fault_doio(args, uio, td);
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
if (error != EFAULT)
|
|
|
|
goto out;
|
|
|
|
|
2012-06-03 16:06:56 +00:00
|
|
|
atomic_add_long(&vn_io_faults_cnt, 1);
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
uio_clone->uio_segflg = UIO_NOCOPY;
|
|
|
|
uiomove(NULL, resid - uio->uio_resid, uio_clone);
|
|
|
|
uio_clone->uio_segflg = uio->uio_segflg;
|
|
|
|
|
|
|
|
saveheld = curthread_pflags_set(TDP_UIOHELD);
|
|
|
|
prev_td_ma = td->td_ma;
|
|
|
|
prev_td_ma_cnt = td->td_ma_cnt;
|
|
|
|
|
|
|
|
while (uio_clone->uio_resid != 0) {
|
|
|
|
len = uio_clone->uio_iov->iov_len;
|
|
|
|
if (len == 0) {
|
|
|
|
KASSERT(uio_clone->uio_iovcnt >= 1,
|
|
|
|
("iovcnt underflow"));
|
|
|
|
uio_clone->uio_iov++;
|
|
|
|
uio_clone->uio_iovcnt--;
|
|
|
|
continue;
|
|
|
|
}
|
2013-11-20 08:45:26 +00:00
|
|
|
if (len > io_hold_cnt * PAGE_SIZE)
|
|
|
|
len = io_hold_cnt * PAGE_SIZE;
|
|
|
|
addr = (uintptr_t)uio_clone->uio_iov->iov_base;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
end = round_page(addr + len);
|
2013-11-20 08:45:26 +00:00
|
|
|
if (end < addr) {
|
|
|
|
error = EFAULT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cnt = atop(end - trunc_page(addr));
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
/*
|
|
|
|
* A perfectly misaligned address and length could cause
|
|
|
|
* both the start and the end of the chunk to use partial
|
|
|
|
* page. +2 accounts for such a situation.
|
|
|
|
*/
|
|
|
|
cnt = vm_fault_quick_hold_pages(&td->td_proc->p_vmspace->vm_map,
|
|
|
|
addr, len, prot, ma, io_hold_cnt + 2);
|
|
|
|
if (cnt == -1) {
|
|
|
|
error = EFAULT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
short_uio.uio_iov = &short_iovec[0];
|
|
|
|
short_iovec[0].iov_base = (void *)addr;
|
|
|
|
short_uio.uio_iovcnt = 1;
|
|
|
|
short_uio.uio_resid = short_iovec[0].iov_len = len;
|
|
|
|
short_uio.uio_offset = uio_clone->uio_offset;
|
|
|
|
td->td_ma = ma;
|
|
|
|
td->td_ma_cnt = cnt;
|
|
|
|
|
2014-06-15 04:51:53 +00:00
|
|
|
error = vn_io_fault_doio(args, &short_uio, td);
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
vm_page_unhold_pages(ma, cnt);
|
|
|
|
adv = len - short_uio.uio_resid;
|
|
|
|
|
|
|
|
uio_clone->uio_iov->iov_base =
|
|
|
|
(char *)uio_clone->uio_iov->iov_base + adv;
|
|
|
|
uio_clone->uio_iov->iov_len -= adv;
|
|
|
|
uio_clone->uio_resid -= adv;
|
|
|
|
uio_clone->uio_offset += adv;
|
|
|
|
|
|
|
|
uio->uio_resid -= adv;
|
|
|
|
uio->uio_offset += adv;
|
|
|
|
|
|
|
|
if (error != 0 || adv == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
td->td_ma = prev_td_ma;
|
|
|
|
td->td_ma_cnt = prev_td_ma_cnt;
|
|
|
|
curthread_pflags_restore(saveheld);
|
|
|
|
out:
|
|
|
|
free(uio_clone, M_IOV);
|
2014-06-15 04:51:53 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vn_io_fault(struct file *fp, struct uio *uio, struct ucred *active_cred,
|
|
|
|
int flags, struct thread *td)
|
|
|
|
{
|
|
|
|
fo_rdwr_t *doio;
|
|
|
|
struct vnode *vp;
|
|
|
|
void *rl_cookie;
|
|
|
|
struct vn_io_fault_args args;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
doio = uio->uio_rw == UIO_READ ? vn_read : vn_write;
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
foffset_lock_uio(fp, uio, flags);
|
|
|
|
if (do_vn_io_fault(vp, uio)) {
|
|
|
|
args.kind = VN_IO_FAULT_FOP;
|
|
|
|
args.args.fop_args.fp = fp;
|
|
|
|
args.args.fop_args.doio = doio;
|
|
|
|
args.cred = active_cred;
|
|
|
|
args.flags = flags | FOF_OFFSET;
|
|
|
|
if (uio->uio_rw == UIO_READ) {
|
|
|
|
rl_cookie = vn_rangelock_rlock(vp, uio->uio_offset,
|
|
|
|
uio->uio_offset + uio->uio_resid);
|
|
|
|
} else if ((fp->f_flag & O_APPEND) != 0 ||
|
|
|
|
(flags & FOF_OFFSET) == 0) {
|
|
|
|
/* For appenders, punt and lock the whole range. */
|
|
|
|
rl_cookie = vn_rangelock_wlock(vp, 0, OFF_MAX);
|
|
|
|
} else {
|
|
|
|
rl_cookie = vn_rangelock_wlock(vp, uio->uio_offset,
|
|
|
|
uio->uio_offset + uio->uio_resid);
|
|
|
|
}
|
|
|
|
error = vn_io_fault1(vp, uio, &args, td);
|
|
|
|
vn_rangelock_unlock(vp, rl_cookie);
|
|
|
|
} else {
|
|
|
|
error = doio(fp, uio, active_cred, flags | FOF_OFFSET, td);
|
|
|
|
}
|
2012-07-02 21:01:03 +00:00
|
|
|
foffset_unlock_uio(fp, uio, flags);
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Helper function to perform the requested uiomove operation using
|
|
|
|
* the held pages for io->uio_iov[0].iov_base buffer instead of
|
|
|
|
* copyin/copyout. Access to the pages with uiomove_fromphys()
|
|
|
|
* instead of iov_base prevents page faults that could occur due to
|
|
|
|
* pmap_collect() invalidating the mapping created by
|
|
|
|
* vm_fault_quick_hold_pages(), or pageout daemon, page laundry or
|
|
|
|
* object cleanup revoking the write access from page mappings.
|
|
|
|
*
|
|
|
|
* Filesystems specified MNTK_NO_IOPF shall use vn_io_fault_uiomove()
|
|
|
|
* instead of plain uiomove().
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
vn_io_fault_uiomove(char *data, int xfersize, struct uio *uio)
|
|
|
|
{
|
|
|
|
struct uio transp_uio;
|
|
|
|
struct iovec transp_iov[1];
|
2008-01-07 20:05:19 +00:00
|
|
|
struct thread *td;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
size_t adv;
|
|
|
|
int error, pgadv;
|
|
|
|
|
|
|
|
td = curthread;
|
|
|
|
if ((td->td_pflags & TDP_UIOHELD) == 0 ||
|
|
|
|
uio->uio_segflg != UIO_USERSPACE)
|
|
|
|
return (uiomove(data, xfersize, uio));
|
|
|
|
|
|
|
|
KASSERT(uio->uio_iovcnt == 1, ("uio_iovcnt %d", uio->uio_iovcnt));
|
|
|
|
transp_iov[0].iov_base = data;
|
|
|
|
transp_uio.uio_iov = &transp_iov[0];
|
|
|
|
transp_uio.uio_iovcnt = 1;
|
|
|
|
if (xfersize > uio->uio_resid)
|
|
|
|
xfersize = uio->uio_resid;
|
|
|
|
transp_uio.uio_resid = transp_iov[0].iov_len = xfersize;
|
|
|
|
transp_uio.uio_offset = 0;
|
|
|
|
transp_uio.uio_segflg = UIO_SYSSPACE;
|
|
|
|
/*
|
|
|
|
* Since transp_iov points to data, and td_ma page array
|
|
|
|
* corresponds to original uio->uio_iov, we need to invert the
|
|
|
|
* direction of the i/o operation as passed to
|
|
|
|
* uiomove_fromphys().
|
|
|
|
*/
|
|
|
|
switch (uio->uio_rw) {
|
|
|
|
case UIO_WRITE:
|
|
|
|
transp_uio.uio_rw = UIO_READ;
|
|
|
|
break;
|
|
|
|
case UIO_READ:
|
|
|
|
transp_uio.uio_rw = UIO_WRITE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
transp_uio.uio_td = uio->uio_td;
|
|
|
|
error = uiomove_fromphys(td->td_ma,
|
|
|
|
((vm_offset_t)uio->uio_iov->iov_base) & PAGE_MASK,
|
|
|
|
xfersize, &transp_uio);
|
|
|
|
adv = xfersize - transp_uio.uio_resid;
|
|
|
|
pgadv =
|
|
|
|
(((vm_offset_t)uio->uio_iov->iov_base + adv) >> PAGE_SHIFT) -
|
|
|
|
(((vm_offset_t)uio->uio_iov->iov_base) >> PAGE_SHIFT);
|
|
|
|
td->td_ma += pgadv;
|
|
|
|
KASSERT(td->td_ma_cnt >= pgadv, ("consumed pages %d %d", td->td_ma_cnt,
|
|
|
|
pgadv));
|
|
|
|
td->td_ma_cnt -= pgadv;
|
|
|
|
uio->uio_iov->iov_base = (char *)uio->uio_iov->iov_base + adv;
|
|
|
|
uio->uio_iov->iov_len -= adv;
|
|
|
|
uio->uio_resid -= adv;
|
|
|
|
uio->uio_offset += adv;
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2013-03-15 11:16:12 +00:00
|
|
|
int
|
|
|
|
vn_io_fault_pgmove(vm_page_t ma[], vm_offset_t offset, int xfersize,
|
|
|
|
struct uio *uio)
|
|
|
|
{
|
|
|
|
struct thread *td;
|
|
|
|
vm_offset_t iov_base;
|
|
|
|
int cnt, pgadv;
|
|
|
|
|
|
|
|
td = curthread;
|
|
|
|
if ((td->td_pflags & TDP_UIOHELD) == 0 ||
|
|
|
|
uio->uio_segflg != UIO_USERSPACE)
|
|
|
|
return (uiomove_fromphys(ma, offset, xfersize, uio));
|
|
|
|
|
|
|
|
KASSERT(uio->uio_iovcnt == 1, ("uio_iovcnt %d", uio->uio_iovcnt));
|
|
|
|
cnt = xfersize > uio->uio_resid ? uio->uio_resid : xfersize;
|
|
|
|
iov_base = (vm_offset_t)uio->uio_iov->iov_base;
|
|
|
|
switch (uio->uio_rw) {
|
|
|
|
case UIO_WRITE:
|
|
|
|
pmap_copy_pages(td->td_ma, iov_base & PAGE_MASK, ma,
|
|
|
|
offset, cnt);
|
|
|
|
break;
|
|
|
|
case UIO_READ:
|
|
|
|
pmap_copy_pages(ma, offset, td->td_ma, iov_base & PAGE_MASK,
|
|
|
|
cnt);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pgadv = ((iov_base + cnt) >> PAGE_SHIFT) - (iov_base >> PAGE_SHIFT);
|
|
|
|
td->td_ma += pgadv;
|
|
|
|
KASSERT(td->td_ma_cnt >= pgadv, ("consumed pages %d %d", td->td_ma_cnt,
|
|
|
|
pgadv));
|
|
|
|
td->td_ma_cnt -= pgadv;
|
|
|
|
uio->uio_iov->iov_base = (char *)(iov_base + cnt);
|
|
|
|
uio->uio_iov->iov_len -= cnt;
|
|
|
|
uio->uio_resid -= cnt;
|
|
|
|
uio->uio_offset += cnt;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
/*
|
|
|
|
* File table truncate routine.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
vn_truncate(struct file *fp, off_t length, struct ucred *active_cred,
|
|
|
|
struct thread *td)
|
2008-01-07 20:05:19 +00:00
|
|
|
{
|
|
|
|
struct mount *mp;
|
|
|
|
struct vnode *vp;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
void *rl_cookie;
|
2008-01-07 20:05:19 +00:00
|
|
|
int error;
|
|
|
|
|
|
|
|
vp = fp->f_vnode;
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Lock the whole range for truncation. Otherwise split i/o
|
|
|
|
* might happen partly before and partly after the truncation.
|
|
|
|
*/
|
|
|
|
rl_cookie = vn_rangelock_wlock(vp, 0, OFF_MAX);
|
2008-01-07 20:05:19 +00:00
|
|
|
error = vn_start_write(vp, &mp, V_WAIT | PCATCH);
|
vn_io_fault() is a facility to prevent page faults while filesystems
perform copyin/copyout of the file data into the usermode
buffer. Typical filesystem hold vnode lock and some buffer locks over
the VOP_READ() and VOP_WRITE() operations, and since page fault
handler may need to recurse into VFS to get the page content, a
deadlock is possible.
The facility works by disabling page faults handling for the current
thread and attempting to execute i/o while allowing uiomove() to
access the usermode mapping of the i/o buffer. If all buffer pages are
resident, uiomove() is successfull and request is finished. If EFAULT
is returned from uiomove(), the pages backing i/o buffer are faulted
in and held, and the copyin/out is performed using uiomove_fromphys()
over the held pages for the second attempt of VOP call.
Since pages are hold in chunks to prevent large i/o requests from
starving free pages pool, and since vnode lock is only taken for
i/o over the current chunk, the vnode lock no longer protect atomicity
of the whole i/o request. Use newly added rangelocks to provide the
required atomicity of i/o regardind other i/o and truncations.
Filesystems need to explicitely opt-in into the scheme, by setting the
MNTK_NO_IOPF struct mount flag, and optionally by using
vn_io_fault_uiomove(9) helper which takes care of calling uiomove() or
converting uio into request for uiomove_fromphys().
Reviewed by: bf (comments), mdf, pjd (previous version)
Tested by: pho
Tested by: flo, Gustau P?rez <gperez entel upc edu> (previous version)
MFC after: 2 months
2012-05-30 16:42:08 +00:00
|
|
|
if (error)
|
|
|
|
goto out1;
|
2008-01-10 01:10:58 +00:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY);
|
2016-08-20 18:51:48 +00:00
|
|
|
AUDIT_ARG_VNODE1(vp);
|
2008-01-07 20:05:19 +00:00
|
|
|
if (vp->v_type == VDIR) {
|
|
|
|
error = EISDIR;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
#ifdef MAC
|
|
|
|
error = mac_vnode_check_write(active_cred, fp->f_cred, vp);
|
|
|
|
if (error)
|
|
|
|
goto out;
|
|
|
|
#endif
|
2019-07-01 20:41:43 +00:00
|
|
|
error = vn_truncate_locked(vp, length, (fp->f_flag & O_FSYNC) != 0,
|
|
|
|
fp->f_cred);
|
|
|
|
out:
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2019-07-01 20:41:43 +00:00
|
|
|
vn_finished_write(mp);
|
|
|
|
out1:
|
|
|
|
vn_rangelock_unlock(vp, rl_cookie);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Truncate a file that is already locked.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
vn_truncate_locked(struct vnode *vp, off_t length, bool sync,
|
|
|
|
struct ucred *cred)
|
|
|
|
{
|
|
|
|
struct vattr vattr;
|
|
|
|
int error;
|
|
|
|
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
error = VOP_ADD_WRITECOUNT(vp, 1);
|
2008-01-07 20:05:19 +00:00
|
|
|
if (error == 0) {
|
|
|
|
VATTR_NULL(&vattr);
|
|
|
|
vattr.va_size = length;
|
2019-07-01 20:41:43 +00:00
|
|
|
if (sync)
|
2016-05-18 12:03:57 +00:00
|
|
|
vattr.va_vaflags |= VA_SYNC;
|
2019-07-01 20:41:43 +00:00
|
|
|
error = VOP_SETATTR(vp, &vattr, cred);
|
Switch to use shared vnode locks for text files during image activation.
kern_execve() locks text vnode exclusive to be able to set and clear
VV_TEXT flag. VV_TEXT is mutually exclusive with the v_writecount > 0
condition.
The change removes VV_TEXT, replacing it with the condition
v_writecount <= -1, and puts v_writecount under the vnode interlock.
Each text reference decrements v_writecount. To clear the text
reference when the segment is unmapped, it is recorded in the
vm_map_entry backed by the text file as MAP_ENTRY_VN_TEXT flag, and
v_writecount is incremented on the map entry removal
The operations like VOP_ADD_WRITECOUNT() and VOP_SET_TEXT() check that
v_writecount does not contradict the desired change. vn_writecheck()
is now racy and its use was eliminated everywhere except access.
Atomic check for writeability and increment of v_writecount is
performed by the VOP. vn_truncate() now increments v_writecount
around VOP_SETATTR() call, lack of which is arguably a bug on its own.
nullfs bypasses v_writecount to the lower vnode always, so nullfs
vnode has its own v_writecount correct, and lower vnode gets all
references, since object->handle is always lower vnode.
On the text vnode' vm object dealloc, the v_writecount value is reset
to zero, and deadfs vop_unset_text short-circuit the operation.
Reclamation of lowervp always reclaims all nullfs vnodes referencing
lowervp first, so no stray references are left.
Reviewed by: markj, trasz
Tested by: mjg, pho
Sponsored by: The FreeBSD Foundation
MFC after: 1 month
Differential revision: https://reviews.freebsd.org/D19923
2019-05-05 11:20:43 +00:00
|
|
|
VOP_ADD_WRITECOUNT_CHECKED(vp, -1);
|
2008-01-07 20:05:19 +00:00
|
|
|
}
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* File table vnode stat routine.
|
|
|
|
*/
|
1999-11-08 03:32:15 +00:00
|
|
|
static int
|
2018-06-01 13:26:45 +00:00
|
|
|
vn_statfile(struct file *fp, struct stat *sb, struct ucred *active_cred,
|
|
|
|
struct thread *td)
|
1999-11-08 03:32:15 +00:00
|
|
|
{
|
2003-06-22 08:41:43 +00:00
|
|
|
struct vnode *vp = fp->f_vnode;
|
2002-02-10 21:44:30 +00:00
|
|
|
int error;
|
1999-11-08 03:32:15 +00:00
|
|
|
|
Use shared vnode locks instead of exclusive vnode locks for the access(),
chdir(), chroot(), eaccess(), fpathconf(), fstat(), fstatfs(), lseek()
(when figuring out the current size of the file in the SEEK_END case),
pathconf(), readlink(), and statfs() system calls.
Submitted by: ups (mostly)
Tested by: pho
MFC after: 1 month
2008-11-03 20:31:00 +00:00
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
Make similar changes to fo_stat() and fo_poll() as made earlier to
fo_read() and fo_write(): explicitly use the cred argument to fo_poll()
as "active_cred" using the passed file descriptor's f_cred reference
to provide access to the file credential. Add an active_cred
argument to fo_stat() so that implementers have access to the active
credential as well as the file credential. Generally modify callers
of fo_stat() to pass in td->td_ucred rather than fp->f_cred, which
was redundantly provided via the fp argument. This set of modifications
also permits threads to perform these operations on behalf of another
thread without modifying their credential.
Trickle this change down into fo_stat/poll() implementations:
- badfo_poll(), badfo_stat(): modify/add arguments.
- kqueue_poll(), kqueue_stat(): modify arguments.
- pipe_poll(), pipe_stat(): modify/add arguments, pass active_cred to
MAC checks rather than td->td_ucred.
- soo_poll(), soo_stat(): modify/add arguments, pass fp->f_cred rather
than cred to pru_sopoll() to maintain current semantics.
- sopoll(): moidfy arguments.
- vn_poll(), vn_statfile(): modify/add arguments, pass new arguments
to vn_stat(). Pass active_cred to MAC and fp->f_cred to VOP_POLL()
to maintian current semantics.
- vn_close(): rename cred to file_cred to reflect reality while I'm here.
- vn_stat(): Add active_cred and file_cred arguments to vn_stat()
and consumers so that this distinction is maintained at the VFS
as well as 'struct file' layer. Pass active_cred instead of
td->td_ucred to MAC and to VOP_GETATTR() to maintain current semantics.
- fifofs: modify the creation of a "filetemp" so that the file
credential is properly initialized and can be used in the socket
code if desired. Pass ap->a_td->td_ucred as the active
credential to soo_poll(). If we teach the vnop interface about
the distinction between file and active credentials, we would use
the active credential here.
Note that current inconsistent passing of active_cred vs. file_cred to
VOP's is maintained. It's not clear why GETATTR would be authorized
using active_cred while POLL would be authorized using file_cred at
the file system level.
Obtained from: TrustedBSD Project
Sponsored by: DARPA, NAI Labs
2002-08-16 12:52:03 +00:00
|
|
|
error = vn_stat(vp, sb, active_cred, fp->f_cred, td);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2002-02-10 21:44:30 +00:00
|
|
|
|
|
|
|
return (error);
|
1999-11-08 03:32:15 +00:00
|
|
|
}
|
|
|
|
|
2002-03-05 15:38:49 +00:00
|
|
|
/*
|
|
|
|
* Stat a vnode; implementation for the stat syscall
|
|
|
|
*/
|
1994-05-25 09:21:21 +00:00
|
|
|
int
|
2017-05-17 00:34:34 +00:00
|
|
|
vn_stat(struct vnode *vp, struct stat *sb, struct ucred *active_cred,
|
|
|
|
struct ucred *file_cred, struct thread *td)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
|
|
|
struct vattr vattr;
|
2017-05-17 00:34:34 +00:00
|
|
|
struct vattr *vap;
|
1994-05-24 10:09:53 +00:00
|
|
|
int error;
|
|
|
|
u_short mode;
|
|
|
|
|
2016-07-05 16:37:01 +00:00
|
|
|
AUDIT_ARG_VNODE1(vp);
|
2002-08-01 17:23:22 +00:00
|
|
|
#ifdef MAC
|
2007-10-24 19:04:04 +00:00
|
|
|
error = mac_vnode_check_stat(active_cred, file_cred, vp);
|
2002-08-01 17:23:22 +00:00
|
|
|
if (error)
|
|
|
|
return (error);
|
|
|
|
#endif
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
vap = &vattr;
|
2008-09-20 19:43:22 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize defaults for new and unusual fields, so that file
|
|
|
|
* systems which don't support these fields don't need to know
|
|
|
|
* about them.
|
|
|
|
*/
|
|
|
|
vap->va_birthtime.tv_sec = -1;
|
|
|
|
vap->va_birthtime.tv_nsec = 0;
|
2008-09-20 19:48:24 +00:00
|
|
|
vap->va_fsid = VNOVAL;
|
|
|
|
vap->va_rdev = NODEV;
|
2008-09-20 19:43:22 +00:00
|
|
|
|
2008-08-28 15:23:18 +00:00
|
|
|
error = VOP_GETATTR(vp, vap, active_cred);
|
1994-05-24 10:09:53 +00:00
|
|
|
if (error)
|
|
|
|
return (error);
|
1999-11-18 08:14:20 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Zero the spare stat fields
|
|
|
|
*/
|
2002-06-24 07:14:44 +00:00
|
|
|
bzero(sb, sizeof *sb);
|
1999-11-18 08:14:20 +00:00
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Copy from vattr table
|
|
|
|
*/
|
1999-07-02 16:29:47 +00:00
|
|
|
if (vap->va_fsid != VNOVAL)
|
|
|
|
sb->st_dev = vap->va_fsid;
|
|
|
|
else
|
|
|
|
sb->st_dev = vp->v_mount->mnt_stat.f_fsid.val[0];
|
1994-05-24 10:09:53 +00:00
|
|
|
sb->st_ino = vap->va_fileid;
|
|
|
|
mode = vap->va_mode;
|
1998-06-27 06:43:09 +00:00
|
|
|
switch (vap->va_type) {
|
1994-05-24 10:09:53 +00:00
|
|
|
case VREG:
|
|
|
|
mode |= S_IFREG;
|
|
|
|
break;
|
|
|
|
case VDIR:
|
|
|
|
mode |= S_IFDIR;
|
|
|
|
break;
|
|
|
|
case VBLK:
|
|
|
|
mode |= S_IFBLK;
|
|
|
|
break;
|
|
|
|
case VCHR:
|
|
|
|
mode |= S_IFCHR;
|
|
|
|
break;
|
|
|
|
case VLNK:
|
|
|
|
mode |= S_IFLNK;
|
|
|
|
break;
|
|
|
|
case VSOCK:
|
|
|
|
mode |= S_IFSOCK;
|
|
|
|
break;
|
|
|
|
case VFIFO:
|
|
|
|
mode |= S_IFIFO;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return (EBADF);
|
2016-04-10 23:07:00 +00:00
|
|
|
}
|
1994-05-24 10:09:53 +00:00
|
|
|
sb->st_mode = mode;
|
|
|
|
sb->st_nlink = vap->va_nlink;
|
|
|
|
sb->st_uid = vap->va_uid;
|
|
|
|
sb->st_gid = vap->va_gid;
|
|
|
|
sb->st_rdev = vap->va_rdev;
|
2001-08-23 17:56:48 +00:00
|
|
|
if (vap->va_size > OFF_MAX)
|
|
|
|
return (EOVERFLOW);
|
1994-05-24 10:09:53 +00:00
|
|
|
sb->st_size = vap->va_size;
|
2019-10-16 13:21:01 +00:00
|
|
|
sb->st_atim.tv_sec = vap->va_atime.tv_sec;
|
|
|
|
sb->st_atim.tv_nsec = vap->va_atime.tv_nsec;
|
|
|
|
sb->st_mtim.tv_sec = vap->va_mtime.tv_sec;
|
|
|
|
sb->st_mtim.tv_nsec = vap->va_mtime.tv_nsec;
|
|
|
|
sb->st_ctim.tv_sec = vap->va_ctime.tv_sec;
|
|
|
|
sb->st_ctim.tv_nsec = vap->va_ctime.tv_nsec;
|
|
|
|
sb->st_birthtim.tv_sec = vap->va_birthtime.tv_sec;
|
|
|
|
sb->st_birthtim.tv_nsec = vap->va_birthtime.tv_nsec;
|
1999-09-09 19:08:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* According to www.opengroup.org, the meaning of st_blksize is
|
|
|
|
* "a filesystem-specific preferred I/O block size for this
|
|
|
|
* object. In some filesystem types, this may vary from file
|
|
|
|
* to file"
|
2010-04-03 08:39:00 +00:00
|
|
|
* Use miminum/default of PAGE_SIZE (e.g. for VCHR).
|
1999-08-13 10:56:07 +00:00
|
|
|
*/
|
1999-09-09 19:08:44 +00:00
|
|
|
|
2010-04-03 08:39:00 +00:00
|
|
|
sb->st_blksize = max(PAGE_SIZE, vap->va_blocksize);
|
1999-09-09 19:08:44 +00:00
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
sb->st_flags = vap->va_flags;
|
2006-11-06 13:42:10 +00:00
|
|
|
if (priv_check(td, PRIV_VFS_GENERATION))
|
1997-03-08 15:14:30 +00:00
|
|
|
sb->st_gen = 0;
|
|
|
|
else
|
|
|
|
sb->st_gen = vap->va_gen;
|
1997-03-07 07:42:41 +00:00
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
sb->st_blocks = vap->va_bytes / S_BLKSIZE;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* File table vnode ioctl routine.
|
|
|
|
*/
|
1995-12-17 21:23:44 +00:00
|
|
|
static int
|
2017-05-17 00:34:34 +00:00
|
|
|
vn_ioctl(struct file *fp, u_long com, void *data, struct ucred *active_cred,
|
|
|
|
struct thread *td)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
|
|
|
struct vattr vattr;
|
2013-05-31 16:15:22 +00:00
|
|
|
struct vnode *vp;
|
2019-06-27 23:39:06 +00:00
|
|
|
struct fiobmap2_arg *bmarg;
|
1994-05-24 10:09:53 +00:00
|
|
|
int error;
|
|
|
|
|
2013-05-31 16:15:22 +00:00
|
|
|
vp = fp->f_vnode;
|
1994-05-24 10:09:53 +00:00
|
|
|
switch (vp->v_type) {
|
|
|
|
case VDIR:
|
2013-05-31 16:15:22 +00:00
|
|
|
case VREG:
|
|
|
|
switch (com) {
|
|
|
|
case FIONREAD:
|
2013-05-03 19:08:58 +00:00
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
2008-08-28 15:23:18 +00:00
|
|
|
error = VOP_GETATTR(vp, &vattr, active_cred);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2013-05-31 16:15:22 +00:00
|
|
|
if (error == 0)
|
2004-11-17 09:09:55 +00:00
|
|
|
*(int *)data = vattr.va_size - fp->f_offset;
|
2013-05-31 16:15:22 +00:00
|
|
|
return (error);
|
2019-06-20 14:13:10 +00:00
|
|
|
case FIOBMAP2:
|
2019-06-27 23:39:06 +00:00
|
|
|
bmarg = (struct fiobmap2_arg *)data;
|
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
|
|
|
#ifdef MAC
|
|
|
|
error = mac_vnode_check_read(active_cred, fp->f_cred,
|
|
|
|
vp);
|
|
|
|
if (error == 0)
|
|
|
|
#endif
|
|
|
|
error = VOP_BMAP(vp, bmarg->bn, NULL,
|
|
|
|
&bmarg->bn, &bmarg->runp, &bmarg->runb);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2019-06-27 23:39:06 +00:00
|
|
|
return (error);
|
2013-05-31 16:15:22 +00:00
|
|
|
case FIONBIO:
|
|
|
|
case FIOASYNC:
|
|
|
|
return (0);
|
|
|
|
default:
|
|
|
|
return (VOP_IOCTL(vp, com, data, fp->f_flag,
|
|
|
|
active_cred, td));
|
|
|
|
}
|
2016-07-25 16:28:02 +00:00
|
|
|
break;
|
|
|
|
case VCHR:
|
|
|
|
return (VOP_IOCTL(vp, com, data, fp->f_flag,
|
|
|
|
active_cred, td));
|
1994-05-24 10:09:53 +00:00
|
|
|
default:
|
2013-05-31 16:15:22 +00:00
|
|
|
return (ENOTTY);
|
1994-05-24 10:09:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
1997-09-14 02:51:16 +00:00
|
|
|
* File table vnode poll routine.
|
1994-05-24 10:09:53 +00:00
|
|
|
*/
|
1995-12-17 21:23:44 +00:00
|
|
|
static int
|
2017-05-17 00:34:34 +00:00
|
|
|
vn_poll(struct file *fp, int events, struct ucred *active_cred,
|
|
|
|
struct thread *td)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
2002-08-01 17:23:22 +00:00
|
|
|
struct vnode *vp;
|
|
|
|
int error;
|
|
|
|
|
2003-06-22 08:41:43 +00:00
|
|
|
vp = fp->f_vnode;
|
2002-08-01 17:23:22 +00:00
|
|
|
#ifdef MAC
|
2008-01-10 01:10:58 +00:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY);
|
2016-07-05 16:37:01 +00:00
|
|
|
AUDIT_ARG_VNODE1(vp);
|
2007-10-24 19:04:04 +00:00
|
|
|
error = mac_vnode_check_poll(active_cred, fp->f_cred, vp);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2004-11-15 21:56:42 +00:00
|
|
|
if (!error)
|
2002-08-01 17:23:22 +00:00
|
|
|
#endif
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2004-11-15 21:56:42 +00:00
|
|
|
error = VOP_POLL(vp, events, fp->f_cred, td);
|
|
|
|
return (error);
|
1994-05-24 10:09:53 +00:00
|
|
|
}
|
|
|
|
|
1997-02-10 02:22:35 +00:00
|
|
|
/*
|
2008-03-29 23:36:26 +00:00
|
|
|
* Acquire the requested lock and then check for validity. LK_RETRY
|
|
|
|
* permits vn_lock to return doomed vnodes.
|
1997-02-10 02:22:35 +00:00
|
|
|
*/
|
2020-01-05 01:00:11 +00:00
|
|
|
static int __noinline
|
|
|
|
_vn_lock_fallback(struct vnode *vp, int flags, char *file, int line, int error)
|
1997-02-10 02:22:35 +00:00
|
|
|
{
|
2001-03-26 12:45:35 +00:00
|
|
|
|
2017-01-21 18:38:16 +00:00
|
|
|
KASSERT((flags & LK_RETRY) == 0 || error == 0,
|
2017-01-22 19:38:45 +00:00
|
|
|
("vn_lock: error %d incompatible with flags %#x", error, flags));
|
2017-01-21 20:34:20 +00:00
|
|
|
|
2020-01-05 01:00:11 +00:00
|
|
|
if (error == 0)
|
|
|
|
VNASSERT(VN_IS_DOOMED(vp), vp, ("vnode not doomed"));
|
|
|
|
|
2017-01-21 20:34:20 +00:00
|
|
|
if ((flags & LK_RETRY) == 0) {
|
2020-01-05 01:00:11 +00:00
|
|
|
if (error == 0) {
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2005-03-13 11:56:28 +00:00
|
|
|
error = ENOENT;
|
|
|
|
}
|
2020-01-05 01:00:11 +00:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* LK_RETRY case.
|
|
|
|
*
|
|
|
|
* Nothing to do if we got the lock.
|
|
|
|
*/
|
|
|
|
if (error == 0)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Interlock was dropped by the call in _vn_lock.
|
|
|
|
*/
|
|
|
|
flags &= ~LK_INTERLOCK;
|
|
|
|
do {
|
|
|
|
error = VOP_LOCK1(vp, flags, file, line);
|
|
|
|
} while (error != 0);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
_vn_lock(struct vnode *vp, int flags, char *file, int line)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
|
|
|
|
VNASSERT((flags & LK_TYPE_MASK) != 0, vp, ("vn_lock: no locktype"));
|
|
|
|
VNASSERT(vp->v_holdcnt != 0, vp, ("vn_lock: zero hold count"));
|
|
|
|
error = VOP_LOCK1(vp, flags, file, line);
|
|
|
|
if (__predict_false(error != 0 || VN_IS_DOOMED(vp)))
|
|
|
|
return (_vn_lock_fallback(vp, flags, file, line, error));
|
|
|
|
return (0);
|
1997-02-10 02:22:35 +00:00
|
|
|
}
|
1997-10-27 15:26:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* File table vnode close routine.
|
|
|
|
*/
|
|
|
|
static int
|
2017-01-22 19:38:45 +00:00
|
|
|
vn_closefile(struct file *fp, struct thread *td)
|
1997-10-27 15:26:23 +00:00
|
|
|
{
|
2004-06-01 18:03:20 +00:00
|
|
|
struct vnode *vp;
|
|
|
|
struct flock lf;
|
2004-07-22 18:35:43 +00:00
|
|
|
int error;
|
2017-02-09 23:36:50 +00:00
|
|
|
bool ref;
|
2004-06-01 18:03:20 +00:00
|
|
|
|
|
|
|
vp = fp->f_vnode;
|
2012-07-31 18:25:00 +00:00
|
|
|
fp->f_ops = &badfileops;
|
2017-02-09 23:36:50 +00:00
|
|
|
ref= (fp->f_flag & FHASLOCK) != 0 && fp->f_type == DTYPE_VNODE;
|
2004-06-01 18:03:20 +00:00
|
|
|
|
2017-02-09 23:36:50 +00:00
|
|
|
error = vn_close1(vp, fp->f_flag, fp->f_cred, td, ref);
|
2012-07-31 18:25:00 +00:00
|
|
|
|
2017-02-09 23:36:50 +00:00
|
|
|
if (__predict_false(ref)) {
|
2004-06-01 18:03:20 +00:00
|
|
|
lf.l_whence = SEEK_SET;
|
|
|
|
lf.l_start = 0;
|
|
|
|
lf.l_len = 0;
|
|
|
|
lf.l_type = F_UNLCK;
|
2005-12-14 00:49:52 +00:00
|
|
|
(void) VOP_ADVLOCK(vp, fp, F_UNLCK, &lf, F_FLOCK);
|
2012-07-31 18:25:00 +00:00
|
|
|
vrele(vp);
|
2004-06-01 18:03:20 +00:00
|
|
|
}
|
2004-07-22 18:35:43 +00:00
|
|
|
return (error);
|
1997-10-27 15:26:23 +00:00
|
|
|
}
|
2000-04-16 18:53:38 +00:00
|
|
|
|
2014-10-19 06:59:33 +00:00
|
|
|
static bool
|
2015-07-05 22:37:33 +00:00
|
|
|
vn_suspendable(struct mount *mp)
|
2014-10-19 06:59:33 +00:00
|
|
|
{
|
|
|
|
|
2015-07-05 22:37:33 +00:00
|
|
|
return (mp->mnt_op->vfs_susp_clean != NULL);
|
2014-10-19 06:59:33 +00:00
|
|
|
}
|
|
|
|
|
2000-07-11 22:07:57 +00:00
|
|
|
/*
|
|
|
|
* Preparing to start a filesystem write operation. If the operation is
|
|
|
|
* permitted, then we bump the count of operations in progress and
|
|
|
|
* proceed. If a suspend request is in progress, we wait until the
|
|
|
|
* suspension is over, and then proceed.
|
|
|
|
*/
|
2012-12-28 23:08:30 +00:00
|
|
|
static int
|
2019-09-16 21:33:16 +00:00
|
|
|
vn_start_write_refed(struct mount *mp, int flags, bool mplocked)
|
2012-12-28 23:08:30 +00:00
|
|
|
{
|
2014-12-13 16:07:01 +00:00
|
|
|
int error, mflags;
|
2012-12-28 23:08:30 +00:00
|
|
|
|
2019-09-16 21:33:16 +00:00
|
|
|
if (__predict_true(!mplocked) && (flags & V_XSLEEP) == 0 &&
|
|
|
|
vfs_op_thread_enter(mp)) {
|
|
|
|
MPASS((mp->mnt_kern_flag & MNTK_SUSPEND) == 0);
|
2019-09-16 21:37:47 +00:00
|
|
|
vfs_mp_count_add_pcpu(mp, writeopcount, 1);
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_op_thread_exit(mp);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mplocked)
|
|
|
|
mtx_assert(MNT_MTX(mp), MA_OWNED);
|
|
|
|
else
|
|
|
|
MNT_ILOCK(mp);
|
|
|
|
|
2012-12-28 23:08:30 +00:00
|
|
|
error = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check on status of suspension.
|
|
|
|
*/
|
|
|
|
if ((curthread->td_pflags & TDP_IGNSUSP) == 0 ||
|
|
|
|
mp->mnt_susp_owner != curthread) {
|
2014-12-13 16:07:01 +00:00
|
|
|
mflags = ((mp->mnt_vfc->vfc_flags & VFCF_SBDRY) != 0 ?
|
|
|
|
(flags & PCATCH) : 0) | (PUSER - 1);
|
2012-12-28 23:08:30 +00:00
|
|
|
while ((mp->mnt_kern_flag & MNTK_SUSPEND) != 0) {
|
|
|
|
if (flags & V_NOWAIT) {
|
|
|
|
error = EWOULDBLOCK;
|
|
|
|
goto unlock;
|
|
|
|
}
|
2014-12-13 16:07:01 +00:00
|
|
|
error = msleep(&mp->mnt_flag, MNT_MTX(mp), mflags,
|
|
|
|
"suspfs", 0);
|
2012-12-28 23:08:30 +00:00
|
|
|
if (error)
|
|
|
|
goto unlock;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (flags & V_XSLEEP)
|
|
|
|
goto unlock;
|
2019-09-16 21:37:47 +00:00
|
|
|
mp->mnt_writeopcount++;
|
2012-12-28 23:08:30 +00:00
|
|
|
unlock:
|
|
|
|
if (error != 0 || (flags & V_XSLEEP) != 0)
|
|
|
|
MNT_REL(mp);
|
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2000-07-11 22:07:57 +00:00
|
|
|
int
|
2015-05-27 09:21:47 +00:00
|
|
|
vn_start_write(struct vnode *vp, struct mount **mpp, int flags)
|
2000-07-11 22:07:57 +00:00
|
|
|
{
|
|
|
|
struct mount *mp;
|
|
|
|
int error;
|
|
|
|
|
2015-05-27 09:21:47 +00:00
|
|
|
KASSERT((flags & V_MNTREF) == 0 || (*mpp != NULL && vp == NULL),
|
|
|
|
("V_MNTREF requires mp"));
|
2014-10-19 06:59:33 +00:00
|
|
|
|
2005-01-24 10:31:42 +00:00
|
|
|
error = 0;
|
2000-07-11 22:07:57 +00:00
|
|
|
/*
|
|
|
|
* If a vnode is provided, get and return the mount point that
|
|
|
|
* to which it will write.
|
|
|
|
*/
|
|
|
|
if (vp != NULL) {
|
|
|
|
if ((error = VOP_GETWRITEMOUNT(vp, mpp)) != 0) {
|
|
|
|
*mpp = NULL;
|
|
|
|
if (error != EOPNOTSUPP)
|
|
|
|
return (error);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((mp = *mpp) == NULL)
|
|
|
|
return (0);
|
2008-11-02 10:15:42 +00:00
|
|
|
|
2015-07-05 22:37:33 +00:00
|
|
|
if (!vn_suspendable(mp)) {
|
|
|
|
if (vp != NULL || (flags & V_MNTREF) != 0)
|
|
|
|
vfs_rel(mp);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2008-11-02 10:15:42 +00:00
|
|
|
/*
|
|
|
|
* VOP_GETWRITEMOUNT() returns with the mp refcount held through
|
|
|
|
* a vfs_ref().
|
|
|
|
* As long as a vnode is not provided we need to acquire a
|
|
|
|
* refcount for the provided mountpoint too, in order to
|
|
|
|
* emulate a vfs_ref().
|
|
|
|
*/
|
2015-05-27 09:21:47 +00:00
|
|
|
if (vp == NULL && (flags & V_MNTREF) == 0)
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_ref(mp);
|
2008-11-02 10:15:42 +00:00
|
|
|
|
2019-09-16 21:33:16 +00:00
|
|
|
return (vn_start_write_refed(mp, flags, false));
|
2000-07-11 22:07:57 +00:00
|
|
|
}
|
|
|
|
|
2006-03-08 23:43:39 +00:00
|
|
|
/*
|
|
|
|
* Secondary suspension. Used by operations such as vop_inactive
|
|
|
|
* routines that are needed by the higher level functions. These
|
|
|
|
* are allowed to proceed until all the higher level functions have
|
|
|
|
* completed (indicated by mnt_writeopcount dropping to zero). At that
|
|
|
|
* time, these operations are halted until the suspension is over.
|
|
|
|
*/
|
|
|
|
int
|
2015-05-27 09:21:47 +00:00
|
|
|
vn_start_secondary_write(struct vnode *vp, struct mount **mpp, int flags)
|
2006-03-08 23:43:39 +00:00
|
|
|
{
|
|
|
|
struct mount *mp;
|
|
|
|
int error;
|
|
|
|
|
2015-05-27 09:21:47 +00:00
|
|
|
KASSERT((flags & V_MNTREF) == 0 || (*mpp != NULL && vp == NULL),
|
|
|
|
("V_MNTREF requires mp"));
|
2014-10-19 06:59:33 +00:00
|
|
|
|
2006-03-08 23:43:39 +00:00
|
|
|
retry:
|
|
|
|
if (vp != NULL) {
|
|
|
|
if ((error = VOP_GETWRITEMOUNT(vp, mpp)) != 0) {
|
|
|
|
*mpp = NULL;
|
|
|
|
if (error != EOPNOTSUPP)
|
|
|
|
return (error);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* If we are not suspended or have not yet reached suspended
|
|
|
|
* mode, then let the operation proceed.
|
|
|
|
*/
|
|
|
|
if ((mp = *mpp) == NULL)
|
|
|
|
return (0);
|
2008-11-02 10:15:42 +00:00
|
|
|
|
2015-07-05 22:37:33 +00:00
|
|
|
if (!vn_suspendable(mp)) {
|
|
|
|
if (vp != NULL || (flags & V_MNTREF) != 0)
|
|
|
|
vfs_rel(mp);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2008-11-02 10:15:42 +00:00
|
|
|
/*
|
|
|
|
* VOP_GETWRITEMOUNT() returns with the mp refcount held through
|
|
|
|
* a vfs_ref().
|
|
|
|
* As long as a vnode is not provided we need to acquire a
|
|
|
|
* refcount for the provided mountpoint too, in order to
|
|
|
|
* emulate a vfs_ref().
|
|
|
|
*/
|
2006-03-08 23:43:39 +00:00
|
|
|
MNT_ILOCK(mp);
|
2015-05-27 09:21:47 +00:00
|
|
|
if (vp == NULL && (flags & V_MNTREF) == 0)
|
2006-03-31 03:54:20 +00:00
|
|
|
MNT_REF(mp);
|
2006-03-11 01:08:37 +00:00
|
|
|
if ((mp->mnt_kern_flag & (MNTK_SUSPENDED | MNTK_SUSPEND2)) == 0) {
|
2006-03-08 23:43:39 +00:00
|
|
|
mp->mnt_secondary_writes++;
|
|
|
|
mp->mnt_secondary_accwrites++;
|
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
if (flags & V_NOWAIT) {
|
2006-03-31 03:54:20 +00:00
|
|
|
MNT_REL(mp);
|
2006-03-08 23:43:39 +00:00
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
return (EWOULDBLOCK);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Wait for the suspension to finish.
|
|
|
|
*/
|
2014-12-13 16:07:01 +00:00
|
|
|
error = msleep(&mp->mnt_flag, MNT_MTX(mp), (PUSER - 1) | PDROP |
|
|
|
|
((mp->mnt_vfc->vfc_flags & VFCF_SBDRY) != 0 ? (flags & PCATCH) : 0),
|
|
|
|
"suspfs", 0);
|
2006-03-31 03:54:20 +00:00
|
|
|
vfs_rel(mp);
|
2006-03-08 23:43:39 +00:00
|
|
|
if (error == 0)
|
|
|
|
goto retry;
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
2000-07-11 22:07:57 +00:00
|
|
|
/*
|
|
|
|
* Filesystem write operation has completed. If we are suspending and this
|
|
|
|
* operation is the last one, notify the suspender that the suspension is
|
|
|
|
* now in effect.
|
|
|
|
*/
|
|
|
|
void
|
2017-05-17 00:34:34 +00:00
|
|
|
vn_finished_write(struct mount *mp)
|
2000-07-11 22:07:57 +00:00
|
|
|
{
|
2019-09-16 21:33:16 +00:00
|
|
|
int c;
|
|
|
|
|
2015-07-05 22:37:33 +00:00
|
|
|
if (mp == NULL || !vn_suspendable(mp))
|
2000-07-11 22:07:57 +00:00
|
|
|
return;
|
2019-09-16 21:33:16 +00:00
|
|
|
|
|
|
|
if (vfs_op_thread_enter(mp)) {
|
2019-09-16 21:37:47 +00:00
|
|
|
vfs_mp_count_sub_pcpu(mp, writeopcount, 1);
|
|
|
|
vfs_mp_count_sub_pcpu(mp, ref, 1);
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_op_thread_exit(mp);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2005-01-24 10:31:42 +00:00
|
|
|
MNT_ILOCK(mp);
|
2019-09-16 21:37:47 +00:00
|
|
|
vfs_assert_mount_counters(mp);
|
2009-08-31 10:20:52 +00:00
|
|
|
MNT_REL(mp);
|
2019-09-16 21:37:47 +00:00
|
|
|
c = --mp->mnt_writeopcount;
|
|
|
|
if (mp->mnt_vfs_ops == 0) {
|
|
|
|
MPASS((mp->mnt_kern_flag & MNTK_SUSPEND) == 0);
|
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
return;
|
|
|
|
}
|
2019-09-16 21:33:16 +00:00
|
|
|
if (c < 0)
|
2019-09-16 21:37:47 +00:00
|
|
|
vfs_dump_mount_counters(mp);
|
2019-09-16 21:33:16 +00:00
|
|
|
if ((mp->mnt_kern_flag & MNTK_SUSPEND) != 0 && c == 0)
|
2000-07-11 22:07:57 +00:00
|
|
|
wakeup(&mp->mnt_writeopcount);
|
2005-01-24 10:31:42 +00:00
|
|
|
MNT_IUNLOCK(mp);
|
2000-07-11 22:07:57 +00:00
|
|
|
}
|
|
|
|
|
2006-03-08 23:43:39 +00:00
|
|
|
/*
|
|
|
|
* Filesystem secondary write operation has completed. If we are
|
|
|
|
* suspending and this operation is the last one, notify the suspender
|
|
|
|
* that the suspension is now in effect.
|
|
|
|
*/
|
|
|
|
void
|
2017-05-17 00:34:34 +00:00
|
|
|
vn_finished_secondary_write(struct mount *mp)
|
2006-03-08 23:43:39 +00:00
|
|
|
{
|
2015-07-05 22:37:33 +00:00
|
|
|
if (mp == NULL || !vn_suspendable(mp))
|
2006-03-08 23:43:39 +00:00
|
|
|
return;
|
|
|
|
MNT_ILOCK(mp);
|
2009-08-31 10:20:52 +00:00
|
|
|
MNT_REL(mp);
|
2006-03-08 23:43:39 +00:00
|
|
|
mp->mnt_secondary_writes--;
|
|
|
|
if (mp->mnt_secondary_writes < 0)
|
|
|
|
panic("vn_finished_secondary_write: neg cnt");
|
|
|
|
if ((mp->mnt_kern_flag & MNTK_SUSPEND) != 0 &&
|
|
|
|
mp->mnt_secondary_writes <= 0)
|
|
|
|
wakeup(&mp->mnt_secondary_writes);
|
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
}
|
|
|
|
|
2000-07-11 22:07:57 +00:00
|
|
|
/*
|
|
|
|
* Request a filesystem to suspend write operations.
|
|
|
|
*/
|
2002-10-25 00:20:37 +00:00
|
|
|
int
|
2013-07-09 20:49:32 +00:00
|
|
|
vfs_write_suspend(struct mount *mp, int flags)
|
2000-07-11 22:07:57 +00:00
|
|
|
{
|
2002-10-25 00:20:37 +00:00
|
|
|
int error;
|
2000-07-11 22:07:57 +00:00
|
|
|
|
2015-07-05 22:37:33 +00:00
|
|
|
MPASS(vn_suspendable(mp));
|
2014-10-19 06:59:33 +00:00
|
|
|
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_op_enter(mp);
|
|
|
|
|
2005-01-24 10:31:42 +00:00
|
|
|
MNT_ILOCK(mp);
|
2019-09-16 21:37:47 +00:00
|
|
|
vfs_assert_mount_counters(mp);
|
2008-09-16 11:51:06 +00:00
|
|
|
if (mp->mnt_susp_owner == curthread) {
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_op_exit_locked(mp);
|
2006-06-24 22:55:43 +00:00
|
|
|
MNT_IUNLOCK(mp);
|
2008-09-16 11:51:06 +00:00
|
|
|
return (EALREADY);
|
2006-06-24 22:55:43 +00:00
|
|
|
}
|
2008-09-16 11:51:06 +00:00
|
|
|
while (mp->mnt_kern_flag & MNTK_SUSPEND)
|
|
|
|
msleep(&mp->mnt_flag, MNT_MTX(mp), PUSER - 1, "wsuspfs", 0);
|
2013-07-09 20:49:32 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Unmount holds a write reference on the mount point. If we
|
|
|
|
* own busy reference and drain for writers, we deadlock with
|
|
|
|
* the reference draining in the unmount path. Callers of
|
|
|
|
* vfs_write_suspend() must specify VS_SKIP_UNMOUNT if
|
|
|
|
* vfs_busy() reference is owned and caller is not in the
|
|
|
|
* unmount context.
|
|
|
|
*/
|
|
|
|
if ((flags & VS_SKIP_UNMOUNT) != 0 &&
|
|
|
|
(mp->mnt_kern_flag & MNTK_UNMOUNT) != 0) {
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_op_exit_locked(mp);
|
2013-07-09 20:49:32 +00:00
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
return (EBUSY);
|
|
|
|
}
|
|
|
|
|
2000-07-11 22:07:57 +00:00
|
|
|
mp->mnt_kern_flag |= MNTK_SUSPEND;
|
2008-09-16 11:51:06 +00:00
|
|
|
mp->mnt_susp_owner = curthread;
|
2000-07-11 22:07:57 +00:00
|
|
|
if (mp->mnt_writeopcount > 0)
|
2005-01-24 10:31:42 +00:00
|
|
|
(void) msleep(&mp->mnt_writeopcount,
|
|
|
|
MNT_MTX(mp), (PUSER - 1)|PDROP, "suspwt", 0);
|
|
|
|
else
|
|
|
|
MNT_IUNLOCK(mp);
|
2019-09-16 21:33:16 +00:00
|
|
|
if ((error = VFS_SYNC(mp, MNT_SUSPEND)) != 0) {
|
2013-01-11 06:08:32 +00:00
|
|
|
vfs_write_resume(mp, 0);
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_op_exit(mp);
|
|
|
|
}
|
2005-01-24 10:31:42 +00:00
|
|
|
return (error);
|
2000-07-11 22:07:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Request a filesystem to resume write operations.
|
|
|
|
*/
|
|
|
|
void
|
2013-01-11 06:08:32 +00:00
|
|
|
vfs_write_resume(struct mount *mp, int flags)
|
2000-07-11 22:07:57 +00:00
|
|
|
{
|
|
|
|
|
2015-07-05 22:37:33 +00:00
|
|
|
MPASS(vn_suspendable(mp));
|
2014-10-19 06:59:33 +00:00
|
|
|
|
2005-01-24 10:31:42 +00:00
|
|
|
MNT_ILOCK(mp);
|
|
|
|
if ((mp->mnt_kern_flag & MNTK_SUSPEND) != 0) {
|
2008-09-16 11:51:06 +00:00
|
|
|
KASSERT(mp->mnt_susp_owner == curthread, ("mnt_susp_owner"));
|
2006-03-11 01:08:37 +00:00
|
|
|
mp->mnt_kern_flag &= ~(MNTK_SUSPEND | MNTK_SUSPEND2 |
|
|
|
|
MNTK_SUSPENDED);
|
2008-09-16 11:51:06 +00:00
|
|
|
mp->mnt_susp_owner = NULL;
|
2005-01-24 10:31:42 +00:00
|
|
|
wakeup(&mp->mnt_writeopcount);
|
|
|
|
wakeup(&mp->mnt_flag);
|
2008-09-16 11:51:06 +00:00
|
|
|
curthread->td_pflags &= ~TDP_IGNSUSP;
|
2012-12-28 23:08:30 +00:00
|
|
|
if ((flags & VR_START_WRITE) != 0) {
|
|
|
|
MNT_REF(mp);
|
2019-09-16 21:37:47 +00:00
|
|
|
mp->mnt_writeopcount++;
|
2012-12-28 23:08:30 +00:00
|
|
|
}
|
2008-09-16 11:51:06 +00:00
|
|
|
MNT_IUNLOCK(mp);
|
2013-01-01 16:14:48 +00:00
|
|
|
if ((flags & VR_NO_SUSPCLR) == 0)
|
|
|
|
VFS_SUSP_CLEAN(mp);
|
2019-09-16 21:33:16 +00:00
|
|
|
vfs_op_exit(mp);
|
2012-12-28 23:08:30 +00:00
|
|
|
} else if ((flags & VR_START_WRITE) != 0) {
|
|
|
|
MNT_REF(mp);
|
2019-09-16 21:33:16 +00:00
|
|
|
vn_start_write_refed(mp, 0, true);
|
2012-12-28 23:08:30 +00:00
|
|
|
} else {
|
2008-09-16 11:51:06 +00:00
|
|
|
MNT_IUNLOCK(mp);
|
2012-12-28 23:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-14 09:10:00 +00:00
|
|
|
/*
|
|
|
|
* Helper loop around vfs_write_suspend() for filesystem unmount VFS
|
|
|
|
* methods.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
vfs_write_suspend_umnt(struct mount *mp)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
|
2015-07-05 22:37:33 +00:00
|
|
|
MPASS(vn_suspendable(mp));
|
2014-07-14 09:10:00 +00:00
|
|
|
KASSERT((curthread->td_pflags & TDP_IGNSUSP) == 0,
|
|
|
|
("vfs_write_suspend_umnt: recursed"));
|
|
|
|
|
|
|
|
/* dounmount() already called vn_start_write(). */
|
|
|
|
for (;;) {
|
|
|
|
vn_finished_write(mp);
|
|
|
|
error = vfs_write_suspend(mp, 0);
|
2014-11-14 11:31:10 +00:00
|
|
|
if (error != 0) {
|
|
|
|
vn_start_write(NULL, &mp, V_WAIT);
|
2014-07-14 09:10:00 +00:00
|
|
|
return (error);
|
2014-11-14 11:31:10 +00:00
|
|
|
}
|
2014-07-14 09:10:00 +00:00
|
|
|
MNT_ILOCK(mp);
|
|
|
|
if ((mp->mnt_kern_flag & MNTK_SUSPENDED) != 0)
|
|
|
|
break;
|
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
vn_start_write(NULL, &mp, V_WAIT);
|
|
|
|
}
|
|
|
|
mp->mnt_kern_flag &= ~(MNTK_SUSPENDED | MNTK_SUSPEND2);
|
|
|
|
wakeup(&mp->mnt_flag);
|
|
|
|
MNT_IUNLOCK(mp);
|
|
|
|
curthread->td_pflags |= TDP_IGNSUSP;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2002-03-05 15:38:49 +00:00
|
|
|
/*
|
|
|
|
* Implement kqueues for files by translating it to vnode operation.
|
|
|
|
*/
|
2000-04-16 18:53:38 +00:00
|
|
|
static int
|
2001-02-15 16:34:11 +00:00
|
|
|
vn_kqfilter(struct file *fp, struct knote *kn)
|
2000-04-16 18:53:38 +00:00
|
|
|
{
|
|
|
|
|
2013-01-31 22:12:48 +00:00
|
|
|
return (VOP_KQFILTER(fp->f_vnode, kn));
|
2000-04-16 18:53:38 +00:00
|
|
|
}
|
2000-08-08 17:15:32 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Simplified in-kernel wrapper calls for extended attribute access.
|
|
|
|
* Both calls pass in a NULL credential, authorizing as "kernel" access.
|
|
|
|
* Set IO_NODELOCKED in ioflg if the vnode is already locked.
|
|
|
|
*/
|
|
|
|
int
|
2001-03-19 05:44:15 +00:00
|
|
|
vn_extattr_get(struct vnode *vp, int ioflg, int attrnamespace,
|
2001-09-12 08:38:13 +00:00
|
|
|
const char *attrname, int *buflen, char *buf, struct thread *td)
|
2000-08-08 17:15:32 +00:00
|
|
|
{
|
|
|
|
struct uio auio;
|
|
|
|
struct iovec iov;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
iov.iov_len = *buflen;
|
|
|
|
iov.iov_base = buf;
|
|
|
|
|
|
|
|
auio.uio_iov = &iov;
|
|
|
|
auio.uio_iovcnt = 1;
|
|
|
|
auio.uio_rw = UIO_READ;
|
|
|
|
auio.uio_segflg = UIO_SYSSPACE;
|
2001-09-12 08:38:13 +00:00
|
|
|
auio.uio_td = td;
|
2000-08-08 17:15:32 +00:00
|
|
|
auio.uio_offset = 0;
|
|
|
|
auio.uio_resid = *buflen;
|
|
|
|
|
|
|
|
if ((ioflg & IO_NODELOCKED) == 0)
|
2013-03-30 15:09:04 +00:00
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
2000-08-08 17:15:32 +00:00
|
|
|
|
2005-02-24 00:13:16 +00:00
|
|
|
ASSERT_VOP_LOCKED(vp, "IO_NODELOCKED with no vp lock held");
|
|
|
|
|
2000-08-08 17:15:32 +00:00
|
|
|
/* authorize attribute retrieval as kernel */
|
Part I: Update extended attribute API and ABI:
o Modify the system call syntax for extattr_{get,set}_{fd,file}() so
as not to use the scatter gather API (which appeared not to be used
by any consumers, and be less portable), rather, accepts 'data'
and 'nbytes' in the style of other simple read/write interfaces.
This changes the API and ABI.
o Modify system call semantics so that extattr_get_{fd,file}() return
a size_t. When performing a read, the number of bytes read will
be returned, unless the data pointer is NULL, in which case the
number of bytes of data are returned. This changes the API only.
o Modify the VOP_GETEXTATTR() vnode operation to accept a *size_t
argument so as to return the size, if desirable. If set to NULL,
the size will not be returned.
o Update various filesystems (pseodofs, ufs) to DTRT.
These changes should make extended attributes more useful and more
portable. More commits to rebuild the system call files, as well
as update userland utilities to follow.
Obtained from: TrustedBSD Project
Sponsored by: DARPA, NAI Labs
2002-02-10 04:43:22 +00:00
|
|
|
error = VOP_GETEXTATTR(vp, attrnamespace, attrname, &auio, NULL, NULL,
|
|
|
|
td);
|
2000-08-08 17:15:32 +00:00
|
|
|
|
|
|
|
if ((ioflg & IO_NODELOCKED) == 0)
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2000-08-08 17:15:32 +00:00
|
|
|
|
|
|
|
if (error == 0) {
|
|
|
|
*buflen = *buflen - auio.uio_resid;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX failure mode if partially written?
|
|
|
|
*/
|
|
|
|
int
|
2001-03-19 05:44:15 +00:00
|
|
|
vn_extattr_set(struct vnode *vp, int ioflg, int attrnamespace,
|
2001-09-12 08:38:13 +00:00
|
|
|
const char *attrname, int buflen, char *buf, struct thread *td)
|
2000-08-08 17:15:32 +00:00
|
|
|
{
|
|
|
|
struct uio auio;
|
|
|
|
struct iovec iov;
|
2000-09-05 03:15:02 +00:00
|
|
|
struct mount *mp;
|
2000-08-08 17:15:32 +00:00
|
|
|
int error;
|
|
|
|
|
|
|
|
iov.iov_len = buflen;
|
|
|
|
iov.iov_base = buf;
|
|
|
|
|
|
|
|
auio.uio_iov = &iov;
|
|
|
|
auio.uio_iovcnt = 1;
|
|
|
|
auio.uio_rw = UIO_WRITE;
|
|
|
|
auio.uio_segflg = UIO_SYSSPACE;
|
2001-09-12 08:38:13 +00:00
|
|
|
auio.uio_td = td;
|
2000-08-08 17:15:32 +00:00
|
|
|
auio.uio_offset = 0;
|
|
|
|
auio.uio_resid = buflen;
|
|
|
|
|
2000-09-05 03:15:02 +00:00
|
|
|
if ((ioflg & IO_NODELOCKED) == 0) {
|
|
|
|
if ((error = vn_start_write(vp, &mp, V_WAIT)) != 0)
|
|
|
|
return (error);
|
2008-01-10 01:10:58 +00:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY);
|
2000-09-05 03:15:02 +00:00
|
|
|
}
|
2000-08-08 17:15:32 +00:00
|
|
|
|
2005-02-24 00:13:16 +00:00
|
|
|
ASSERT_VOP_LOCKED(vp, "IO_NODELOCKED with no vp lock held");
|
|
|
|
|
2000-08-08 17:15:32 +00:00
|
|
|
/* authorize attribute setting as kernel */
|
2001-09-12 08:38:13 +00:00
|
|
|
error = VOP_SETEXTATTR(vp, attrnamespace, attrname, &auio, NULL, td);
|
2000-08-08 17:15:32 +00:00
|
|
|
|
2000-09-05 03:15:02 +00:00
|
|
|
if ((ioflg & IO_NODELOCKED) == 0) {
|
|
|
|
vn_finished_write(mp);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2000-09-05 03:15:02 +00:00
|
|
|
}
|
2000-08-08 17:15:32 +00:00
|
|
|
|
|
|
|
return (error);
|
|
|
|
}
|
2000-09-22 22:33:13 +00:00
|
|
|
|
|
|
|
int
|
2001-03-19 05:44:15 +00:00
|
|
|
vn_extattr_rm(struct vnode *vp, int ioflg, int attrnamespace,
|
2001-09-12 08:38:13 +00:00
|
|
|
const char *attrname, struct thread *td)
|
2000-09-22 22:33:13 +00:00
|
|
|
{
|
|
|
|
struct mount *mp;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
if ((ioflg & IO_NODELOCKED) == 0) {
|
|
|
|
if ((error = vn_start_write(vp, &mp, V_WAIT)) != 0)
|
|
|
|
return (error);
|
2008-01-10 01:10:58 +00:00
|
|
|
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY);
|
2000-09-22 22:33:13 +00:00
|
|
|
}
|
|
|
|
|
2005-02-24 00:13:16 +00:00
|
|
|
ASSERT_VOP_LOCKED(vp, "IO_NODELOCKED with no vp lock held");
|
|
|
|
|
2000-09-22 22:33:13 +00:00
|
|
|
/* authorize attribute removal as kernel */
|
2003-07-28 18:53:29 +00:00
|
|
|
error = VOP_DELETEEXTATTR(vp, attrnamespace, attrname, NULL, td);
|
2003-06-22 23:03:07 +00:00
|
|
|
if (error == EOPNOTSUPP)
|
|
|
|
error = VOP_SETEXTATTR(vp, attrnamespace, attrname, NULL,
|
|
|
|
NULL, td);
|
2000-09-22 22:33:13 +00:00
|
|
|
|
|
|
|
if ((ioflg & IO_NODELOCKED) == 0) {
|
|
|
|
vn_finished_write(mp);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2000-09-22 22:33:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (error);
|
|
|
|
}
|
2009-01-21 14:51:38 +00:00
|
|
|
|
2014-07-14 08:34:54 +00:00
|
|
|
static int
|
|
|
|
vn_get_ino_alloc_vget(struct mount *mp, void *arg, int lkflags,
|
|
|
|
struct vnode **rvp)
|
|
|
|
{
|
|
|
|
|
|
|
|
return (VFS_VGET(mp, *(ino_t *)arg, lkflags, rvp));
|
|
|
|
}
|
|
|
|
|
2009-01-21 14:51:38 +00:00
|
|
|
int
|
|
|
|
vn_vget_ino(struct vnode *vp, ino_t ino, int lkflags, struct vnode **rvp)
|
2014-07-14 08:34:54 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
return (vn_vget_ino_gen(vp, vn_get_ino_alloc_vget, &ino,
|
|
|
|
lkflags, rvp));
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
vn_vget_ino_gen(struct vnode *vp, vn_get_ino_t alloc, void *alloc_arg,
|
|
|
|
int lkflags, struct vnode **rvp)
|
2009-01-21 14:51:38 +00:00
|
|
|
{
|
|
|
|
struct mount *mp;
|
|
|
|
int ltype, error;
|
|
|
|
|
2014-07-14 08:34:54 +00:00
|
|
|
ASSERT_VOP_LOCKED(vp, "vn_vget_ino_get");
|
2009-01-21 14:51:38 +00:00
|
|
|
mp = vp->v_mount;
|
|
|
|
ltype = VOP_ISLOCKED(vp);
|
|
|
|
KASSERT(ltype == LK_EXCLUSIVE || ltype == LK_SHARED,
|
|
|
|
("vn_vget_ino: vp not locked"));
|
2009-05-07 18:14:21 +00:00
|
|
|
error = vfs_busy(mp, MBF_NOWAIT);
|
|
|
|
if (error != 0) {
|
2009-07-02 18:02:55 +00:00
|
|
|
vfs_ref(mp);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2009-05-07 18:14:21 +00:00
|
|
|
error = vfs_busy(mp, 0);
|
2009-01-21 14:51:38 +00:00
|
|
|
vn_lock(vp, ltype | LK_RETRY);
|
2009-07-02 18:02:55 +00:00
|
|
|
vfs_rel(mp);
|
2009-05-07 18:14:21 +00:00
|
|
|
if (error != 0)
|
2009-01-21 14:51:38 +00:00
|
|
|
return (ENOENT);
|
2019-12-08 21:30:04 +00:00
|
|
|
if (VN_IS_DOOMED(vp)) {
|
2009-05-07 18:14:21 +00:00
|
|
|
vfs_unbusy(mp);
|
|
|
|
return (ENOENT);
|
|
|
|
}
|
2009-01-21 14:51:38 +00:00
|
|
|
}
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2014-07-14 08:34:54 +00:00
|
|
|
error = alloc(mp, alloc_arg, lkflags, rvp);
|
2009-01-21 14:51:38 +00:00
|
|
|
vfs_unbusy(mp);
|
2019-08-27 08:28:38 +00:00
|
|
|
if (error != 0 || *rvp != vp)
|
2014-07-14 08:34:54 +00:00
|
|
|
vn_lock(vp, ltype | LK_RETRY);
|
2019-12-08 21:30:04 +00:00
|
|
|
if (VN_IS_DOOMED(vp)) {
|
2014-07-14 08:34:54 +00:00
|
|
|
if (error == 0) {
|
|
|
|
if (*rvp == vp)
|
|
|
|
vunref(vp);
|
|
|
|
else
|
|
|
|
vput(*rvp);
|
|
|
|
}
|
2009-01-21 14:51:38 +00:00
|
|
|
error = ENOENT;
|
|
|
|
}
|
|
|
|
return (error);
|
|
|
|
}
|
2010-05-05 16:44:25 +00:00
|
|
|
|
|
|
|
int
|
2010-05-06 18:43:19 +00:00
|
|
|
vn_rlimit_fsize(const struct vnode *vp, const struct uio *uio,
|
2015-06-10 10:48:12 +00:00
|
|
|
struct thread *td)
|
2010-05-05 16:44:25 +00:00
|
|
|
{
|
2010-05-06 18:43:19 +00:00
|
|
|
|
2010-05-05 16:44:25 +00:00
|
|
|
if (vp->v_type != VREG || td == NULL)
|
|
|
|
return (0);
|
2010-05-06 18:52:41 +00:00
|
|
|
if ((uoff_t)uio->uio_offset + uio->uio_resid >
|
2015-06-10 10:48:12 +00:00
|
|
|
lim_cur(td, RLIMIT_FSIZE)) {
|
|
|
|
PROC_LOCK(td->td_proc);
|
2011-09-16 13:58:51 +00:00
|
|
|
kern_psignal(td->td_proc, SIGXFSZ);
|
2010-05-05 16:44:25 +00:00
|
|
|
PROC_UNLOCK(td->td_proc);
|
|
|
|
return (EFBIG);
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
2011-08-16 20:07:47 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
vn_chmod(struct file *fp, mode_t mode, struct ucred *active_cred,
|
|
|
|
struct thread *td)
|
|
|
|
{
|
|
|
|
struct vnode *vp;
|
|
|
|
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
#ifdef AUDIT
|
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
|
|
|
AUDIT_ARG_VNODE1(vp);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2011-08-16 20:07:47 +00:00
|
|
|
#endif
|
2013-03-01 21:58:56 +00:00
|
|
|
return (setfmode(td, active_cred, vp, mode));
|
2011-08-16 20:07:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
vn_chown(struct file *fp, uid_t uid, gid_t gid, struct ucred *active_cred,
|
|
|
|
struct thread *td)
|
|
|
|
{
|
|
|
|
struct vnode *vp;
|
|
|
|
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
#ifdef AUDIT
|
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
|
|
|
AUDIT_ARG_VNODE1(vp);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2011-08-16 20:07:47 +00:00
|
|
|
#endif
|
2013-03-01 21:58:56 +00:00
|
|
|
return (setfown(td, active_cred, vp, uid, gid));
|
2011-08-16 20:07:47 +00:00
|
|
|
}
|
2011-08-25 08:17:39 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
vn_pages_remove(struct vnode *vp, vm_pindex_t start, vm_pindex_t end)
|
|
|
|
{
|
|
|
|
vm_object_t object;
|
|
|
|
|
|
|
|
if ((object = vp->v_object) == NULL)
|
|
|
|
return;
|
2013-03-09 02:32:23 +00:00
|
|
|
VM_OBJECT_WLOCK(object);
|
2011-08-25 08:17:39 +00:00
|
|
|
vm_object_page_remove(object, start, end, 0);
|
2013-03-09 02:32:23 +00:00
|
|
|
VM_OBJECT_WUNLOCK(object);
|
2011-08-25 08:17:39 +00:00
|
|
|
}
|
2012-05-26 05:28:47 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
vn_bmap_seekhole(struct vnode *vp, u_long cmd, off_t *off, struct ucred *cred)
|
|
|
|
{
|
|
|
|
struct vattr va;
|
|
|
|
daddr_t bn, bnp;
|
|
|
|
uint64_t bsize;
|
|
|
|
off_t noff;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
KASSERT(cmd == FIOSEEKHOLE || cmd == FIOSEEKDATA,
|
|
|
|
("Wrong command %lu", cmd));
|
|
|
|
|
|
|
|
if (vn_lock(vp, LK_SHARED) != 0)
|
|
|
|
return (EBADF);
|
|
|
|
if (vp->v_type != VREG) {
|
|
|
|
error = ENOTTY;
|
|
|
|
goto unlock;
|
|
|
|
}
|
|
|
|
error = VOP_GETATTR(vp, &va, cred);
|
|
|
|
if (error != 0)
|
|
|
|
goto unlock;
|
|
|
|
noff = *off;
|
|
|
|
if (noff >= va.va_size) {
|
|
|
|
error = ENXIO;
|
|
|
|
goto unlock;
|
|
|
|
}
|
|
|
|
bsize = vp->v_mount->mnt_stat.f_iosize;
|
2019-04-05 16:14:16 +00:00
|
|
|
for (bn = noff / bsize; noff < va.va_size; bn++, noff += bsize -
|
|
|
|
noff % bsize) {
|
2012-05-26 05:28:47 +00:00
|
|
|
error = VOP_BMAP(vp, bn, NULL, &bnp, NULL, NULL);
|
|
|
|
if (error == EOPNOTSUPP) {
|
|
|
|
error = ENOTTY;
|
|
|
|
goto unlock;
|
|
|
|
}
|
|
|
|
if ((bnp == -1 && cmd == FIOSEEKHOLE) ||
|
|
|
|
(bnp != -1 && cmd == FIOSEEKDATA)) {
|
|
|
|
noff = bn * bsize;
|
|
|
|
if (noff < *off)
|
|
|
|
noff = *off;
|
|
|
|
goto unlock;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (noff > va.va_size)
|
|
|
|
noff = va.va_size;
|
|
|
|
/* noff == va.va_size. There is an implicit hole at the end of file. */
|
|
|
|
if (cmd == FIOSEEKDATA)
|
|
|
|
error = ENXIO;
|
|
|
|
unlock:
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2012-05-26 05:28:47 +00:00
|
|
|
if (error == 0)
|
|
|
|
*off = noff;
|
|
|
|
return (error);
|
|
|
|
}
|
2013-08-21 17:36:01 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
vn_seek(struct file *fp, off_t offset, int whence, struct thread *td)
|
|
|
|
{
|
|
|
|
struct ucred *cred;
|
|
|
|
struct vnode *vp;
|
|
|
|
struct vattr vattr;
|
|
|
|
off_t foffset, size;
|
|
|
|
int error, noneg;
|
|
|
|
|
|
|
|
cred = td->td_ucred;
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
foffset = foffset_lock(fp, 0);
|
|
|
|
noneg = (vp->v_type != VCHR);
|
|
|
|
error = 0;
|
|
|
|
switch (whence) {
|
|
|
|
case L_INCR:
|
|
|
|
if (noneg &&
|
|
|
|
(foffset < 0 ||
|
|
|
|
(offset > 0 && foffset > OFF_MAX - offset))) {
|
|
|
|
error = EOVERFLOW;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
offset += foffset;
|
|
|
|
break;
|
|
|
|
case L_XTND:
|
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
|
|
|
error = VOP_GETATTR(vp, &vattr, cred);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2013-08-21 17:36:01 +00:00
|
|
|
if (error)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the file references a disk device, then fetch
|
|
|
|
* the media size and use that to determine the ending
|
|
|
|
* offset.
|
|
|
|
*/
|
|
|
|
if (vattr.va_size == 0 && vp->v_type == VCHR &&
|
|
|
|
fo_ioctl(fp, DIOCGMEDIASIZE, &size, cred, td) == 0)
|
|
|
|
vattr.va_size = size;
|
|
|
|
if (noneg &&
|
|
|
|
(vattr.va_size > OFF_MAX ||
|
|
|
|
(offset > 0 && vattr.va_size > OFF_MAX - offset))) {
|
|
|
|
error = EOVERFLOW;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
offset += vattr.va_size;
|
|
|
|
break;
|
|
|
|
case L_SET:
|
|
|
|
break;
|
|
|
|
case SEEK_DATA:
|
|
|
|
error = fo_ioctl(fp, FIOSEEKDATA, &offset, cred, td);
|
2019-08-22 01:15:06 +00:00
|
|
|
if (error == ENOTTY)
|
|
|
|
error = EINVAL;
|
2013-08-21 17:36:01 +00:00
|
|
|
break;
|
|
|
|
case SEEK_HOLE:
|
|
|
|
error = fo_ioctl(fp, FIOSEEKHOLE, &offset, cred, td);
|
2019-08-22 01:15:06 +00:00
|
|
|
if (error == ENOTTY)
|
|
|
|
error = EINVAL;
|
2013-08-21 17:36:01 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
error = EINVAL;
|
|
|
|
}
|
|
|
|
if (error == 0 && noneg && offset < 0)
|
|
|
|
error = EINVAL;
|
|
|
|
if (error != 0)
|
|
|
|
goto drop;
|
|
|
|
VFS_KNOTE_UNLOCKED(vp, 0);
|
2014-03-16 00:53:40 +00:00
|
|
|
td->td_uretoff.tdu_off = offset;
|
2013-08-21 17:36:01 +00:00
|
|
|
drop:
|
|
|
|
foffset_unlock(fp, offset, error != 0 ? FOF_NOUPDATE : 0);
|
|
|
|
return (error);
|
|
|
|
}
|
2014-06-17 07:11:00 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
vn_utimes_perm(struct vnode *vp, struct vattr *vap, struct ucred *cred,
|
|
|
|
struct thread *td)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
|
|
|
|
/*
|
2014-10-04 18:51:55 +00:00
|
|
|
* Grant permission if the caller is the owner of the file, or
|
|
|
|
* the super-user, or has ACL_WRITE_ATTRIBUTES permission on
|
|
|
|
* on the file. If the time pointer is null, then write
|
2014-06-17 07:11:00 +00:00
|
|
|
* permission on the file is also sufficient.
|
|
|
|
*
|
|
|
|
* From NFSv4.1, draft 21, 6.2.1.3.1, Discussion of Mask Attributes:
|
|
|
|
* A user having ACL_WRITE_DATA or ACL_WRITE_ATTRIBUTES
|
|
|
|
* will be allowed to set the times [..] to the current
|
|
|
|
* server time.
|
|
|
|
*/
|
2014-10-04 18:51:55 +00:00
|
|
|
error = VOP_ACCESSX(vp, VWRITE_ATTRIBUTES, cred, td);
|
2014-06-17 07:11:00 +00:00
|
|
|
if (error != 0 && (vap->va_vaflags & VA_UTIMES_NULL) != 0)
|
|
|
|
error = VOP_ACCESS(vp, VWRITE, cred, td);
|
|
|
|
return (error);
|
|
|
|
}
|
2014-09-22 16:20:47 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
vn_fill_kinfo(struct file *fp, struct kinfo_file *kif, struct filedesc *fdp)
|
|
|
|
{
|
|
|
|
struct vnode *vp;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
if (fp->f_type == DTYPE_FIFO)
|
|
|
|
kif->kf_type = KF_TYPE_FIFO;
|
|
|
|
else
|
|
|
|
kif->kf_type = KF_TYPE_VNODE;
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
vref(vp);
|
|
|
|
FILEDESC_SUNLOCK(fdp);
|
|
|
|
error = vn_fill_kinfo_vnode(vp, kif);
|
|
|
|
vrele(vp);
|
|
|
|
FILEDESC_SLOCK(fdp);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
Detect badly behaved coredump note helpers
Coredump notes depend on being able to invoke dump routines twice; once
in a dry-run mode to get the size of the note, and another to actually
emit the note to the corefile.
When a note helper emits a different length section the second time
around than the length it requested the first time, the kernel produces
a corrupt coredump.
NT_PROCSTAT_FILES output length, when packing kinfo structs, is tied to
the length of filenames corresponding to vnodes in the process' fd table
via vn_fullpath. As vnodes may move around during dump, this is racy.
So:
- Detect badly behaved notes in putnote() and pad underfilled notes.
- Add a fail point, debug.fail_point.fill_kinfo_vnode__random_path to
exercise the NT_PROCSTAT_FILES corruption. It simply picks random
lengths to expand or truncate paths to in fo_fill_kinfo_vnode().
- Add a sysctl, kern.coredump_pack_fileinfo, to allow users to
disable kinfo packing for PROCSTAT_FILES notes. This should avoid
both FILES note corruption and truncation, even if filenames change,
at the cost of about 1 kiB in padding bloat per open fd. Document
the new sysctl in core.5.
- Fix note_procstat_files to self-limit in the 2nd pass. Since
sometimes this will result in a short write, pad up to our advertised
size. This addresses note corruption, at the risk of sometimes
truncating the last several fd info entries.
- Fix NT_PROCSTAT_FILES consumers libutil and libprocstat to grok the
zero padding.
With suggestions from: bjk, jhb, kib, wblock
Approved by: markj (mentor)
Relnotes: yes
Sponsored by: EMC / Isilon Storage Division
Differential Revision: https://reviews.freebsd.org/D3548
2015-09-03 20:32:10 +00:00
|
|
|
static inline void
|
|
|
|
vn_fill_junk(struct kinfo_file *kif)
|
|
|
|
{
|
|
|
|
size_t len, olen;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simulate vn_fullpath returning changing values for a given
|
|
|
|
* vp during e.g. coredump.
|
|
|
|
*/
|
|
|
|
len = (arc4random() % (sizeof(kif->kf_path) - 2)) + 1;
|
|
|
|
olen = strlen(kif->kf_path);
|
|
|
|
if (len < olen)
|
|
|
|
strcpy(&kif->kf_path[len - 1], "$");
|
|
|
|
else
|
|
|
|
for (; olen < len; olen++)
|
|
|
|
strcpy(&kif->kf_path[olen], "A");
|
|
|
|
}
|
|
|
|
|
2014-09-22 16:20:47 +00:00
|
|
|
int
|
|
|
|
vn_fill_kinfo_vnode(struct vnode *vp, struct kinfo_file *kif)
|
|
|
|
{
|
|
|
|
struct vattr va;
|
|
|
|
char *fullpath, *freepath;
|
|
|
|
int error;
|
|
|
|
|
Commit the 64-bit inode project.
Extend the ino_t, dev_t, nlink_t types to 64-bit ints. Modify
struct dirent layout to add d_off, increase the size of d_fileno
to 64-bits, increase the size of d_namlen to 16-bits, and change
the required alignment. Increase struct statfs f_mntfromname[] and
f_mntonname[] array length MNAMELEN to 1024.
ABI breakage is mitigated by providing compatibility using versioned
symbols, ingenious use of the existing padding in structures, and
by employing other tricks. Unfortunately, not everything can be
fixed, especially outside the base system. For instance, third-party
APIs which pass struct stat around are broken in backward and
forward incompatible ways.
Kinfo sysctl MIBs ABI is changed in backward-compatible way, but
there is no general mechanism to handle other sysctl MIBS which
return structures where the layout has changed. It was considered
that the breakage is either in the management interfaces, where we
usually allow ABI slip, or is not important.
Struct xvnode changed layout, no compat shims are provided.
For struct xtty, dev_t tty device member was reduced to uint32_t.
It was decided that keeping ABI compat in this case is more useful
than reporting 64-bit dev_t, for the sake of pstat.
Update note: strictly follow the instructions in UPDATING. Build
and install the new kernel with COMPAT_FREEBSD11 option enabled,
then reboot, and only then install new world.
Credits: The 64-bit inode project, also known as ino64, started life
many years ago as a project by Gleb Kurtsou (gleb). Kirk McKusick
(mckusick) then picked up and updated the patch, and acted as a
flag-waver. Feedback, suggestions, and discussions were carried
by Ed Maste (emaste), John Baldwin (jhb), Jilles Tjoelker (jilles),
and Rick Macklem (rmacklem). Kris Moore (kris) performed an initial
ports investigation followed by an exp-run by Antoine Brodin (antoine).
Essential and all-embracing testing was done by Peter Holm (pho).
The heavy lifting of coordinating all these efforts and bringing the
project to completion were done by Konstantin Belousov (kib).
Sponsored by: The FreeBSD Foundation (emaste, kib)
Differential revision: https://reviews.freebsd.org/D10439
2017-05-23 09:29:05 +00:00
|
|
|
kif->kf_un.kf_file.kf_file_type = vntype_to_kinfo(vp->v_type);
|
2014-09-22 16:20:47 +00:00
|
|
|
freepath = NULL;
|
|
|
|
fullpath = "-";
|
|
|
|
error = vn_fullpath(curthread, vp, &fullpath, &freepath);
|
|
|
|
if (error == 0) {
|
|
|
|
strlcpy(kif->kf_path, fullpath, sizeof(kif->kf_path));
|
|
|
|
}
|
|
|
|
if (freepath != NULL)
|
|
|
|
free(freepath, M_TEMP);
|
|
|
|
|
Detect badly behaved coredump note helpers
Coredump notes depend on being able to invoke dump routines twice; once
in a dry-run mode to get the size of the note, and another to actually
emit the note to the corefile.
When a note helper emits a different length section the second time
around than the length it requested the first time, the kernel produces
a corrupt coredump.
NT_PROCSTAT_FILES output length, when packing kinfo structs, is tied to
the length of filenames corresponding to vnodes in the process' fd table
via vn_fullpath. As vnodes may move around during dump, this is racy.
So:
- Detect badly behaved notes in putnote() and pad underfilled notes.
- Add a fail point, debug.fail_point.fill_kinfo_vnode__random_path to
exercise the NT_PROCSTAT_FILES corruption. It simply picks random
lengths to expand or truncate paths to in fo_fill_kinfo_vnode().
- Add a sysctl, kern.coredump_pack_fileinfo, to allow users to
disable kinfo packing for PROCSTAT_FILES notes. This should avoid
both FILES note corruption and truncation, even if filenames change,
at the cost of about 1 kiB in padding bloat per open fd. Document
the new sysctl in core.5.
- Fix note_procstat_files to self-limit in the 2nd pass. Since
sometimes this will result in a short write, pad up to our advertised
size. This addresses note corruption, at the risk of sometimes
truncating the last several fd info entries.
- Fix NT_PROCSTAT_FILES consumers libutil and libprocstat to grok the
zero padding.
With suggestions from: bjk, jhb, kib, wblock
Approved by: markj (mentor)
Relnotes: yes
Sponsored by: EMC / Isilon Storage Division
Differential Revision: https://reviews.freebsd.org/D3548
2015-09-03 20:32:10 +00:00
|
|
|
KFAIL_POINT_CODE(DEBUG_FP, fill_kinfo_vnode__random_path,
|
|
|
|
vn_fill_junk(kif);
|
|
|
|
);
|
|
|
|
|
2014-09-22 16:20:47 +00:00
|
|
|
/*
|
|
|
|
* Retrieve vnode attributes.
|
|
|
|
*/
|
|
|
|
va.va_fsid = VNOVAL;
|
|
|
|
va.va_rdev = NODEV;
|
|
|
|
vn_lock(vp, LK_SHARED | LK_RETRY);
|
|
|
|
error = VOP_GETATTR(vp, &va, curthread->td_ucred);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(vp);
|
2014-09-22 16:20:47 +00:00
|
|
|
if (error != 0)
|
|
|
|
return (error);
|
|
|
|
if (va.va_fsid != VNOVAL)
|
|
|
|
kif->kf_un.kf_file.kf_file_fsid = va.va_fsid;
|
|
|
|
else
|
|
|
|
kif->kf_un.kf_file.kf_file_fsid =
|
|
|
|
vp->v_mount->mnt_stat.f_fsid.val[0];
|
Commit the 64-bit inode project.
Extend the ino_t, dev_t, nlink_t types to 64-bit ints. Modify
struct dirent layout to add d_off, increase the size of d_fileno
to 64-bits, increase the size of d_namlen to 16-bits, and change
the required alignment. Increase struct statfs f_mntfromname[] and
f_mntonname[] array length MNAMELEN to 1024.
ABI breakage is mitigated by providing compatibility using versioned
symbols, ingenious use of the existing padding in structures, and
by employing other tricks. Unfortunately, not everything can be
fixed, especially outside the base system. For instance, third-party
APIs which pass struct stat around are broken in backward and
forward incompatible ways.
Kinfo sysctl MIBs ABI is changed in backward-compatible way, but
there is no general mechanism to handle other sysctl MIBS which
return structures where the layout has changed. It was considered
that the breakage is either in the management interfaces, where we
usually allow ABI slip, or is not important.
Struct xvnode changed layout, no compat shims are provided.
For struct xtty, dev_t tty device member was reduced to uint32_t.
It was decided that keeping ABI compat in this case is more useful
than reporting 64-bit dev_t, for the sake of pstat.
Update note: strictly follow the instructions in UPDATING. Build
and install the new kernel with COMPAT_FREEBSD11 option enabled,
then reboot, and only then install new world.
Credits: The 64-bit inode project, also known as ino64, started life
many years ago as a project by Gleb Kurtsou (gleb). Kirk McKusick
(mckusick) then picked up and updated the patch, and acted as a
flag-waver. Feedback, suggestions, and discussions were carried
by Ed Maste (emaste), John Baldwin (jhb), Jilles Tjoelker (jilles),
and Rick Macklem (rmacklem). Kris Moore (kris) performed an initial
ports investigation followed by an exp-run by Antoine Brodin (antoine).
Essential and all-embracing testing was done by Peter Holm (pho).
The heavy lifting of coordinating all these efforts and bringing the
project to completion were done by Konstantin Belousov (kib).
Sponsored by: The FreeBSD Foundation (emaste, kib)
Differential revision: https://reviews.freebsd.org/D10439
2017-05-23 09:29:05 +00:00
|
|
|
kif->kf_un.kf_file.kf_file_fsid_freebsd11 =
|
|
|
|
kif->kf_un.kf_file.kf_file_fsid; /* truncate */
|
2014-09-22 16:20:47 +00:00
|
|
|
kif->kf_un.kf_file.kf_file_fileid = va.va_fileid;
|
|
|
|
kif->kf_un.kf_file.kf_file_mode = MAKEIMODE(va.va_type, va.va_mode);
|
|
|
|
kif->kf_un.kf_file.kf_file_size = va.va_size;
|
|
|
|
kif->kf_un.kf_file.kf_file_rdev = va.va_rdev;
|
Commit the 64-bit inode project.
Extend the ino_t, dev_t, nlink_t types to 64-bit ints. Modify
struct dirent layout to add d_off, increase the size of d_fileno
to 64-bits, increase the size of d_namlen to 16-bits, and change
the required alignment. Increase struct statfs f_mntfromname[] and
f_mntonname[] array length MNAMELEN to 1024.
ABI breakage is mitigated by providing compatibility using versioned
symbols, ingenious use of the existing padding in structures, and
by employing other tricks. Unfortunately, not everything can be
fixed, especially outside the base system. For instance, third-party
APIs which pass struct stat around are broken in backward and
forward incompatible ways.
Kinfo sysctl MIBs ABI is changed in backward-compatible way, but
there is no general mechanism to handle other sysctl MIBS which
return structures where the layout has changed. It was considered
that the breakage is either in the management interfaces, where we
usually allow ABI slip, or is not important.
Struct xvnode changed layout, no compat shims are provided.
For struct xtty, dev_t tty device member was reduced to uint32_t.
It was decided that keeping ABI compat in this case is more useful
than reporting 64-bit dev_t, for the sake of pstat.
Update note: strictly follow the instructions in UPDATING. Build
and install the new kernel with COMPAT_FREEBSD11 option enabled,
then reboot, and only then install new world.
Credits: The 64-bit inode project, also known as ino64, started life
many years ago as a project by Gleb Kurtsou (gleb). Kirk McKusick
(mckusick) then picked up and updated the patch, and acted as a
flag-waver. Feedback, suggestions, and discussions were carried
by Ed Maste (emaste), John Baldwin (jhb), Jilles Tjoelker (jilles),
and Rick Macklem (rmacklem). Kris Moore (kris) performed an initial
ports investigation followed by an exp-run by Antoine Brodin (antoine).
Essential and all-embracing testing was done by Peter Holm (pho).
The heavy lifting of coordinating all these efforts and bringing the
project to completion were done by Konstantin Belousov (kib).
Sponsored by: The FreeBSD Foundation (emaste, kib)
Differential revision: https://reviews.freebsd.org/D10439
2017-05-23 09:29:05 +00:00
|
|
|
kif->kf_un.kf_file.kf_file_rdev_freebsd11 =
|
|
|
|
kif->kf_un.kf_file.kf_file_rdev; /* truncate */
|
2014-09-22 16:20:47 +00:00
|
|
|
return (0);
|
|
|
|
}
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
vn_mmap(struct file *fp, vm_map_t map, vm_offset_t *addr, vm_size_t size,
|
|
|
|
vm_prot_t prot, vm_prot_t cap_maxprot, int flags, vm_ooffset_t foff,
|
|
|
|
struct thread *td)
|
|
|
|
{
|
|
|
|
#ifdef HWPMC_HOOKS
|
|
|
|
struct pmckern_map_in pkm;
|
|
|
|
#endif
|
|
|
|
struct mount *mp;
|
|
|
|
struct vnode *vp;
|
|
|
|
vm_object_t object;
|
|
|
|
vm_prot_t maxprot;
|
|
|
|
boolean_t writecounted;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
#if defined(COMPAT_FREEBSD7) || defined(COMPAT_FREEBSD6) || \
|
|
|
|
defined(COMPAT_FREEBSD5) || defined(COMPAT_FREEBSD4)
|
|
|
|
/*
|
|
|
|
* POSIX shared-memory objects are defined to have
|
|
|
|
* kernel persistence, and are not defined to support
|
|
|
|
* read(2)/write(2) -- or even open(2). Thus, we can
|
|
|
|
* use MAP_ASYNC to trade on-disk coherence for speed.
|
|
|
|
* The shm_open(3) library routine turns on the FPOSIXSHM
|
|
|
|
* flag to request this behavior.
|
|
|
|
*/
|
|
|
|
if ((fp->f_flag & FPOSIXSHM) != 0)
|
|
|
|
flags |= MAP_NOSYNC;
|
|
|
|
#endif
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Ensure that file and memory protections are
|
|
|
|
* compatible. Note that we only worry about
|
|
|
|
* writability if mapping is shared; in this case,
|
|
|
|
* current and max prot are dictated by the open file.
|
|
|
|
* XXX use the vnode instead? Problem is: what
|
|
|
|
* credentials do we use for determination? What if
|
|
|
|
* proc does a setuid?
|
|
|
|
*/
|
|
|
|
mp = vp->v_mount;
|
2017-02-19 20:51:04 +00:00
|
|
|
if (mp != NULL && (mp->mnt_flag & MNT_NOEXEC) != 0) {
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
maxprot = VM_PROT_NONE;
|
2017-02-19 20:51:04 +00:00
|
|
|
if ((prot & VM_PROT_EXECUTE) != 0)
|
|
|
|
return (EACCES);
|
|
|
|
} else
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
maxprot = VM_PROT_EXECUTE;
|
|
|
|
if ((fp->f_flag & FREAD) != 0)
|
|
|
|
maxprot |= VM_PROT_READ;
|
|
|
|
else if ((prot & VM_PROT_READ) != 0)
|
|
|
|
return (EACCES);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we are sharing potential changes via MAP_SHARED and we
|
|
|
|
* are trying to get write permission although we opened it
|
|
|
|
* without asking for it, bail out.
|
|
|
|
*/
|
|
|
|
if ((flags & MAP_SHARED) != 0) {
|
|
|
|
if ((fp->f_flag & FWRITE) != 0)
|
|
|
|
maxprot |= VM_PROT_WRITE;
|
|
|
|
else if ((prot & VM_PROT_WRITE) != 0)
|
|
|
|
return (EACCES);
|
|
|
|
} else {
|
|
|
|
maxprot |= VM_PROT_WRITE;
|
|
|
|
cap_maxprot |= VM_PROT_WRITE;
|
|
|
|
}
|
|
|
|
maxprot &= cap_maxprot;
|
|
|
|
|
2017-02-12 21:05:44 +00:00
|
|
|
/*
|
|
|
|
* For regular files and shared memory, POSIX requires that
|
|
|
|
* the value of foff be a legitimate offset within the data
|
|
|
|
* object. In particular, negative offsets are invalid.
|
|
|
|
* Blocking negative offsets and overflows here avoids
|
|
|
|
* possible wraparound or user-level access into reserved
|
|
|
|
* ranges of the data object later. In contrast, POSIX does
|
|
|
|
* not dictate how offsets are used by device drivers, so in
|
|
|
|
* the case of a device mapping a negative offset is passed
|
|
|
|
* on.
|
|
|
|
*/
|
|
|
|
if (
|
|
|
|
#ifdef _LP64
|
|
|
|
size > OFF_MAX ||
|
|
|
|
#endif
|
|
|
|
foff < 0 || foff > OFF_MAX - size)
|
|
|
|
return (EINVAL);
|
|
|
|
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
writecounted = FALSE;
|
|
|
|
error = vm_mmap_vnode(td, size, prot, &maxprot, &flags, vp,
|
|
|
|
&foff, &object, &writecounted);
|
|
|
|
if (error != 0)
|
|
|
|
return (error);
|
|
|
|
error = vm_mmap_object(map, addr, size, prot, maxprot, flags, object,
|
|
|
|
foff, writecounted, td);
|
|
|
|
if (error != 0) {
|
|
|
|
/*
|
|
|
|
* If this mapping was accounted for in the vnode's
|
|
|
|
* writecount, then undo that now.
|
|
|
|
*/
|
|
|
|
if (writecounted)
|
2019-09-03 20:31:48 +00:00
|
|
|
vm_pager_release_writecount(object, 0, size);
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
vm_object_deallocate(object);
|
|
|
|
}
|
|
|
|
#ifdef HWPMC_HOOKS
|
|
|
|
/* Inform hwpmc(4) if an executable is being mapped. */
|
2017-01-27 22:13:15 +00:00
|
|
|
if (PMC_HOOK_INSTALLED(PMC_FN_MMAP)) {
|
|
|
|
if ((prot & VM_PROT_EXECUTE) != 0 && error == 0) {
|
|
|
|
pkm.pm_file = vp;
|
|
|
|
pkm.pm_address = (uintptr_t) *addr;
|
2018-05-29 18:03:48 +00:00
|
|
|
PMC_CALL_HOOK_UNLOCKED(td, PMC_FN_MMAP, (void *) &pkm);
|
2017-01-27 22:13:15 +00:00
|
|
|
}
|
Add a new file operations hook for mmap operations. File type-specific
logic is now placed in the mmap hook implementation rather than requiring
it to be placed in sys/vm/vm_mmap.c. This hook allows new file types to
support mmap() as well as potentially allowing mmap() for existing file
types that do not currently support any mapping.
The vm_mmap() function is now split up into two functions. A new
vm_mmap_object() function handles the "back half" of vm_mmap() and accepts
a referenced VM object to map rather than a (handle, handle_type) tuple.
vm_mmap() is now reduced to converting a (handle, handle_type) tuple to a
a VM object and then calling vm_mmap_object() to handle the actual mapping.
The vm_mmap() function remains for use by other parts of the kernel
(e.g. device drivers and exec) but now only supports mapping vnodes,
character devices, and anonymous memory.
The mmap() system call invokes vm_mmap_object() directly with a NULL object
for anonymous mappings. For mappings using a file descriptor, the
descriptors fo_mmap() hook is invoked instead. The fo_mmap() hook is
responsible for performing type-specific checks and adjustments to
arguments as well as possibly modifying mapping parameters such as flags
or the object offset. The fo_mmap() hook routines then call
vm_mmap_object() to handle the actual mapping.
The fo_mmap() hook is optional. If it is not set, then fo_mmap() will
fail with ENODEV. A fo_mmap() hook is implemented for regular files,
character devices, and shared memory objects (created via shm_open()).
While here, consistently use the VM_PROT_* constants for the vm_prot_t
type for the 'prot' variable passed to vm_mmap() and vm_mmap_object()
as well as the vm_mmap_vnode() and vm_mmap_cdev() helper routines.
Previously some places were using the mmap()-specific PROT_* constants
instead. While this happens to work because PROT_xx == VM_PROT_xx,
using VM_PROT_* is more correct.
Differential Revision: https://reviews.freebsd.org/D2658
Reviewed by: alc (glanced over), kib
MFC after: 1 month
Sponsored by: Chelsio
2015-06-04 19:41:15 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return (error);
|
|
|
|
}
|
2017-05-27 17:00:30 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
vn_fsid(struct vnode *vp, struct vattr *va)
|
|
|
|
{
|
|
|
|
fsid_t *f;
|
|
|
|
|
|
|
|
f = &vp->v_mount->mnt_stat.f_fsid;
|
|
|
|
va->va_fsid = (uint32_t)f->val[1];
|
|
|
|
va->va_fsid <<= sizeof(f->val[1]) * NBBY;
|
|
|
|
va->va_fsid += (uint32_t)f->val[0];
|
|
|
|
}
|
2019-04-09 20:20:04 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
vn_fsync_buf(struct vnode *vp, int waitfor)
|
|
|
|
{
|
|
|
|
struct buf *bp, *nbp;
|
|
|
|
struct bufobj *bo;
|
|
|
|
struct mount *mp;
|
|
|
|
int error, maxretry;
|
|
|
|
|
|
|
|
error = 0;
|
|
|
|
maxretry = 10000; /* large, arbitrarily chosen */
|
|
|
|
mp = NULL;
|
|
|
|
if (vp->v_type == VCHR) {
|
|
|
|
VI_LOCK(vp);
|
|
|
|
mp = vp->v_rdev->si_mountpt;
|
|
|
|
VI_UNLOCK(vp);
|
|
|
|
}
|
|
|
|
bo = &vp->v_bufobj;
|
|
|
|
BO_LOCK(bo);
|
|
|
|
loop1:
|
|
|
|
/*
|
|
|
|
* MARK/SCAN initialization to avoid infinite loops.
|
|
|
|
*/
|
|
|
|
TAILQ_FOREACH(bp, &bo->bo_dirty.bv_hd, b_bobufs) {
|
|
|
|
bp->b_vflags &= ~BV_SCANNED;
|
|
|
|
bp->b_error = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Flush all dirty buffers associated with a vnode.
|
|
|
|
*/
|
|
|
|
loop2:
|
|
|
|
TAILQ_FOREACH_SAFE(bp, &bo->bo_dirty.bv_hd, b_bobufs, nbp) {
|
|
|
|
if ((bp->b_vflags & BV_SCANNED) != 0)
|
|
|
|
continue;
|
|
|
|
bp->b_vflags |= BV_SCANNED;
|
|
|
|
if (BUF_LOCK(bp, LK_EXCLUSIVE | LK_NOWAIT, NULL)) {
|
|
|
|
if (waitfor != MNT_WAIT)
|
|
|
|
continue;
|
|
|
|
if (BUF_LOCK(bp,
|
|
|
|
LK_EXCLUSIVE | LK_INTERLOCK | LK_SLEEPFAIL,
|
|
|
|
BO_LOCKPTR(bo)) != 0) {
|
|
|
|
BO_LOCK(bo);
|
|
|
|
goto loop1;
|
|
|
|
}
|
|
|
|
BO_LOCK(bo);
|
|
|
|
}
|
|
|
|
BO_UNLOCK(bo);
|
|
|
|
KASSERT(bp->b_bufobj == bo,
|
|
|
|
("bp %p wrong b_bufobj %p should be %p",
|
|
|
|
bp, bp->b_bufobj, bo));
|
|
|
|
if ((bp->b_flags & B_DELWRI) == 0)
|
|
|
|
panic("fsync: not dirty");
|
|
|
|
if ((vp->v_object != NULL) && (bp->b_flags & B_CLUSTEROK)) {
|
|
|
|
vfs_bio_awrite(bp);
|
|
|
|
} else {
|
|
|
|
bremfree(bp);
|
|
|
|
bawrite(bp);
|
|
|
|
}
|
|
|
|
if (maxretry < 1000)
|
|
|
|
pause("dirty", hz < 1000 ? 1 : hz / 1000);
|
|
|
|
BO_LOCK(bo);
|
|
|
|
goto loop2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If synchronous the caller expects us to completely resolve all
|
|
|
|
* dirty buffers in the system. Wait for in-progress I/O to
|
|
|
|
* complete (which could include background bitmap writes), then
|
|
|
|
* retry if dirty blocks still exist.
|
|
|
|
*/
|
|
|
|
if (waitfor == MNT_WAIT) {
|
|
|
|
bufobj_wwait(bo, 0, 0);
|
|
|
|
if (bo->bo_dirty.bv_cnt > 0) {
|
|
|
|
/*
|
|
|
|
* If we are unable to write any of these buffers
|
|
|
|
* then we fail now rather than trying endlessly
|
|
|
|
* to write them out.
|
|
|
|
*/
|
|
|
|
TAILQ_FOREACH(bp, &bo->bo_dirty.bv_hd, b_bobufs)
|
|
|
|
if ((error = bp->b_error) != 0)
|
|
|
|
break;
|
|
|
|
if ((mp != NULL && mp->mnt_secondary_writes > 0) ||
|
|
|
|
(error == 0 && --maxretry >= 0))
|
|
|
|
goto loop1;
|
|
|
|
if (error == 0)
|
|
|
|
error = EAGAIN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
BO_UNLOCK(bo);
|
|
|
|
if (error != 0)
|
|
|
|
vn_printf(vp, "fsync: giving up on dirty (error = %d) ", error);
|
|
|
|
|
|
|
|
return (error);
|
|
|
|
}
|
2019-07-25 05:46:16 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copies a byte range from invp to outvp. Calls VOP_COPY_FILE_RANGE()
|
|
|
|
* or vn_generic_copy_file_range() after rangelocking the byte ranges,
|
|
|
|
* to do the actual copy.
|
|
|
|
* vn_generic_copy_file_range() is factored out, so it can be called
|
|
|
|
* from a VOP_COPY_FILE_RANGE() call as well, but handles vnodes from
|
|
|
|
* different file systems.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
vn_copy_file_range(struct vnode *invp, off_t *inoffp, struct vnode *outvp,
|
|
|
|
off_t *outoffp, size_t *lenp, unsigned int flags, struct ucred *incred,
|
|
|
|
struct ucred *outcred, struct thread *fsize_td)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
size_t len;
|
|
|
|
uint64_t uvalin, uvalout;
|
|
|
|
|
|
|
|
len = *lenp;
|
|
|
|
*lenp = 0; /* For error returns. */
|
|
|
|
error = 0;
|
|
|
|
|
|
|
|
/* Do some sanity checks on the arguments. */
|
|
|
|
uvalin = *inoffp;
|
|
|
|
uvalin += len;
|
|
|
|
uvalout = *outoffp;
|
|
|
|
uvalout += len;
|
|
|
|
if (invp->v_type == VDIR || outvp->v_type == VDIR)
|
|
|
|
error = EISDIR;
|
|
|
|
else if (*inoffp < 0 || uvalin > INT64_MAX || uvalin <
|
|
|
|
(uint64_t)*inoffp || *outoffp < 0 || uvalout > INT64_MAX ||
|
|
|
|
uvalout < (uint64_t)*outoffp || invp->v_type != VREG ||
|
|
|
|
outvp->v_type != VREG)
|
|
|
|
error = EINVAL;
|
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the two vnode are for the same file system, call
|
|
|
|
* VOP_COPY_FILE_RANGE(), otherwise call vn_generic_copy_file_range()
|
|
|
|
* which can handle copies across multiple file systems.
|
|
|
|
*/
|
|
|
|
*lenp = len;
|
|
|
|
if (invp->v_mount == outvp->v_mount)
|
|
|
|
error = VOP_COPY_FILE_RANGE(invp, inoffp, outvp, outoffp,
|
|
|
|
lenp, flags, incred, outcred, fsize_td);
|
|
|
|
else
|
|
|
|
error = vn_generic_copy_file_range(invp, inoffp, outvp,
|
|
|
|
outoffp, lenp, flags, incred, outcred, fsize_td);
|
|
|
|
out:
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Test len bytes of data starting at dat for all bytes == 0.
|
|
|
|
* Return true if all bytes are zero, false otherwise.
|
|
|
|
* Expects dat to be well aligned.
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
mem_iszero(void *dat, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
const u_int *p;
|
|
|
|
const char *cp;
|
|
|
|
|
|
|
|
for (p = dat; len > 0; len -= sizeof(*p), p++) {
|
|
|
|
if (len >= sizeof(*p)) {
|
|
|
|
if (*p != 0)
|
|
|
|
return (false);
|
|
|
|
} else {
|
|
|
|
cp = (const char *)p;
|
|
|
|
for (i = 0; i < len; i++, cp++)
|
|
|
|
if (*cp != '\0')
|
|
|
|
return (false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (true);
|
|
|
|
}
|
|
|
|
|
2019-08-15 23:21:41 +00:00
|
|
|
/*
|
|
|
|
* Look for a hole in the output file and, if found, adjust *outoffp
|
|
|
|
* and *xferp to skip past the hole.
|
|
|
|
* *xferp is the entire hole length to be written and xfer2 is how many bytes
|
|
|
|
* to be written as 0's upon return.
|
|
|
|
*/
|
|
|
|
static off_t
|
|
|
|
vn_skip_hole(struct vnode *outvp, off_t xfer2, off_t *outoffp, off_t *xferp,
|
|
|
|
off_t *dataoffp, off_t *holeoffp, struct ucred *cred)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
off_t delta;
|
|
|
|
|
|
|
|
if (*holeoffp == 0 || *holeoffp <= *outoffp) {
|
|
|
|
*dataoffp = *outoffp;
|
|
|
|
error = VOP_IOCTL(outvp, FIOSEEKDATA, dataoffp, 0, cred,
|
|
|
|
curthread);
|
|
|
|
if (error == 0) {
|
|
|
|
*holeoffp = *dataoffp;
|
|
|
|
error = VOP_IOCTL(outvp, FIOSEEKHOLE, holeoffp, 0, cred,
|
|
|
|
curthread);
|
|
|
|
}
|
|
|
|
if (error != 0 || *holeoffp == *dataoffp) {
|
|
|
|
/*
|
|
|
|
* Since outvp is unlocked, it may be possible for
|
|
|
|
* another thread to do a truncate(), lseek(), write()
|
|
|
|
* creating a hole at startoff between the above
|
|
|
|
* VOP_IOCTL() calls, if the other thread does not do
|
|
|
|
* rangelocking.
|
|
|
|
* If that happens, *holeoffp == *dataoffp and finding
|
|
|
|
* the hole has failed, so disable vn_skip_hole().
|
|
|
|
*/
|
|
|
|
*holeoffp = -1; /* Disable use of vn_skip_hole(). */
|
|
|
|
return (xfer2);
|
|
|
|
}
|
|
|
|
KASSERT(*dataoffp >= *outoffp,
|
|
|
|
("vn_skip_hole: dataoff=%jd < outoff=%jd",
|
|
|
|
(intmax_t)*dataoffp, (intmax_t)*outoffp));
|
|
|
|
KASSERT(*holeoffp > *dataoffp,
|
|
|
|
("vn_skip_hole: holeoff=%jd <= dataoff=%jd",
|
|
|
|
(intmax_t)*holeoffp, (intmax_t)*dataoffp));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If there is a hole before the data starts, advance *outoffp and
|
|
|
|
* *xferp past the hole.
|
|
|
|
*/
|
|
|
|
if (*dataoffp > *outoffp) {
|
|
|
|
delta = *dataoffp - *outoffp;
|
|
|
|
if (delta >= *xferp) {
|
|
|
|
/* Entire *xferp is a hole. */
|
|
|
|
*outoffp += *xferp;
|
|
|
|
*xferp = 0;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
*xferp -= delta;
|
|
|
|
*outoffp += delta;
|
|
|
|
xfer2 = MIN(xfer2, *xferp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If a hole starts before the end of this xfer2, reduce this xfer2 so
|
|
|
|
* that the write ends at the start of the hole.
|
|
|
|
* *holeoffp should always be greater than *outoffp, but for the
|
|
|
|
* non-INVARIANTS case, check this to make sure xfer2 remains a sane
|
|
|
|
* value.
|
|
|
|
*/
|
|
|
|
if (*holeoffp > *outoffp && *holeoffp < *outoffp + xfer2)
|
|
|
|
xfer2 = *holeoffp - *outoffp;
|
|
|
|
return (xfer2);
|
|
|
|
}
|
|
|
|
|
2019-07-25 05:46:16 +00:00
|
|
|
/*
|
|
|
|
* Write an xfer sized chunk to outvp in blksize blocks from dat.
|
|
|
|
* dat is a maximum of blksize in length and can be written repeatedly in
|
|
|
|
* the chunk.
|
|
|
|
* If growfile == true, just grow the file via vn_truncate_locked() instead
|
|
|
|
* of doing actual writes.
|
2019-08-15 23:21:41 +00:00
|
|
|
* If checkhole == true, a hole is being punched, so skip over any hole
|
|
|
|
* already in the output file.
|
2019-07-25 05:46:16 +00:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
vn_write_outvp(struct vnode *outvp, char *dat, off_t outoff, off_t xfer,
|
2019-08-15 23:21:41 +00:00
|
|
|
u_long blksize, bool growfile, bool checkhole, struct ucred *cred)
|
2019-07-25 05:46:16 +00:00
|
|
|
{
|
|
|
|
struct mount *mp;
|
2019-08-15 23:21:41 +00:00
|
|
|
off_t dataoff, holeoff, xfer2;
|
2019-07-25 05:46:16 +00:00
|
|
|
int error, lckf;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Loop around doing writes of blksize until write has been completed.
|
|
|
|
* Lock/unlock on each loop iteration so that a bwillwrite() can be
|
|
|
|
* done for each iteration, since the xfer argument can be very
|
|
|
|
* large if there is a large hole to punch in the output file.
|
|
|
|
*/
|
2019-08-15 23:21:41 +00:00
|
|
|
error = 0;
|
|
|
|
holeoff = 0;
|
2019-07-25 05:46:16 +00:00
|
|
|
do {
|
2019-08-15 23:21:41 +00:00
|
|
|
xfer2 = MIN(xfer, blksize);
|
|
|
|
if (checkhole) {
|
|
|
|
/*
|
|
|
|
* Punching a hole. Skip writing if there is
|
|
|
|
* already a hole in the output file.
|
|
|
|
*/
|
|
|
|
xfer2 = vn_skip_hole(outvp, xfer2, &outoff, &xfer,
|
|
|
|
&dataoff, &holeoff, cred);
|
|
|
|
if (xfer == 0)
|
|
|
|
break;
|
|
|
|
if (holeoff < 0)
|
|
|
|
checkhole = false;
|
|
|
|
KASSERT(xfer2 > 0, ("vn_write_outvp: xfer2=%jd",
|
|
|
|
(intmax_t)xfer2));
|
|
|
|
}
|
2019-07-25 05:46:16 +00:00
|
|
|
bwillwrite();
|
|
|
|
mp = NULL;
|
|
|
|
error = vn_start_write(outvp, &mp, V_WAIT);
|
|
|
|
if (error == 0) {
|
|
|
|
if (MNT_SHARED_WRITES(mp))
|
|
|
|
lckf = LK_SHARED;
|
|
|
|
else
|
|
|
|
lckf = LK_EXCLUSIVE;
|
|
|
|
error = vn_lock(outvp, lckf);
|
|
|
|
}
|
|
|
|
if (error == 0) {
|
|
|
|
if (growfile)
|
|
|
|
error = vn_truncate_locked(outvp, outoff + xfer,
|
|
|
|
false, cred);
|
|
|
|
else {
|
|
|
|
error = vn_rdwr(UIO_WRITE, outvp, dat, xfer2,
|
|
|
|
outoff, UIO_SYSSPACE, IO_NODELOCKED,
|
|
|
|
curthread->td_ucred, cred, NULL, curthread);
|
|
|
|
outoff += xfer2;
|
|
|
|
xfer -= xfer2;
|
|
|
|
}
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(outvp);
|
2019-07-25 05:46:16 +00:00
|
|
|
}
|
|
|
|
if (mp != NULL)
|
|
|
|
vn_finished_write(mp);
|
|
|
|
} while (!growfile && xfer > 0 && error == 0);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copy a byte range of one file to another. This function can handle the
|
|
|
|
* case where invp and outvp are on different file systems.
|
|
|
|
* It can also be called by a VOP_COPY_FILE_RANGE() to do the work, if there
|
|
|
|
* is no better file system specific way to do it.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
vn_generic_copy_file_range(struct vnode *invp, off_t *inoffp,
|
|
|
|
struct vnode *outvp, off_t *outoffp, size_t *lenp, unsigned int flags,
|
|
|
|
struct ucred *incred, struct ucred *outcred, struct thread *fsize_td)
|
|
|
|
{
|
|
|
|
struct vattr va;
|
|
|
|
struct mount *mp;
|
|
|
|
struct uio io;
|
|
|
|
off_t startoff, endoff, xfer, xfer2;
|
|
|
|
u_long blksize;
|
|
|
|
int error;
|
2019-11-08 23:39:17 +00:00
|
|
|
bool cantseek, readzeros, eof, lastblock;
|
2019-07-25 05:46:16 +00:00
|
|
|
ssize_t aresid;
|
|
|
|
size_t copylen, len, savlen;
|
|
|
|
char *dat;
|
|
|
|
long holein, holeout;
|
|
|
|
|
|
|
|
holein = holeout = 0;
|
|
|
|
savlen = len = *lenp;
|
|
|
|
error = 0;
|
|
|
|
dat = NULL;
|
|
|
|
|
|
|
|
error = vn_lock(invp, LK_SHARED);
|
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
if (VOP_PATHCONF(invp, _PC_MIN_HOLE_SIZE, &holein) != 0)
|
|
|
|
holein = 0;
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(invp);
|
2019-07-25 05:46:16 +00:00
|
|
|
|
|
|
|
mp = NULL;
|
|
|
|
error = vn_start_write(outvp, &mp, V_WAIT);
|
|
|
|
if (error == 0)
|
|
|
|
error = vn_lock(outvp, LK_EXCLUSIVE);
|
|
|
|
if (error == 0) {
|
|
|
|
/*
|
|
|
|
* If fsize_td != NULL, do a vn_rlimit_fsize() call,
|
|
|
|
* now that outvp is locked.
|
|
|
|
*/
|
|
|
|
if (fsize_td != NULL) {
|
|
|
|
io.uio_offset = *outoffp;
|
|
|
|
io.uio_resid = len;
|
|
|
|
error = vn_rlimit_fsize(outvp, &io, fsize_td);
|
|
|
|
if (error != 0)
|
|
|
|
error = EFBIG;
|
|
|
|
}
|
|
|
|
if (VOP_PATHCONF(outvp, _PC_MIN_HOLE_SIZE, &holeout) != 0)
|
|
|
|
holeout = 0;
|
|
|
|
/*
|
|
|
|
* Holes that are past EOF do not need to be written as a block
|
|
|
|
* of zero bytes. So, truncate the output file as far as
|
|
|
|
* possible and then use va.va_size to decide if writing 0
|
|
|
|
* bytes is necessary in the loop below.
|
|
|
|
*/
|
|
|
|
if (error == 0)
|
|
|
|
error = VOP_GETATTR(outvp, &va, outcred);
|
|
|
|
if (error == 0 && va.va_size > *outoffp && va.va_size <=
|
|
|
|
*outoffp + len) {
|
|
|
|
#ifdef MAC
|
|
|
|
error = mac_vnode_check_write(curthread->td_ucred,
|
|
|
|
outcred, outvp);
|
|
|
|
if (error == 0)
|
|
|
|
#endif
|
|
|
|
error = vn_truncate_locked(outvp, *outoffp,
|
|
|
|
false, outcred);
|
|
|
|
if (error == 0)
|
|
|
|
va.va_size = *outoffp;
|
|
|
|
}
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(outvp);
|
2019-07-25 05:46:16 +00:00
|
|
|
}
|
|
|
|
if (mp != NULL)
|
|
|
|
vn_finished_write(mp);
|
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the blksize to the larger of the hole sizes for invp and outvp.
|
|
|
|
* If hole sizes aren't available, set the blksize to the larger
|
|
|
|
* f_iosize of invp and outvp.
|
|
|
|
* This code expects the hole sizes and f_iosizes to be powers of 2.
|
|
|
|
* This value is clipped at 4Kbytes and 1Mbyte.
|
|
|
|
*/
|
|
|
|
blksize = MAX(holein, holeout);
|
|
|
|
if (blksize == 0)
|
|
|
|
blksize = MAX(invp->v_mount->mnt_stat.f_iosize,
|
|
|
|
outvp->v_mount->mnt_stat.f_iosize);
|
|
|
|
if (blksize < 4096)
|
|
|
|
blksize = 4096;
|
|
|
|
else if (blksize > 1024 * 1024)
|
|
|
|
blksize = 1024 * 1024;
|
|
|
|
dat = malloc(blksize, M_TEMP, M_WAITOK);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If VOP_IOCTL(FIOSEEKHOLE) works for invp, use it and FIOSEEKDATA
|
|
|
|
* to find holes. Otherwise, just scan the read block for all 0s
|
|
|
|
* in the inner loop where the data copying is done.
|
|
|
|
* Note that some file systems such as NFSv3, NFSv4.0 and NFSv4.1 may
|
|
|
|
* support holes on the server, but do not support FIOSEEKHOLE.
|
|
|
|
*/
|
2019-11-08 23:39:17 +00:00
|
|
|
eof = false;
|
|
|
|
while (len > 0 && error == 0 && !eof) {
|
2019-07-25 05:46:16 +00:00
|
|
|
endoff = 0; /* To shut up compilers. */
|
|
|
|
cantseek = true;
|
|
|
|
startoff = *inoffp;
|
|
|
|
copylen = len;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find the next data area. If there is just a hole to EOF,
|
|
|
|
* FIOSEEKDATA should fail and then we drop down into the
|
|
|
|
* inner loop and create the hole on the outvp file.
|
|
|
|
* (I do not know if any file system will report a hole to
|
|
|
|
* EOF via FIOSEEKHOLE, but I am pretty sure FIOSEEKDATA
|
|
|
|
* will fail for those file systems.)
|
|
|
|
*
|
|
|
|
* For input files that don't support FIOSEEKDATA/FIOSEEKHOLE,
|
|
|
|
* the code just falls through to the inner copy loop.
|
|
|
|
*/
|
|
|
|
error = EINVAL;
|
|
|
|
if (holein > 0)
|
|
|
|
error = VOP_IOCTL(invp, FIOSEEKDATA, &startoff, 0,
|
|
|
|
incred, curthread);
|
|
|
|
if (error == 0) {
|
|
|
|
endoff = startoff;
|
|
|
|
error = VOP_IOCTL(invp, FIOSEEKHOLE, &endoff, 0,
|
|
|
|
incred, curthread);
|
2019-08-08 19:53:07 +00:00
|
|
|
/*
|
|
|
|
* Since invp is unlocked, it may be possible for
|
|
|
|
* another thread to do a truncate(), lseek(), write()
|
|
|
|
* creating a hole at startoff between the above
|
|
|
|
* VOP_IOCTL() calls, if the other thread does not do
|
|
|
|
* rangelocking.
|
|
|
|
* If that happens, startoff == endoff and finding
|
|
|
|
* the hole has failed, so set an error.
|
|
|
|
*/
|
|
|
|
if (error == 0 && startoff == endoff)
|
|
|
|
error = EINVAL; /* Any error. Reset to 0. */
|
2019-07-25 05:46:16 +00:00
|
|
|
}
|
|
|
|
if (error == 0) {
|
|
|
|
if (startoff > *inoffp) {
|
|
|
|
/* Found hole before data block. */
|
|
|
|
xfer = MIN(startoff - *inoffp, len);
|
|
|
|
if (*outoffp < va.va_size) {
|
|
|
|
/* Must write 0s to punch hole. */
|
|
|
|
xfer2 = MIN(va.va_size - *outoffp,
|
|
|
|
xfer);
|
|
|
|
memset(dat, 0, MIN(xfer2, blksize));
|
|
|
|
error = vn_write_outvp(outvp, dat,
|
|
|
|
*outoffp, xfer2, blksize, false,
|
2019-08-15 23:21:41 +00:00
|
|
|
holeout > 0, outcred);
|
2019-07-25 05:46:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (error == 0 && *outoffp + xfer >
|
|
|
|
va.va_size && xfer == len)
|
|
|
|
/* Grow last block. */
|
|
|
|
error = vn_write_outvp(outvp, dat,
|
|
|
|
*outoffp, xfer, blksize, true,
|
2019-08-15 23:21:41 +00:00
|
|
|
false, outcred);
|
2019-07-25 05:46:16 +00:00
|
|
|
if (error == 0) {
|
|
|
|
*inoffp += xfer;
|
|
|
|
*outoffp += xfer;
|
|
|
|
len -= xfer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
copylen = MIN(len, endoff - startoff);
|
|
|
|
cantseek = false;
|
|
|
|
} else {
|
|
|
|
cantseek = true;
|
|
|
|
startoff = *inoffp;
|
|
|
|
copylen = len;
|
|
|
|
error = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
xfer = blksize;
|
|
|
|
if (cantseek) {
|
|
|
|
/*
|
|
|
|
* Set first xfer to end at a block boundary, so that
|
|
|
|
* holes are more likely detected in the loop below via
|
|
|
|
* the for all bytes 0 method.
|
|
|
|
*/
|
|
|
|
xfer -= (*inoffp % blksize);
|
|
|
|
}
|
|
|
|
/* Loop copying the data block. */
|
2019-11-08 23:39:17 +00:00
|
|
|
while (copylen > 0 && error == 0 && !eof) {
|
2019-07-25 05:46:16 +00:00
|
|
|
if (copylen < xfer)
|
|
|
|
xfer = copylen;
|
|
|
|
error = vn_lock(invp, LK_SHARED);
|
|
|
|
if (error != 0)
|
|
|
|
goto out;
|
|
|
|
error = vn_rdwr(UIO_READ, invp, dat, xfer,
|
|
|
|
startoff, UIO_SYSSPACE, IO_NODELOCKED,
|
|
|
|
curthread->td_ucred, incred, &aresid,
|
|
|
|
curthread);
|
2020-01-03 22:29:58 +00:00
|
|
|
VOP_UNLOCK(invp);
|
2019-11-08 23:39:17 +00:00
|
|
|
lastblock = false;
|
|
|
|
if (error == 0 && aresid > 0) {
|
|
|
|
/* Stop the copy at EOF on the input file. */
|
|
|
|
xfer -= aresid;
|
|
|
|
eof = true;
|
|
|
|
lastblock = true;
|
|
|
|
}
|
2019-07-25 05:46:16 +00:00
|
|
|
if (error == 0) {
|
|
|
|
/*
|
|
|
|
* Skip the write for holes past the initial EOF
|
|
|
|
* of the output file, unless this is the last
|
|
|
|
* write of the output file at EOF.
|
|
|
|
*/
|
|
|
|
readzeros = cantseek ? mem_iszero(dat, xfer) :
|
|
|
|
false;
|
2019-11-08 23:39:17 +00:00
|
|
|
if (xfer == len)
|
|
|
|
lastblock = true;
|
2019-07-25 05:46:16 +00:00
|
|
|
if (!cantseek || *outoffp < va.va_size ||
|
2019-11-08 23:39:17 +00:00
|
|
|
lastblock || !readzeros)
|
2019-07-25 05:46:16 +00:00
|
|
|
error = vn_write_outvp(outvp, dat,
|
|
|
|
*outoffp, xfer, blksize,
|
2019-11-08 23:39:17 +00:00
|
|
|
readzeros && lastblock &&
|
2019-08-15 23:21:41 +00:00
|
|
|
*outoffp >= va.va_size, false,
|
|
|
|
outcred);
|
2019-07-25 05:46:16 +00:00
|
|
|
if (error == 0) {
|
|
|
|
*inoffp += xfer;
|
|
|
|
startoff += xfer;
|
|
|
|
*outoffp += xfer;
|
|
|
|
copylen -= xfer;
|
|
|
|
len -= xfer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xfer = blksize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
*lenp = savlen - len;
|
|
|
|
free(dat, M_TEMP);
|
|
|
|
return (error);
|
|
|
|
}
|
2020-01-08 19:05:32 +00:00
|
|
|
|
|
|
|
static int
|
|
|
|
vn_fallocate(struct file *fp, off_t offset, off_t len, struct thread *td)
|
|
|
|
{
|
|
|
|
struct mount *mp;
|
|
|
|
struct vnode *vp;
|
|
|
|
off_t olen, ooffset;
|
|
|
|
int error;
|
|
|
|
#ifdef AUDIT
|
|
|
|
int audited_vnode1 = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
if (vp->v_type != VREG)
|
|
|
|
return (ENODEV);
|
|
|
|
|
|
|
|
/* Allocating blocks may take a long time, so iterate. */
|
|
|
|
for (;;) {
|
|
|
|
olen = len;
|
|
|
|
ooffset = offset;
|
|
|
|
|
|
|
|
bwillwrite();
|
|
|
|
mp = NULL;
|
|
|
|
error = vn_start_write(vp, &mp, V_WAIT | PCATCH);
|
|
|
|
if (error != 0)
|
|
|
|
break;
|
|
|
|
error = vn_lock(vp, LK_EXCLUSIVE);
|
|
|
|
if (error != 0) {
|
|
|
|
vn_finished_write(mp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#ifdef AUDIT
|
|
|
|
if (!audited_vnode1) {
|
|
|
|
AUDIT_ARG_VNODE1(vp);
|
|
|
|
audited_vnode1 = 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef MAC
|
|
|
|
error = mac_vnode_check_write(td->td_ucred, fp->f_cred, vp);
|
|
|
|
if (error == 0)
|
|
|
|
#endif
|
|
|
|
error = VOP_ALLOCATE(vp, &offset, &len);
|
|
|
|
VOP_UNLOCK(vp);
|
|
|
|
vn_finished_write(mp);
|
|
|
|
|
|
|
|
if (olen + ooffset != offset + len) {
|
|
|
|
panic("offset + len changed from %jx/%jx to %jx/%jx",
|
|
|
|
ooffset, olen, offset, len);
|
|
|
|
}
|
|
|
|
if (error != 0 || len == 0)
|
|
|
|
break;
|
|
|
|
KASSERT(olen > len, ("Iteration did not make progress?"));
|
|
|
|
maybe_yield();
|
|
|
|
}
|
|
|
|
|
|
|
|
return (error);
|
|
|
|
}
|