mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	gfs2: Fix mmap + page fault deadlocks
Functions gfs2_file_read_iter and gfs2_file_write_iter are both
 accessing the user buffer to write to or read from while holding the
 inode glock.  In the most basic scenario, that buffer will not be
 resident and it will be mapped to the same file.  Accessing the buffer
 will trigger a page fault, and gfs2 will deadlock trying to take the
 same inode glock again while trying to handle that fault.
 
 Fix that and similar, more complex scenarios by disabling page faults
 while accessing user buffers.  To make this work, introduce a small
 amount of new infrastructure and fix some bugs that didn't trigger so
 far, with page faults enabled.
 -----BEGIN PGP SIGNATURE-----
 
 iQJIBAABCAAyFiEEJZs3krPW0xkhLMTc1b+f6wMTZToFAmGBPisUHGFncnVlbmJh
 QHJlZGhhdC5jb20ACgkQ1b+f6wMTZTpE6A/7BezUnGuNJxJrR8pC+vcLYA7xAgUU
 6STQ6IN7w5UHRlSkNzZxZ2XPxW4uVQ4SxSEeaLqBsHZihepjcLNFZ/8MhQ6UPSD0
 8noHOi7CoIcp6IuWQtCpxRM/xjjm2SlMt2XbVJZaiJcdzCV9gB6TU9EkBRq7Zm/X
 9WFBbv1xZF0skn9ISCJvNtiiI+VyWKgMDUKxJUiTQjmJcklyyqHcVGmQi9BjqPz4
 4s3F+WH6CoGbDKlmNk/6Y9wZ/2+sbvGswVscUxPwJVPoZWsR1xBBUdAeAmEMD1P4
 BgE/Y1J8JXyVPYtyvZKq70XUhKdQkxB7RfX87YasOk9mY4Kjd5rIIGEykh+o2vC9
 kDhCHvf2Mnw5I6Rum3B7UXyB1vemY+fECIHsXhgBnS+ztabRtcAdpCuWoqb43ymw
 yEX1KwXyU4FpRYbrRvdZT42Fmh6ty8TW+N4swg8S2TrffirvgAi5yrcHZ4mPupYv
 lyzvsCW7Wv8hPXn/twNObX+okRgJnsxcCdBXARdCnRXfA8tH23xmu88u8RA1Vdxh
 nzTvv6Dx2EowwojuDWMx29Mw3fA2IqIfbOV+4FaRU7NZ2ZKtknL8yGl27qQUsMoJ
 vYsHTmagasjQr+NDJ3vQRLCw+JQ6B1hENpdkmixFD9moo7X1ZFW3HBi/UL973Bv6
 5CmgeXto8FRUFjI=
 =WeNd
 -----END PGP SIGNATURE-----
Merge tag 'gfs2-v5.15-rc5-mmap-fault' of git://git.kernel.org/pub/scm/linux/kernel/git/gfs2/linux-gfs2
Pull gfs2 mmap + page fault deadlocks fixes from Andreas Gruenbacher:
 "Functions gfs2_file_read_iter and gfs2_file_write_iter are both
  accessing the user buffer to write to or read from while holding the
  inode glock.
  In the most basic deadlock scenario, that buffer will not be resident
  and it will be mapped to the same file. Accessing the buffer will
  trigger a page fault, and gfs2 will deadlock trying to take the same
  inode glock again while trying to handle that fault.
  Fix that and similar, more complex scenarios by disabling page faults
  while accessing user buffers. To make this work, introduce a small
  amount of new infrastructure and fix some bugs that didn't trigger so
  far, with page faults enabled"
* tag 'gfs2-v5.15-rc5-mmap-fault' of git://git.kernel.org/pub/scm/linux/kernel/git/gfs2/linux-gfs2:
  gfs2: Fix mmap + page fault deadlocks for direct I/O
  iov_iter: Introduce nofault flag to disable page faults
  gup: Introduce FOLL_NOFAULT flag to disable page faults
  iomap: Add done_before argument to iomap_dio_rw
  iomap: Support partial direct I/O on user copy failures
  iomap: Fix iomap_dio_rw return value for user copies
  gfs2: Fix mmap + page fault deadlocks for buffered I/O
  gfs2: Eliminate ip->i_gh
  gfs2: Move the inode glock locking to gfs2_file_buffered_write
  gfs2: Introduce flag for glock holder auto-demotion
  gfs2: Clean up function may_grant
  gfs2: Add wrapper for iomap_file_buffered_write
  iov_iter: Introduce fault_in_iov_iter_writeable
  iov_iter: Turn iov_iter_fault_in_readable into fault_in_iov_iter_readable
  gup: Turn fault_in_pages_{readable,writeable} into fault_in_{readable,writeable}
  powerpc/kvm: Fix kvm_use_magic_page
  iov_iter: Fix iov_iter_get_pages{,_alloc} page fault return value
			
			
This commit is contained in:
		
						commit
						c03098d4b9
					
				| @ -669,7 +669,8 @@ static void __init kvm_use_magic_page(void) | |||||||
| 	on_each_cpu(kvm_map_magic_page, &features, 1); | 	on_each_cpu(kvm_map_magic_page, &features, 1); | ||||||
| 
 | 
 | ||||||
| 	/* Quick self-test to see if the mapping works */ | 	/* Quick self-test to see if the mapping works */ | ||||||
| 	if (!fault_in_pages_readable((const char *)KVM_MAGIC_PAGE, sizeof(u32))) { | 	if (fault_in_readable((const char __user *)KVM_MAGIC_PAGE, | ||||||
|  | 			      sizeof(u32))) { | ||||||
| 		kvm_patching_worked = false; | 		kvm_patching_worked = false; | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -1048,7 +1048,7 @@ SYSCALL_DEFINE3(swapcontext, struct ucontext __user *, old_ctx, | |||||||
| 	if (new_ctx == NULL) | 	if (new_ctx == NULL) | ||||||
| 		return 0; | 		return 0; | ||||||
| 	if (!access_ok(new_ctx, ctx_size) || | 	if (!access_ok(new_ctx, ctx_size) || | ||||||
| 	    fault_in_pages_readable((u8 __user *)new_ctx, ctx_size)) | 	    fault_in_readable((char __user *)new_ctx, ctx_size)) | ||||||
| 		return -EFAULT; | 		return -EFAULT; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| @ -1237,7 +1237,7 @@ SYSCALL_DEFINE3(debug_setcontext, struct ucontext __user *, ctx, | |||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| 	if (!access_ok(ctx, sizeof(*ctx)) || | 	if (!access_ok(ctx, sizeof(*ctx)) || | ||||||
| 	    fault_in_pages_readable((u8 __user *)ctx, sizeof(*ctx))) | 	    fault_in_readable((char __user *)ctx, sizeof(*ctx))) | ||||||
| 		return -EFAULT; | 		return -EFAULT; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
|  | |||||||
| @ -688,7 +688,7 @@ SYSCALL_DEFINE3(swapcontext, struct ucontext __user *, old_ctx, | |||||||
| 	if (new_ctx == NULL) | 	if (new_ctx == NULL) | ||||||
| 		return 0; | 		return 0; | ||||||
| 	if (!access_ok(new_ctx, ctx_size) || | 	if (!access_ok(new_ctx, ctx_size) || | ||||||
| 	    fault_in_pages_readable((u8 __user *)new_ctx, ctx_size)) | 	    fault_in_readable((char __user *)new_ctx, ctx_size)) | ||||||
| 		return -EFAULT; | 		return -EFAULT; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
|  | |||||||
| @ -309,7 +309,7 @@ retry: | |||||||
| 		if (ret != X86_TRAP_PF) | 		if (ret != X86_TRAP_PF) | ||||||
| 			return false; | 			return false; | ||||||
| 
 | 
 | ||||||
| 		if (!fault_in_pages_readable(buf, size)) | 		if (!fault_in_readable(buf, size)) | ||||||
| 			goto retry; | 			goto retry; | ||||||
| 		return false; | 		return false; | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -336,7 +336,7 @@ int armada_gem_pwrite_ioctl(struct drm_device *dev, void *data, | |||||||
| 	struct drm_armada_gem_pwrite *args = data; | 	struct drm_armada_gem_pwrite *args = data; | ||||||
| 	struct armada_gem_object *dobj; | 	struct armada_gem_object *dobj; | ||||||
| 	char __user *ptr; | 	char __user *ptr; | ||||||
| 	int ret; | 	int ret = 0; | ||||||
| 
 | 
 | ||||||
| 	DRM_DEBUG_DRIVER("handle %u off %u size %u ptr 0x%llx\n", | 	DRM_DEBUG_DRIVER("handle %u off %u size %u ptr 0x%llx\n", | ||||||
| 		args->handle, args->offset, args->size, args->ptr); | 		args->handle, args->offset, args->size, args->ptr); | ||||||
| @ -349,9 +349,8 @@ int armada_gem_pwrite_ioctl(struct drm_device *dev, void *data, | |||||||
| 	if (!access_ok(ptr, args->size)) | 	if (!access_ok(ptr, args->size)) | ||||||
| 		return -EFAULT; | 		return -EFAULT; | ||||||
| 
 | 
 | ||||||
| 	ret = fault_in_pages_readable(ptr, args->size); | 	if (fault_in_readable(ptr, args->size)) | ||||||
| 	if (ret) | 		return -EFAULT; | ||||||
| 		return ret; |  | ||||||
| 
 | 
 | ||||||
| 	dobj = armada_gem_object_lookup(file, args->handle); | 	dobj = armada_gem_object_lookup(file, args->handle); | ||||||
| 	if (dobj == NULL) | 	if (dobj == NULL) | ||||||
|  | |||||||
| @ -1718,7 +1718,7 @@ static noinline ssize_t btrfs_buffered_write(struct kiocb *iocb, | |||||||
| 		 * Fault pages before locking them in prepare_pages | 		 * Fault pages before locking them in prepare_pages | ||||||
| 		 * to avoid recursive lock | 		 * to avoid recursive lock | ||||||
| 		 */ | 		 */ | ||||||
| 		if (unlikely(iov_iter_fault_in_readable(i, write_bytes))) { | 		if (unlikely(fault_in_iov_iter_readable(i, write_bytes))) { | ||||||
| 			ret = -EFAULT; | 			ret = -EFAULT; | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
| @ -1965,7 +1965,7 @@ relock: | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	dio = __iomap_dio_rw(iocb, from, &btrfs_dio_iomap_ops, &btrfs_dio_ops, | 	dio = __iomap_dio_rw(iocb, from, &btrfs_dio_iomap_ops, &btrfs_dio_ops, | ||||||
| 			     0); | 			     0, 0); | ||||||
| 
 | 
 | ||||||
| 	btrfs_inode_unlock(inode, ilock_flags); | 	btrfs_inode_unlock(inode, ilock_flags); | ||||||
| 
 | 
 | ||||||
| @ -3668,7 +3668,8 @@ static ssize_t btrfs_direct_read(struct kiocb *iocb, struct iov_iter *to) | |||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	btrfs_inode_lock(inode, BTRFS_ILOCK_SHARED); | 	btrfs_inode_lock(inode, BTRFS_ILOCK_SHARED); | ||||||
| 	ret = iomap_dio_rw(iocb, to, &btrfs_dio_iomap_ops, &btrfs_dio_ops, 0); | 	ret = iomap_dio_rw(iocb, to, &btrfs_dio_iomap_ops, &btrfs_dio_ops, | ||||||
|  | 			   0, 0); | ||||||
| 	btrfs_inode_unlock(inode, BTRFS_ILOCK_SHARED); | 	btrfs_inode_unlock(inode, BTRFS_ILOCK_SHARED); | ||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
|  | |||||||
| @ -2222,9 +2222,8 @@ static noinline int search_ioctl(struct inode *inode, | |||||||
| 	key.offset = sk->min_offset; | 	key.offset = sk->min_offset; | ||||||
| 
 | 
 | ||||||
| 	while (1) { | 	while (1) { | ||||||
| 		ret = fault_in_pages_writeable(ubuf + sk_offset, | 		ret = -EFAULT; | ||||||
| 					       *buf_size - sk_offset); | 		if (fault_in_writeable(ubuf + sk_offset, *buf_size - sk_offset)) | ||||||
| 		if (ret) |  | ||||||
| 			break; | 			break; | ||||||
| 
 | 
 | ||||||
| 		ret = btrfs_search_forward(root, &key, path, sk->min_transid); | 		ret = btrfs_search_forward(root, &key, path, sk->min_transid); | ||||||
|  | |||||||
| @ -334,7 +334,7 @@ static ssize_t erofs_file_read_iter(struct kiocb *iocb, struct iov_iter *to) | |||||||
| 
 | 
 | ||||||
| 		if (!err) | 		if (!err) | ||||||
| 			return iomap_dio_rw(iocb, to, &erofs_iomap_ops, | 			return iomap_dio_rw(iocb, to, &erofs_iomap_ops, | ||||||
| 					    NULL, 0); | 					    NULL, 0, 0); | ||||||
| 		if (err < 0) | 		if (err < 0) | ||||||
| 			return err; | 			return err; | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -74,7 +74,7 @@ static ssize_t ext4_dio_read_iter(struct kiocb *iocb, struct iov_iter *to) | |||||||
| 		return generic_file_read_iter(iocb, to); | 		return generic_file_read_iter(iocb, to); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ret = iomap_dio_rw(iocb, to, &ext4_iomap_ops, NULL, 0); | 	ret = iomap_dio_rw(iocb, to, &ext4_iomap_ops, NULL, 0, 0); | ||||||
| 	inode_unlock_shared(inode); | 	inode_unlock_shared(inode); | ||||||
| 
 | 
 | ||||||
| 	file_accessed(iocb->ki_filp); | 	file_accessed(iocb->ki_filp); | ||||||
| @ -566,7 +566,8 @@ static ssize_t ext4_dio_write_iter(struct kiocb *iocb, struct iov_iter *from) | |||||||
| 	if (ilock_shared) | 	if (ilock_shared) | ||||||
| 		iomap_ops = &ext4_iomap_overwrite_ops; | 		iomap_ops = &ext4_iomap_overwrite_ops; | ||||||
| 	ret = iomap_dio_rw(iocb, from, iomap_ops, &ext4_dio_write_ops, | 	ret = iomap_dio_rw(iocb, from, iomap_ops, &ext4_dio_write_ops, | ||||||
| 			   (unaligned_io || extend) ? IOMAP_DIO_FORCE_WAIT : 0); | 			   (unaligned_io || extend) ? IOMAP_DIO_FORCE_WAIT : 0, | ||||||
|  | 			   0); | ||||||
| 	if (ret == -ENOTBLK) | 	if (ret == -ENOTBLK) | ||||||
| 		ret = 0; | 		ret = 0; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -4276,7 +4276,7 @@ static ssize_t f2fs_file_write_iter(struct kiocb *iocb, struct iov_iter *from) | |||||||
| 		size_t target_size = 0; | 		size_t target_size = 0; | ||||||
| 		int err; | 		int err; | ||||||
| 
 | 
 | ||||||
| 		if (iov_iter_fault_in_readable(from, iov_iter_count(from))) | 		if (fault_in_iov_iter_readable(from, iov_iter_count(from))) | ||||||
| 			set_inode_flag(inode, FI_NO_PREALLOC); | 			set_inode_flag(inode, FI_NO_PREALLOC); | ||||||
| 
 | 
 | ||||||
| 		if ((iocb->ki_flags & IOCB_NOWAIT)) { | 		if ((iocb->ki_flags & IOCB_NOWAIT)) { | ||||||
|  | |||||||
| @ -1164,7 +1164,7 @@ static ssize_t fuse_fill_write_pages(struct fuse_io_args *ia, | |||||||
| 
 | 
 | ||||||
|  again: |  again: | ||||||
| 		err = -EFAULT; | 		err = -EFAULT; | ||||||
| 		if (iov_iter_fault_in_readable(ii, bytes)) | 		if (fault_in_iov_iter_readable(ii, bytes)) | ||||||
| 			break; | 			break; | ||||||
| 
 | 
 | ||||||
| 		err = -ENOMEM; | 		err = -ENOMEM; | ||||||
|  | |||||||
| @ -961,46 +961,6 @@ hole_found: | |||||||
| 	goto out; | 	goto out; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int gfs2_write_lock(struct inode *inode) |  | ||||||
| { |  | ||||||
| 	struct gfs2_inode *ip = GFS2_I(inode); |  | ||||||
| 	struct gfs2_sbd *sdp = GFS2_SB(inode); |  | ||||||
| 	int error; |  | ||||||
| 
 |  | ||||||
| 	gfs2_holder_init(ip->i_gl, LM_ST_EXCLUSIVE, 0, &ip->i_gh); |  | ||||||
| 	error = gfs2_glock_nq(&ip->i_gh); |  | ||||||
| 	if (error) |  | ||||||
| 		goto out_uninit; |  | ||||||
| 	if (&ip->i_inode == sdp->sd_rindex) { |  | ||||||
| 		struct gfs2_inode *m_ip = GFS2_I(sdp->sd_statfs_inode); |  | ||||||
| 
 |  | ||||||
| 		error = gfs2_glock_nq_init(m_ip->i_gl, LM_ST_EXCLUSIVE, |  | ||||||
| 					   GL_NOCACHE, &m_ip->i_gh); |  | ||||||
| 		if (error) |  | ||||||
| 			goto out_unlock; |  | ||||||
| 	} |  | ||||||
| 	return 0; |  | ||||||
| 
 |  | ||||||
| out_unlock: |  | ||||||
| 	gfs2_glock_dq(&ip->i_gh); |  | ||||||
| out_uninit: |  | ||||||
| 	gfs2_holder_uninit(&ip->i_gh); |  | ||||||
| 	return error; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static void gfs2_write_unlock(struct inode *inode) |  | ||||||
| { |  | ||||||
| 	struct gfs2_inode *ip = GFS2_I(inode); |  | ||||||
| 	struct gfs2_sbd *sdp = GFS2_SB(inode); |  | ||||||
| 
 |  | ||||||
| 	if (&ip->i_inode == sdp->sd_rindex) { |  | ||||||
| 		struct gfs2_inode *m_ip = GFS2_I(sdp->sd_statfs_inode); |  | ||||||
| 
 |  | ||||||
| 		gfs2_glock_dq_uninit(&m_ip->i_gh); |  | ||||||
| 	} |  | ||||||
| 	gfs2_glock_dq_uninit(&ip->i_gh); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int gfs2_iomap_page_prepare(struct inode *inode, loff_t pos, | static int gfs2_iomap_page_prepare(struct inode *inode, loff_t pos, | ||||||
| 				   unsigned len) | 				   unsigned len) | ||||||
| { | { | ||||||
| @ -1118,11 +1078,6 @@ out_qunlock: | |||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline bool gfs2_iomap_need_write_lock(unsigned flags) |  | ||||||
| { |  | ||||||
| 	return (flags & IOMAP_WRITE) && !(flags & IOMAP_DIRECT); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int gfs2_iomap_begin(struct inode *inode, loff_t pos, loff_t length, | static int gfs2_iomap_begin(struct inode *inode, loff_t pos, loff_t length, | ||||||
| 			    unsigned flags, struct iomap *iomap, | 			    unsigned flags, struct iomap *iomap, | ||||||
| 			    struct iomap *srcmap) | 			    struct iomap *srcmap) | ||||||
| @ -1135,12 +1090,6 @@ static int gfs2_iomap_begin(struct inode *inode, loff_t pos, loff_t length, | |||||||
| 		iomap->flags |= IOMAP_F_BUFFER_HEAD; | 		iomap->flags |= IOMAP_F_BUFFER_HEAD; | ||||||
| 
 | 
 | ||||||
| 	trace_gfs2_iomap_start(ip, pos, length, flags); | 	trace_gfs2_iomap_start(ip, pos, length, flags); | ||||||
| 	if (gfs2_iomap_need_write_lock(flags)) { |  | ||||||
| 		ret = gfs2_write_lock(inode); |  | ||||||
| 		if (ret) |  | ||||||
| 			goto out; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	ret = __gfs2_iomap_get(inode, pos, length, flags, iomap, &mp); | 	ret = __gfs2_iomap_get(inode, pos, length, flags, iomap, &mp); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		goto out_unlock; | 		goto out_unlock; | ||||||
| @ -1168,10 +1117,7 @@ static int gfs2_iomap_begin(struct inode *inode, loff_t pos, loff_t length, | |||||||
| 	ret = gfs2_iomap_begin_write(inode, pos, length, flags, iomap, &mp); | 	ret = gfs2_iomap_begin_write(inode, pos, length, flags, iomap, &mp); | ||||||
| 
 | 
 | ||||||
| out_unlock: | out_unlock: | ||||||
| 	if (ret && gfs2_iomap_need_write_lock(flags)) |  | ||||||
| 		gfs2_write_unlock(inode); |  | ||||||
| 	release_metapath(&mp); | 	release_metapath(&mp); | ||||||
| out: |  | ||||||
| 	trace_gfs2_iomap_end(ip, iomap, ret); | 	trace_gfs2_iomap_end(ip, iomap, ret); | ||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| @ -1219,15 +1165,11 @@ static int gfs2_iomap_end(struct inode *inode, loff_t pos, loff_t length, | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (unlikely(!written)) | 	if (unlikely(!written)) | ||||||
| 		goto out_unlock; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	if (iomap->flags & IOMAP_F_SIZE_CHANGED) | 	if (iomap->flags & IOMAP_F_SIZE_CHANGED) | ||||||
| 		mark_inode_dirty(inode); | 		mark_inode_dirty(inode); | ||||||
| 	set_bit(GLF_DIRTY, &ip->i_gl->gl_flags); | 	set_bit(GLF_DIRTY, &ip->i_gl->gl_flags); | ||||||
| 
 |  | ||||||
| out_unlock: |  | ||||||
| 	if (gfs2_iomap_need_write_lock(flags)) |  | ||||||
| 		gfs2_write_unlock(inode); |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										242
									
								
								fs/gfs2/file.c
									
									
									
									
									
								
							
							
						
						
									
										242
									
								
								fs/gfs2/file.c
									
									
									
									
									
								
							| @ -776,27 +776,99 @@ static int gfs2_fsync(struct file *file, loff_t start, loff_t end, | |||||||
| 	return ret ? ret : ret1; | 	return ret ? ret : ret1; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static inline bool should_fault_in_pages(ssize_t ret, struct iov_iter *i, | ||||||
|  | 					 size_t *prev_count, | ||||||
|  | 					 size_t *window_size) | ||||||
|  | { | ||||||
|  | 	char __user *p = i->iov[0].iov_base + i->iov_offset; | ||||||
|  | 	size_t count = iov_iter_count(i); | ||||||
|  | 	int pages = 1; | ||||||
|  | 
 | ||||||
|  | 	if (likely(!count)) | ||||||
|  | 		return false; | ||||||
|  | 	if (ret <= 0 && ret != -EFAULT) | ||||||
|  | 		return false; | ||||||
|  | 	if (!iter_is_iovec(i)) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	if (*prev_count != count || !*window_size) { | ||||||
|  | 		int pages, nr_dirtied; | ||||||
|  | 
 | ||||||
|  | 		pages = min_t(int, BIO_MAX_VECS, | ||||||
|  | 			      DIV_ROUND_UP(iov_iter_count(i), PAGE_SIZE)); | ||||||
|  | 		nr_dirtied = max(current->nr_dirtied_pause - | ||||||
|  | 				 current->nr_dirtied, 1); | ||||||
|  | 		pages = min(pages, nr_dirtied); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	*prev_count = count; | ||||||
|  | 	*window_size = (size_t)PAGE_SIZE * pages - offset_in_page(p); | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static ssize_t gfs2_file_direct_read(struct kiocb *iocb, struct iov_iter *to, | static ssize_t gfs2_file_direct_read(struct kiocb *iocb, struct iov_iter *to, | ||||||
| 				     struct gfs2_holder *gh) | 				     struct gfs2_holder *gh) | ||||||
| { | { | ||||||
| 	struct file *file = iocb->ki_filp; | 	struct file *file = iocb->ki_filp; | ||||||
| 	struct gfs2_inode *ip = GFS2_I(file->f_mapping->host); | 	struct gfs2_inode *ip = GFS2_I(file->f_mapping->host); | ||||||
| 	size_t count = iov_iter_count(to); | 	size_t prev_count = 0, window_size = 0; | ||||||
|  | 	size_t written = 0; | ||||||
| 	ssize_t ret; | 	ssize_t ret; | ||||||
| 
 | 
 | ||||||
| 	if (!count) | 	/*
 | ||||||
|  | 	 * In this function, we disable page faults when we're holding the | ||||||
|  | 	 * inode glock while doing I/O.  If a page fault occurs, we indicate | ||||||
|  | 	 * that the inode glock may be dropped, fault in the pages manually, | ||||||
|  | 	 * and retry. | ||||||
|  | 	 * | ||||||
|  | 	 * Unlike generic_file_read_iter, for reads, iomap_dio_rw can trigger | ||||||
|  | 	 * physical as well as manual page faults, and we need to disable both | ||||||
|  | 	 * kinds. | ||||||
|  | 	 * | ||||||
|  | 	 * For direct I/O, gfs2 takes the inode glock in deferred mode.  This | ||||||
|  | 	 * locking mode is compatible with other deferred holders, so multiple | ||||||
|  | 	 * processes and nodes can do direct I/O to a file at the same time. | ||||||
|  | 	 * There's no guarantee that reads or writes will be atomic.  Any | ||||||
|  | 	 * coordination among readers and writers needs to happen externally. | ||||||
|  | 	 */ | ||||||
|  | 
 | ||||||
|  | 	if (!iov_iter_count(to)) | ||||||
| 		return 0; /* skip atime */ | 		return 0; /* skip atime */ | ||||||
| 
 | 
 | ||||||
| 	gfs2_holder_init(ip->i_gl, LM_ST_DEFERRED, 0, gh); | 	gfs2_holder_init(ip->i_gl, LM_ST_DEFERRED, 0, gh); | ||||||
|  | retry: | ||||||
| 	ret = gfs2_glock_nq(gh); | 	ret = gfs2_glock_nq(gh); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		goto out_uninit; | 		goto out_uninit; | ||||||
|  | retry_under_glock: | ||||||
|  | 	pagefault_disable(); | ||||||
|  | 	to->nofault = true; | ||||||
|  | 	ret = iomap_dio_rw(iocb, to, &gfs2_iomap_ops, NULL, | ||||||
|  | 			   IOMAP_DIO_PARTIAL, written); | ||||||
|  | 	to->nofault = false; | ||||||
|  | 	pagefault_enable(); | ||||||
|  | 	if (ret > 0) | ||||||
|  | 		written = ret; | ||||||
| 
 | 
 | ||||||
| 	ret = iomap_dio_rw(iocb, to, &gfs2_iomap_ops, NULL, 0); | 	if (should_fault_in_pages(ret, to, &prev_count, &window_size)) { | ||||||
|  | 		size_t leftover; | ||||||
|  | 
 | ||||||
|  | 		gfs2_holder_allow_demote(gh); | ||||||
|  | 		leftover = fault_in_iov_iter_writeable(to, window_size); | ||||||
|  | 		gfs2_holder_disallow_demote(gh); | ||||||
|  | 		if (leftover != window_size) { | ||||||
|  | 			if (!gfs2_holder_queued(gh)) | ||||||
|  | 				goto retry; | ||||||
|  | 			goto retry_under_glock; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if (gfs2_holder_queued(gh)) | ||||||
| 		gfs2_glock_dq(gh); | 		gfs2_glock_dq(gh); | ||||||
| out_uninit: | out_uninit: | ||||||
| 	gfs2_holder_uninit(gh); | 	gfs2_holder_uninit(gh); | ||||||
|  | 	if (ret < 0) | ||||||
| 		return ret; | 		return ret; | ||||||
|  | 	return written; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static ssize_t gfs2_file_direct_write(struct kiocb *iocb, struct iov_iter *from, | static ssize_t gfs2_file_direct_write(struct kiocb *iocb, struct iov_iter *from, | ||||||
| @ -805,10 +877,20 @@ static ssize_t gfs2_file_direct_write(struct kiocb *iocb, struct iov_iter *from, | |||||||
| 	struct file *file = iocb->ki_filp; | 	struct file *file = iocb->ki_filp; | ||||||
| 	struct inode *inode = file->f_mapping->host; | 	struct inode *inode = file->f_mapping->host; | ||||||
| 	struct gfs2_inode *ip = GFS2_I(inode); | 	struct gfs2_inode *ip = GFS2_I(inode); | ||||||
| 	size_t len = iov_iter_count(from); | 	size_t prev_count = 0, window_size = 0; | ||||||
| 	loff_t offset = iocb->ki_pos; | 	size_t read = 0; | ||||||
| 	ssize_t ret; | 	ssize_t ret; | ||||||
| 
 | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * In this function, we disable page faults when we're holding the | ||||||
|  | 	 * inode glock while doing I/O.  If a page fault occurs, we indicate | ||||||
|  | 	 * that the inode glock may be dropped, fault in the pages manually, | ||||||
|  | 	 * and retry. | ||||||
|  | 	 * | ||||||
|  | 	 * For writes, iomap_dio_rw only triggers manual page faults, so we | ||||||
|  | 	 * don't need to disable physical ones. | ||||||
|  | 	 */ | ||||||
|  | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Deferred lock, even if its a write, since we do no allocation on | 	 * Deferred lock, even if its a write, since we do no allocation on | ||||||
| 	 * this path. All we need to change is the atime, and this lock mode | 	 * this path. All we need to change is the atime, and this lock mode | ||||||
| @ -818,31 +900,62 @@ static ssize_t gfs2_file_direct_write(struct kiocb *iocb, struct iov_iter *from, | |||||||
| 	 * VFS does. | 	 * VFS does. | ||||||
| 	 */ | 	 */ | ||||||
| 	gfs2_holder_init(ip->i_gl, LM_ST_DEFERRED, 0, gh); | 	gfs2_holder_init(ip->i_gl, LM_ST_DEFERRED, 0, gh); | ||||||
|  | retry: | ||||||
| 	ret = gfs2_glock_nq(gh); | 	ret = gfs2_glock_nq(gh); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		goto out_uninit; | 		goto out_uninit; | ||||||
| 
 | retry_under_glock: | ||||||
| 	/* Silently fall back to buffered I/O when writing beyond EOF */ | 	/* Silently fall back to buffered I/O when writing beyond EOF */ | ||||||
| 	if (offset + len > i_size_read(&ip->i_inode)) | 	if (iocb->ki_pos + iov_iter_count(from) > i_size_read(&ip->i_inode)) | ||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	ret = iomap_dio_rw(iocb, from, &gfs2_iomap_ops, NULL, 0); | 	from->nofault = true; | ||||||
|  | 	ret = iomap_dio_rw(iocb, from, &gfs2_iomap_ops, NULL, | ||||||
|  | 			   IOMAP_DIO_PARTIAL, read); | ||||||
|  | 	from->nofault = false; | ||||||
|  | 
 | ||||||
| 	if (ret == -ENOTBLK) | 	if (ret == -ENOTBLK) | ||||||
| 		ret = 0; | 		ret = 0; | ||||||
|  | 	if (ret > 0) | ||||||
|  | 		read = ret; | ||||||
|  | 
 | ||||||
|  | 	if (should_fault_in_pages(ret, from, &prev_count, &window_size)) { | ||||||
|  | 		size_t leftover; | ||||||
|  | 
 | ||||||
|  | 		gfs2_holder_allow_demote(gh); | ||||||
|  | 		leftover = fault_in_iov_iter_readable(from, window_size); | ||||||
|  | 		gfs2_holder_disallow_demote(gh); | ||||||
|  | 		if (leftover != window_size) { | ||||||
|  | 			if (!gfs2_holder_queued(gh)) | ||||||
|  | 				goto retry; | ||||||
|  | 			goto retry_under_glock; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| out: | out: | ||||||
|  | 	if (gfs2_holder_queued(gh)) | ||||||
| 		gfs2_glock_dq(gh); | 		gfs2_glock_dq(gh); | ||||||
| out_uninit: | out_uninit: | ||||||
| 	gfs2_holder_uninit(gh); | 	gfs2_holder_uninit(gh); | ||||||
|  | 	if (ret < 0) | ||||||
| 		return ret; | 		return ret; | ||||||
|  | 	return read; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static ssize_t gfs2_file_read_iter(struct kiocb *iocb, struct iov_iter *to) | static ssize_t gfs2_file_read_iter(struct kiocb *iocb, struct iov_iter *to) | ||||||
| { | { | ||||||
| 	struct gfs2_inode *ip; | 	struct gfs2_inode *ip; | ||||||
| 	struct gfs2_holder gh; | 	struct gfs2_holder gh; | ||||||
|  | 	size_t prev_count = 0, window_size = 0; | ||||||
| 	size_t written = 0; | 	size_t written = 0; | ||||||
| 	ssize_t ret; | 	ssize_t ret; | ||||||
| 
 | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * In this function, we disable page faults when we're holding the | ||||||
|  | 	 * inode glock while doing I/O.  If a page fault occurs, we indicate | ||||||
|  | 	 * that the inode glock may be dropped, fault in the pages manually, | ||||||
|  | 	 * and retry. | ||||||
|  | 	 */ | ||||||
|  | 
 | ||||||
| 	if (iocb->ki_flags & IOCB_DIRECT) { | 	if (iocb->ki_flags & IOCB_DIRECT) { | ||||||
| 		ret = gfs2_file_direct_read(iocb, to, &gh); | 		ret = gfs2_file_direct_read(iocb, to, &gh); | ||||||
| 		if (likely(ret != -ENOTBLK)) | 		if (likely(ret != -ENOTBLK)) | ||||||
| @ -864,18 +977,118 @@ static ssize_t gfs2_file_read_iter(struct kiocb *iocb, struct iov_iter *to) | |||||||
| 	} | 	} | ||||||
| 	ip = GFS2_I(iocb->ki_filp->f_mapping->host); | 	ip = GFS2_I(iocb->ki_filp->f_mapping->host); | ||||||
| 	gfs2_holder_init(ip->i_gl, LM_ST_SHARED, 0, &gh); | 	gfs2_holder_init(ip->i_gl, LM_ST_SHARED, 0, &gh); | ||||||
|  | retry: | ||||||
| 	ret = gfs2_glock_nq(&gh); | 	ret = gfs2_glock_nq(&gh); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		goto out_uninit; | 		goto out_uninit; | ||||||
|  | retry_under_glock: | ||||||
|  | 	pagefault_disable(); | ||||||
| 	ret = generic_file_read_iter(iocb, to); | 	ret = generic_file_read_iter(iocb, to); | ||||||
|  | 	pagefault_enable(); | ||||||
| 	if (ret > 0) | 	if (ret > 0) | ||||||
| 		written += ret; | 		written += ret; | ||||||
|  | 
 | ||||||
|  | 	if (should_fault_in_pages(ret, to, &prev_count, &window_size)) { | ||||||
|  | 		size_t leftover; | ||||||
|  | 
 | ||||||
|  | 		gfs2_holder_allow_demote(&gh); | ||||||
|  | 		leftover = fault_in_iov_iter_writeable(to, window_size); | ||||||
|  | 		gfs2_holder_disallow_demote(&gh); | ||||||
|  | 		if (leftover != window_size) { | ||||||
|  | 			if (!gfs2_holder_queued(&gh)) { | ||||||
|  | 				if (written) | ||||||
|  | 					goto out_uninit; | ||||||
|  | 				goto retry; | ||||||
|  | 			} | ||||||
|  | 			goto retry_under_glock; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if (gfs2_holder_queued(&gh)) | ||||||
| 		gfs2_glock_dq(&gh); | 		gfs2_glock_dq(&gh); | ||||||
| out_uninit: | out_uninit: | ||||||
| 	gfs2_holder_uninit(&gh); | 	gfs2_holder_uninit(&gh); | ||||||
| 	return written ? written : ret; | 	return written ? written : ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static ssize_t gfs2_file_buffered_write(struct kiocb *iocb, | ||||||
|  | 					struct iov_iter *from, | ||||||
|  | 					struct gfs2_holder *gh) | ||||||
|  | { | ||||||
|  | 	struct file *file = iocb->ki_filp; | ||||||
|  | 	struct inode *inode = file_inode(file); | ||||||
|  | 	struct gfs2_inode *ip = GFS2_I(inode); | ||||||
|  | 	struct gfs2_sbd *sdp = GFS2_SB(inode); | ||||||
|  | 	struct gfs2_holder *statfs_gh = NULL; | ||||||
|  | 	size_t prev_count = 0, window_size = 0; | ||||||
|  | 	size_t read = 0; | ||||||
|  | 	ssize_t ret; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * In this function, we disable page faults when we're holding the | ||||||
|  | 	 * inode glock while doing I/O.  If a page fault occurs, we indicate | ||||||
|  | 	 * that the inode glock may be dropped, fault in the pages manually, | ||||||
|  | 	 * and retry. | ||||||
|  | 	 */ | ||||||
|  | 
 | ||||||
|  | 	if (inode == sdp->sd_rindex) { | ||||||
|  | 		statfs_gh = kmalloc(sizeof(*statfs_gh), GFP_NOFS); | ||||||
|  | 		if (!statfs_gh) | ||||||
|  | 			return -ENOMEM; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	gfs2_holder_init(ip->i_gl, LM_ST_EXCLUSIVE, 0, gh); | ||||||
|  | retry: | ||||||
|  | 	ret = gfs2_glock_nq(gh); | ||||||
|  | 	if (ret) | ||||||
|  | 		goto out_uninit; | ||||||
|  | retry_under_glock: | ||||||
|  | 	if (inode == sdp->sd_rindex) { | ||||||
|  | 		struct gfs2_inode *m_ip = GFS2_I(sdp->sd_statfs_inode); | ||||||
|  | 
 | ||||||
|  | 		ret = gfs2_glock_nq_init(m_ip->i_gl, LM_ST_EXCLUSIVE, | ||||||
|  | 					 GL_NOCACHE, statfs_gh); | ||||||
|  | 		if (ret) | ||||||
|  | 			goto out_unlock; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	current->backing_dev_info = inode_to_bdi(inode); | ||||||
|  | 	pagefault_disable(); | ||||||
|  | 	ret = iomap_file_buffered_write(iocb, from, &gfs2_iomap_ops); | ||||||
|  | 	pagefault_enable(); | ||||||
|  | 	current->backing_dev_info = NULL; | ||||||
|  | 	if (ret > 0) { | ||||||
|  | 		iocb->ki_pos += ret; | ||||||
|  | 		read += ret; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (inode == sdp->sd_rindex) | ||||||
|  | 		gfs2_glock_dq_uninit(statfs_gh); | ||||||
|  | 
 | ||||||
|  | 	if (should_fault_in_pages(ret, from, &prev_count, &window_size)) { | ||||||
|  | 		size_t leftover; | ||||||
|  | 
 | ||||||
|  | 		gfs2_holder_allow_demote(gh); | ||||||
|  | 		leftover = fault_in_iov_iter_readable(from, window_size); | ||||||
|  | 		gfs2_holder_disallow_demote(gh); | ||||||
|  | 		if (leftover != window_size) { | ||||||
|  | 			if (!gfs2_holder_queued(gh)) { | ||||||
|  | 				if (read) | ||||||
|  | 					goto out_uninit; | ||||||
|  | 				goto retry; | ||||||
|  | 			} | ||||||
|  | 			goto retry_under_glock; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | out_unlock: | ||||||
|  | 	if (gfs2_holder_queued(gh)) | ||||||
|  | 		gfs2_glock_dq(gh); | ||||||
|  | out_uninit: | ||||||
|  | 	gfs2_holder_uninit(gh); | ||||||
|  | 	if (statfs_gh) | ||||||
|  | 		kfree(statfs_gh); | ||||||
|  | 	return read ? read : ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * gfs2_file_write_iter - Perform a write to a file |  * gfs2_file_write_iter - Perform a write to a file | ||||||
|  * @iocb: The io context |  * @iocb: The io context | ||||||
| @ -927,9 +1140,7 @@ static ssize_t gfs2_file_write_iter(struct kiocb *iocb, struct iov_iter *from) | |||||||
| 			goto out_unlock; | 			goto out_unlock; | ||||||
| 
 | 
 | ||||||
| 		iocb->ki_flags |= IOCB_DSYNC; | 		iocb->ki_flags |= IOCB_DSYNC; | ||||||
| 		current->backing_dev_info = inode_to_bdi(inode); | 		buffered = gfs2_file_buffered_write(iocb, from, &gh); | ||||||
| 		buffered = iomap_file_buffered_write(iocb, from, &gfs2_iomap_ops); |  | ||||||
| 		current->backing_dev_info = NULL; |  | ||||||
| 		if (unlikely(buffered <= 0)) { | 		if (unlikely(buffered <= 0)) { | ||||||
| 			if (!ret) | 			if (!ret) | ||||||
| 				ret = buffered; | 				ret = buffered; | ||||||
| @ -943,7 +1154,6 @@ static ssize_t gfs2_file_write_iter(struct kiocb *iocb, struct iov_iter *from) | |||||||
| 		 * the direct I/O range as we don't know if the buffered pages | 		 * the direct I/O range as we don't know if the buffered pages | ||||||
| 		 * made it to disk. | 		 * made it to disk. | ||||||
| 		 */ | 		 */ | ||||||
| 		iocb->ki_pos += buffered; |  | ||||||
| 		ret2 = generic_write_sync(iocb, buffered); | 		ret2 = generic_write_sync(iocb, buffered); | ||||||
| 		invalidate_mapping_pages(mapping, | 		invalidate_mapping_pages(mapping, | ||||||
| 				(iocb->ki_pos - buffered) >> PAGE_SHIFT, | 				(iocb->ki_pos - buffered) >> PAGE_SHIFT, | ||||||
| @ -951,14 +1161,10 @@ static ssize_t gfs2_file_write_iter(struct kiocb *iocb, struct iov_iter *from) | |||||||
| 		if (!ret || ret2 > 0) | 		if (!ret || ret2 > 0) | ||||||
| 			ret += ret2; | 			ret += ret2; | ||||||
| 	} else { | 	} else { | ||||||
| 		current->backing_dev_info = inode_to_bdi(inode); | 		ret = gfs2_file_buffered_write(iocb, from, &gh); | ||||||
| 		ret = iomap_file_buffered_write(iocb, from, &gfs2_iomap_ops); | 		if (likely(ret > 0)) | ||||||
| 		current->backing_dev_info = NULL; |  | ||||||
| 		if (likely(ret > 0)) { |  | ||||||
| 			iocb->ki_pos += ret; |  | ||||||
| 			ret = generic_write_sync(iocb, ret); | 			ret = generic_write_sync(iocb, ret); | ||||||
| 	} | 	} | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| out_unlock: | out_unlock: | ||||||
| 	inode_unlock(inode); | 	inode_unlock(inode); | ||||||
|  | |||||||
							
								
								
									
										296
									
								
								fs/gfs2/glock.c
									
									
									
									
									
								
							
							
						
						
									
										296
									
								
								fs/gfs2/glock.c
									
									
									
									
									
								
							| @ -58,6 +58,7 @@ struct gfs2_glock_iter { | |||||||
| typedef void (*glock_examiner) (struct gfs2_glock * gl); | typedef void (*glock_examiner) (struct gfs2_glock * gl); | ||||||
| 
 | 
 | ||||||
| static void do_xmote(struct gfs2_glock *gl, struct gfs2_holder *gh, unsigned int target); | static void do_xmote(struct gfs2_glock *gl, struct gfs2_holder *gh, unsigned int target); | ||||||
|  | static void __gfs2_glock_dq(struct gfs2_holder *gh); | ||||||
| 
 | 
 | ||||||
| static struct dentry *gfs2_root; | static struct dentry *gfs2_root; | ||||||
| static struct workqueue_struct *glock_workqueue; | static struct workqueue_struct *glock_workqueue; | ||||||
| @ -197,6 +198,12 @@ static int demote_ok(const struct gfs2_glock *gl) | |||||||
| 
 | 
 | ||||||
| 	if (gl->gl_state == LM_ST_UNLOCKED) | 	if (gl->gl_state == LM_ST_UNLOCKED) | ||||||
| 		return 0; | 		return 0; | ||||||
|  | 	/*
 | ||||||
|  | 	 * Note that demote_ok is used for the lru process of disposing of | ||||||
|  | 	 * glocks. For this purpose, we don't care if the glock's holders | ||||||
|  | 	 * have the HIF_MAY_DEMOTE flag set or not. If someone is using | ||||||
|  | 	 * them, don't demote. | ||||||
|  | 	 */ | ||||||
| 	if (!list_empty(&gl->gl_holders)) | 	if (!list_empty(&gl->gl_holders)) | ||||||
| 		return 0; | 		return 0; | ||||||
| 	if (glops->go_demote_ok) | 	if (glops->go_demote_ok) | ||||||
| @ -301,46 +308,59 @@ void gfs2_glock_put(struct gfs2_glock *gl) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * may_grant - check if its ok to grant a new lock |  * may_grant - check if it's ok to grant a new lock | ||||||
|  * @gl: The glock |  * @gl: The glock | ||||||
|  |  * @current_gh: One of the current holders of @gl | ||||||
|  * @gh: The lock request which we wish to grant |  * @gh: The lock request which we wish to grant | ||||||
|  * |  * | ||||||
|  * Returns: true if its ok to grant the lock |  * With our current compatibility rules, if a glock has one or more active | ||||||
|  |  * holders (HIF_HOLDER flag set), any of those holders can be passed in as | ||||||
|  |  * @current_gh; they are all the same as far as compatibility with the new @gh | ||||||
|  |  * goes. | ||||||
|  |  * | ||||||
|  |  * Returns true if it's ok to grant the lock. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| static inline int may_grant(const struct gfs2_glock *gl, const struct gfs2_holder *gh) | static inline bool may_grant(struct gfs2_glock *gl, | ||||||
|  | 			     struct gfs2_holder *current_gh, | ||||||
|  | 			     struct gfs2_holder *gh) | ||||||
| { | { | ||||||
| 	const struct gfs2_holder *gh_head = list_first_entry(&gl->gl_holders, const struct gfs2_holder, gh_list); | 	if (current_gh) { | ||||||
|  | 		GLOCK_BUG_ON(gl, !test_bit(HIF_HOLDER, ¤t_gh->gh_iflags)); | ||||||
| 
 | 
 | ||||||
| 	if (gh != gh_head) { | 		switch(current_gh->gh_state) { | ||||||
| 		/**
 | 		case LM_ST_EXCLUSIVE: | ||||||
| 		 * Here we make a special exception to grant holders who agree | 			/*
 | ||||||
| 		 * to share the EX lock with other holders who also have the | 			 * Here we make a special exception to grant holders | ||||||
| 		 * bit set. If the original holder has the LM_FLAG_NODE_SCOPE bit | 			 * who agree to share the EX lock with other holders | ||||||
| 		 * is set, we grant more holders with the bit set. | 			 * who also have the bit set. If the original holder | ||||||
|  | 			 * has the LM_FLAG_NODE_SCOPE bit set, we grant more | ||||||
|  | 			 * holders with the bit set. | ||||||
| 			 */ | 			 */ | ||||||
| 		if (gh_head->gh_state == LM_ST_EXCLUSIVE && | 			return gh->gh_state == LM_ST_EXCLUSIVE && | ||||||
| 		    (gh_head->gh_flags & LM_FLAG_NODE_SCOPE) && | 			       (current_gh->gh_flags & LM_FLAG_NODE_SCOPE) && | ||||||
| 		    gh->gh_state == LM_ST_EXCLUSIVE && | 			       (gh->gh_flags & LM_FLAG_NODE_SCOPE); | ||||||
| 		    (gh->gh_flags & LM_FLAG_NODE_SCOPE)) | 
 | ||||||
| 			return 1; | 		case LM_ST_SHARED: | ||||||
| 		if ((gh->gh_state == LM_ST_EXCLUSIVE || | 		case LM_ST_DEFERRED: | ||||||
| 		     gh_head->gh_state == LM_ST_EXCLUSIVE)) | 			return gh->gh_state == current_gh->gh_state; | ||||||
| 			return 0; | 
 | ||||||
|  | 		default: | ||||||
|  | 			return false; | ||||||
| 		} | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	if (gl->gl_state == gh->gh_state) | 	if (gl->gl_state == gh->gh_state) | ||||||
| 		return 1; | 		return true; | ||||||
| 	if (gh->gh_flags & GL_EXACT) | 	if (gh->gh_flags & GL_EXACT) | ||||||
| 		return 0; | 		return false; | ||||||
| 	if (gl->gl_state == LM_ST_EXCLUSIVE) { | 	if (gl->gl_state == LM_ST_EXCLUSIVE) { | ||||||
| 		if (gh->gh_state == LM_ST_SHARED && gh_head->gh_state == LM_ST_SHARED) | 		return gh->gh_state == LM_ST_SHARED || | ||||||
| 			return 1; | 		       gh->gh_state == LM_ST_DEFERRED; | ||||||
| 		if (gh->gh_state == LM_ST_DEFERRED && gh_head->gh_state == LM_ST_DEFERRED) |  | ||||||
| 			return 1; |  | ||||||
| 	} | 	} | ||||||
| 	if (gl->gl_state != LM_ST_UNLOCKED && (gh->gh_flags & LM_FLAG_ANY)) | 	if (gh->gh_flags & LM_FLAG_ANY) | ||||||
| 		return 1; | 		return gl->gl_state != LM_ST_UNLOCKED; | ||||||
| 	return 0; | 	return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void gfs2_holder_wake(struct gfs2_holder *gh) | static void gfs2_holder_wake(struct gfs2_holder *gh) | ||||||
| @ -366,7 +386,7 @@ static void do_error(struct gfs2_glock *gl, const int ret) | |||||||
| 	struct gfs2_holder *gh, *tmp; | 	struct gfs2_holder *gh, *tmp; | ||||||
| 
 | 
 | ||||||
| 	list_for_each_entry_safe(gh, tmp, &gl->gl_holders, gh_list) { | 	list_for_each_entry_safe(gh, tmp, &gl->gl_holders, gh_list) { | ||||||
| 		if (test_bit(HIF_HOLDER, &gh->gh_iflags)) | 		if (!test_bit(HIF_WAIT, &gh->gh_iflags)) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (ret & LM_OUT_ERROR) | 		if (ret & LM_OUT_ERROR) | ||||||
| 			gh->gh_error = -EIO; | 			gh->gh_error = -EIO; | ||||||
| @ -380,6 +400,78 @@ static void do_error(struct gfs2_glock *gl, const int ret) | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * demote_incompat_holders - demote incompatible demoteable holders | ||||||
|  |  * @gl: the glock we want to promote | ||||||
|  |  * @new_gh: the new holder to be promoted | ||||||
|  |  */ | ||||||
|  | static void demote_incompat_holders(struct gfs2_glock *gl, | ||||||
|  | 				    struct gfs2_holder *new_gh) | ||||||
|  | { | ||||||
|  | 	struct gfs2_holder *gh; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Demote incompatible holders before we make ourselves eligible. | ||||||
|  | 	 * (This holder may or may not allow auto-demoting, but we don't want | ||||||
|  | 	 * to demote the new holder before it's even granted.) | ||||||
|  | 	 */ | ||||||
|  | 	list_for_each_entry(gh, &gl->gl_holders, gh_list) { | ||||||
|  | 		/*
 | ||||||
|  | 		 * Since holders are at the front of the list, we stop when we | ||||||
|  | 		 * find the first non-holder. | ||||||
|  | 		 */ | ||||||
|  | 		if (!test_bit(HIF_HOLDER, &gh->gh_iflags)) | ||||||
|  | 			return; | ||||||
|  | 		if (test_bit(HIF_MAY_DEMOTE, &gh->gh_iflags) && | ||||||
|  | 		    !may_grant(gl, new_gh, gh)) { | ||||||
|  | 			/*
 | ||||||
|  | 			 * We should not recurse into do_promote because | ||||||
|  | 			 * __gfs2_glock_dq only calls handle_callback, | ||||||
|  | 			 * gfs2_glock_add_to_lru and __gfs2_glock_queue_work. | ||||||
|  | 			 */ | ||||||
|  | 			__gfs2_glock_dq(gh); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * find_first_holder - find the first "holder" gh | ||||||
|  |  * @gl: the glock | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | static inline struct gfs2_holder *find_first_holder(const struct gfs2_glock *gl) | ||||||
|  | { | ||||||
|  | 	struct gfs2_holder *gh; | ||||||
|  | 
 | ||||||
|  | 	if (!list_empty(&gl->gl_holders)) { | ||||||
|  | 		gh = list_first_entry(&gl->gl_holders, struct gfs2_holder, | ||||||
|  | 				      gh_list); | ||||||
|  | 		if (test_bit(HIF_HOLDER, &gh->gh_iflags)) | ||||||
|  | 			return gh; | ||||||
|  | 	} | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * find_first_strong_holder - find the first non-demoteable holder | ||||||
|  |  * @gl: the glock | ||||||
|  |  * | ||||||
|  |  * Find the first holder that doesn't have the HIF_MAY_DEMOTE flag set. | ||||||
|  |  */ | ||||||
|  | static inline struct gfs2_holder * | ||||||
|  | find_first_strong_holder(struct gfs2_glock *gl) | ||||||
|  | { | ||||||
|  | 	struct gfs2_holder *gh; | ||||||
|  | 
 | ||||||
|  | 	list_for_each_entry(gh, &gl->gl_holders, gh_list) { | ||||||
|  | 		if (!test_bit(HIF_HOLDER, &gh->gh_iflags)) | ||||||
|  | 			return NULL; | ||||||
|  | 		if (!test_bit(HIF_MAY_DEMOTE, &gh->gh_iflags)) | ||||||
|  | 			return gh; | ||||||
|  | 	} | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * do_promote - promote as many requests as possible on the current queue |  * do_promote - promote as many requests as possible on the current queue | ||||||
|  * @gl: The glock |  * @gl: The glock | ||||||
| @ -393,14 +485,21 @@ __releases(&gl->gl_lockref.lock) | |||||||
| __acquires(&gl->gl_lockref.lock) | __acquires(&gl->gl_lockref.lock) | ||||||
| { | { | ||||||
| 	const struct gfs2_glock_operations *glops = gl->gl_ops; | 	const struct gfs2_glock_operations *glops = gl->gl_ops; | ||||||
| 	struct gfs2_holder *gh, *tmp; | 	struct gfs2_holder *gh, *tmp, *first_gh; | ||||||
|  | 	bool incompat_holders_demoted = false; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
| restart: | restart: | ||||||
|  | 	first_gh = find_first_strong_holder(gl); | ||||||
| 	list_for_each_entry_safe(gh, tmp, &gl->gl_holders, gh_list) { | 	list_for_each_entry_safe(gh, tmp, &gl->gl_holders, gh_list) { | ||||||
| 		if (test_bit(HIF_HOLDER, &gh->gh_iflags)) | 		if (!test_bit(HIF_WAIT, &gh->gh_iflags)) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (may_grant(gl, gh)) { | 		if (may_grant(gl, first_gh, gh)) { | ||||||
|  | 			if (!incompat_holders_demoted) { | ||||||
|  | 				demote_incompat_holders(gl, first_gh); | ||||||
|  | 				incompat_holders_demoted = true; | ||||||
|  | 				first_gh = gh; | ||||||
|  | 			} | ||||||
| 			if (gh->gh_list.prev == &gl->gl_holders && | 			if (gh->gh_list.prev == &gl->gl_holders && | ||||||
| 			    glops->go_lock) { | 			    glops->go_lock) { | ||||||
| 				spin_unlock(&gl->gl_lockref.lock); | 				spin_unlock(&gl->gl_lockref.lock); | ||||||
| @ -426,6 +525,11 @@ restart: | |||||||
| 			gfs2_holder_wake(gh); | 			gfs2_holder_wake(gh); | ||||||
| 			continue; | 			continue; | ||||||
| 		} | 		} | ||||||
|  | 		/*
 | ||||||
|  | 		 * If we get here, it means we may not grant this holder for | ||||||
|  | 		 * some reason. If this holder is the head of the list, it | ||||||
|  | 		 * means we have a blocked holder at the head, so return 1. | ||||||
|  | 		 */ | ||||||
| 		if (gh->gh_list.prev == &gl->gl_holders) | 		if (gh->gh_list.prev == &gl->gl_holders) | ||||||
| 			return 1; | 			return 1; | ||||||
| 		do_error(gl, 0); | 		do_error(gl, 0); | ||||||
| @ -722,23 +826,6 @@ out: | |||||||
| 	spin_lock(&gl->gl_lockref.lock); | 	spin_lock(&gl->gl_lockref.lock); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 |  | ||||||
|  * find_first_holder - find the first "holder" gh |  | ||||||
|  * @gl: the glock |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| static inline struct gfs2_holder *find_first_holder(const struct gfs2_glock *gl) |  | ||||||
| { |  | ||||||
| 	struct gfs2_holder *gh; |  | ||||||
| 
 |  | ||||||
| 	if (!list_empty(&gl->gl_holders)) { |  | ||||||
| 		gh = list_first_entry(&gl->gl_holders, struct gfs2_holder, gh_list); |  | ||||||
| 		if (test_bit(HIF_HOLDER, &gh->gh_iflags)) |  | ||||||
| 			return gh; |  | ||||||
| 	} |  | ||||||
| 	return NULL; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /**
 | /**
 | ||||||
|  * run_queue - do all outstanding tasks related to a glock |  * run_queue - do all outstanding tasks related to a glock | ||||||
|  * @gl: The glock in question |  * @gl: The glock in question | ||||||
| @ -1354,15 +1441,20 @@ __acquires(&gl->gl_lockref.lock) | |||||||
| 		GLOCK_BUG_ON(gl, true); | 		GLOCK_BUG_ON(gl, true); | ||||||
| 
 | 
 | ||||||
| 	if (gh->gh_flags & (LM_FLAG_TRY | LM_FLAG_TRY_1CB)) { | 	if (gh->gh_flags & (LM_FLAG_TRY | LM_FLAG_TRY_1CB)) { | ||||||
| 		if (test_bit(GLF_LOCK, &gl->gl_flags)) | 		if (test_bit(GLF_LOCK, &gl->gl_flags)) { | ||||||
| 			try_futile = !may_grant(gl, gh); | 			struct gfs2_holder *first_gh; | ||||||
|  | 
 | ||||||
|  | 			first_gh = find_first_strong_holder(gl); | ||||||
|  | 			try_futile = !may_grant(gl, first_gh, gh); | ||||||
|  | 		} | ||||||
| 		if (test_bit(GLF_INVALIDATE_IN_PROGRESS, &gl->gl_flags)) | 		if (test_bit(GLF_INVALIDATE_IN_PROGRESS, &gl->gl_flags)) | ||||||
| 			goto fail; | 			goto fail; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	list_for_each_entry(gh2, &gl->gl_holders, gh_list) { | 	list_for_each_entry(gh2, &gl->gl_holders, gh_list) { | ||||||
| 		if (unlikely(gh2->gh_owner_pid == gh->gh_owner_pid && | 		if (unlikely(gh2->gh_owner_pid == gh->gh_owner_pid && | ||||||
| 		    (gh->gh_gl->gl_ops->go_type != LM_TYPE_FLOCK))) | 		    (gh->gh_gl->gl_ops->go_type != LM_TYPE_FLOCK) && | ||||||
|  | 		    !test_bit(HIF_MAY_DEMOTE, &gh2->gh_iflags))) | ||||||
| 			goto trap_recursive; | 			goto trap_recursive; | ||||||
| 		if (try_futile && | 		if (try_futile && | ||||||
| 		    !(gh2->gh_flags & (LM_FLAG_TRY | LM_FLAG_TRY_1CB))) { | 		    !(gh2->gh_flags & (LM_FLAG_TRY | LM_FLAG_TRY_1CB))) { | ||||||
| @ -1458,26 +1550,32 @@ int gfs2_glock_poll(struct gfs2_holder *gh) | |||||||
| 	return test_bit(HIF_WAIT, &gh->gh_iflags) ? 0 : 1; | 	return test_bit(HIF_WAIT, &gh->gh_iflags) ? 0 : 1; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | static inline bool needs_demote(struct gfs2_glock *gl) | ||||||
|  * gfs2_glock_dq - dequeue a struct gfs2_holder from a glock (release a glock) | { | ||||||
|  * @gh: the glock holder | 	return (test_bit(GLF_DEMOTE, &gl->gl_flags) || | ||||||
|  * | 		test_bit(GLF_PENDING_DEMOTE, &gl->gl_flags)); | ||||||
|  */ | } | ||||||
| 
 | 
 | ||||||
| void gfs2_glock_dq(struct gfs2_holder *gh) | static void __gfs2_glock_dq(struct gfs2_holder *gh) | ||||||
| { | { | ||||||
| 	struct gfs2_glock *gl = gh->gh_gl; | 	struct gfs2_glock *gl = gh->gh_gl; | ||||||
| 	struct gfs2_sbd *sdp = gl->gl_name.ln_sbd; | 	struct gfs2_sbd *sdp = gl->gl_name.ln_sbd; | ||||||
| 	unsigned delay = 0; | 	unsigned delay = 0; | ||||||
| 	int fast_path = 0; | 	int fast_path = 0; | ||||||
| 
 | 
 | ||||||
| 	spin_lock(&gl->gl_lockref.lock); |  | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * If we're in the process of file system withdraw, we cannot just | 	 * This while loop is similar to function demote_incompat_holders: | ||||||
| 	 * dequeue any glocks until our journal is recovered, lest we | 	 * If the glock is due to be demoted (which may be from another node | ||||||
| 	 * introduce file system corruption. We need two exceptions to this | 	 * or even if this holder is GL_NOCACHE), the weak holders are | ||||||
| 	 * rule: We need to allow unlocking of nondisk glocks and the glock | 	 * demoted as well, allowing the glock to be demoted. | ||||||
| 	 * for our own journal that needs recovery. | 	 */ | ||||||
|  | 	while (gh) { | ||||||
|  | 		/*
 | ||||||
|  | 		 * If we're in the process of file system withdraw, we cannot | ||||||
|  | 		 * just dequeue any glocks until our journal is recovered, lest | ||||||
|  | 		 * we introduce file system corruption. We need two exceptions | ||||||
|  | 		 * to this rule: We need to allow unlocking of nondisk glocks | ||||||
|  | 		 * and the glock for our own journal that needs recovery. | ||||||
| 		 */ | 		 */ | ||||||
| 		if (test_bit(SDF_WITHDRAW_RECOVERY, &sdp->sd_flags) && | 		if (test_bit(SDF_WITHDRAW_RECOVERY, &sdp->sd_flags) && | ||||||
| 		    glock_blocked_by_withdraw(gl) && | 		    glock_blocked_by_withdraw(gl) && | ||||||
| @ -1489,20 +1587,46 @@ void gfs2_glock_dq(struct gfs2_holder *gh) | |||||||
| 				    TASK_UNINTERRUPTIBLE); | 				    TASK_UNINTERRUPTIBLE); | ||||||
| 			spin_lock(&gl->gl_lockref.lock); | 			spin_lock(&gl->gl_lockref.lock); | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
|  | 		/*
 | ||||||
|  | 		 * This holder should not be cached, so mark it for demote. | ||||||
|  | 		 * Note: this should be done before the check for needs_demote | ||||||
|  | 		 * below. | ||||||
|  | 		 */ | ||||||
| 		if (gh->gh_flags & GL_NOCACHE) | 		if (gh->gh_flags & GL_NOCACHE) | ||||||
| 			handle_callback(gl, LM_ST_UNLOCKED, 0, false); | 			handle_callback(gl, LM_ST_UNLOCKED, 0, false); | ||||||
| 
 | 
 | ||||||
| 		list_del_init(&gh->gh_list); | 		list_del_init(&gh->gh_list); | ||||||
| 		clear_bit(HIF_HOLDER, &gh->gh_iflags); | 		clear_bit(HIF_HOLDER, &gh->gh_iflags); | ||||||
| 	if (list_empty(&gl->gl_holders) && | 		trace_gfs2_glock_queue(gh, 0); | ||||||
| 	    !test_bit(GLF_PENDING_DEMOTE, &gl->gl_flags) && | 
 | ||||||
| 	    !test_bit(GLF_DEMOTE, &gl->gl_flags)) | 		/*
 | ||||||
|  | 		 * If there hasn't been a demote request we are done. | ||||||
|  | 		 * (Let the remaining holders, if any, keep holding it.) | ||||||
|  | 		 */ | ||||||
|  | 		if (!needs_demote(gl)) { | ||||||
|  | 			if (list_empty(&gl->gl_holders)) | ||||||
| 				fast_path = 1; | 				fast_path = 1; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 		/*
 | ||||||
|  | 		 * If we have another strong holder (we cannot auto-demote) | ||||||
|  | 		 * we are done. It keeps holding it until it is done. | ||||||
|  | 		 */ | ||||||
|  | 		if (find_first_strong_holder(gl)) | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		/*
 | ||||||
|  | 		 * If we have a weak holder at the head of the list, it | ||||||
|  | 		 * (and all others like it) must be auto-demoted. If there | ||||||
|  | 		 * are no more weak holders, we exit the while loop. | ||||||
|  | 		 */ | ||||||
|  | 		gh = find_first_holder(gl); | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	if (!test_bit(GLF_LFLUSH, &gl->gl_flags) && demote_ok(gl)) | 	if (!test_bit(GLF_LFLUSH, &gl->gl_flags) && demote_ok(gl)) | ||||||
| 		gfs2_glock_add_to_lru(gl); | 		gfs2_glock_add_to_lru(gl); | ||||||
| 
 | 
 | ||||||
| 	trace_gfs2_glock_queue(gh, 0); |  | ||||||
| 	if (unlikely(!fast_path)) { | 	if (unlikely(!fast_path)) { | ||||||
| 		gl->gl_lockref.count++; | 		gl->gl_lockref.count++; | ||||||
| 		if (test_bit(GLF_PENDING_DEMOTE, &gl->gl_flags) && | 		if (test_bit(GLF_PENDING_DEMOTE, &gl->gl_flags) && | ||||||
| @ -1511,6 +1635,19 @@ void gfs2_glock_dq(struct gfs2_holder *gh) | |||||||
| 			delay = gl->gl_hold_time; | 			delay = gl->gl_hold_time; | ||||||
| 		__gfs2_glock_queue_work(gl, delay); | 		__gfs2_glock_queue_work(gl, delay); | ||||||
| 	} | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * gfs2_glock_dq - dequeue a struct gfs2_holder from a glock (release a glock) | ||||||
|  |  * @gh: the glock holder | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | void gfs2_glock_dq(struct gfs2_holder *gh) | ||||||
|  | { | ||||||
|  | 	struct gfs2_glock *gl = gh->gh_gl; | ||||||
|  | 
 | ||||||
|  | 	spin_lock(&gl->gl_lockref.lock); | ||||||
|  | 	__gfs2_glock_dq(gh); | ||||||
| 	spin_unlock(&gl->gl_lockref.lock); | 	spin_unlock(&gl->gl_lockref.lock); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1673,6 +1810,7 @@ void gfs2_glock_dq_m(unsigned int num_gh, struct gfs2_holder *ghs) | |||||||
| 
 | 
 | ||||||
| void gfs2_glock_cb(struct gfs2_glock *gl, unsigned int state) | void gfs2_glock_cb(struct gfs2_glock *gl, unsigned int state) | ||||||
| { | { | ||||||
|  | 	struct gfs2_holder mock_gh = { .gh_gl = gl, .gh_state = state, }; | ||||||
| 	unsigned long delay = 0; | 	unsigned long delay = 0; | ||||||
| 	unsigned long holdtime; | 	unsigned long holdtime; | ||||||
| 	unsigned long now = jiffies; | 	unsigned long now = jiffies; | ||||||
| @ -1687,6 +1825,28 @@ void gfs2_glock_cb(struct gfs2_glock *gl, unsigned int state) | |||||||
| 		if (test_bit(GLF_REPLY_PENDING, &gl->gl_flags)) | 		if (test_bit(GLF_REPLY_PENDING, &gl->gl_flags)) | ||||||
| 			delay = gl->gl_hold_time; | 			delay = gl->gl_hold_time; | ||||||
| 	} | 	} | ||||||
|  | 	/*
 | ||||||
|  | 	 * Note 1: We cannot call demote_incompat_holders from handle_callback | ||||||
|  | 	 * or gfs2_set_demote due to recursion problems like: gfs2_glock_dq -> | ||||||
|  | 	 * handle_callback -> demote_incompat_holders -> gfs2_glock_dq | ||||||
|  | 	 * Plus, we only want to demote the holders if the request comes from | ||||||
|  | 	 * a remote cluster node because local holder conflicts are resolved | ||||||
|  | 	 * elsewhere. | ||||||
|  | 	 * | ||||||
|  | 	 * Note 2: if a remote node wants this glock in EX mode, lock_dlm will | ||||||
|  | 	 * request that we set our state to UNLOCKED. Here we mock up a holder | ||||||
|  | 	 * to make it look like someone wants the lock EX locally. Any SH | ||||||
|  | 	 * and DF requests should be able to share the lock without demoting. | ||||||
|  | 	 * | ||||||
|  | 	 * Note 3: We only want to demote the demoteable holders when there | ||||||
|  | 	 * are no more strong holders. The demoteable holders might as well | ||||||
|  | 	 * keep the glock until the last strong holder is done with it. | ||||||
|  | 	 */ | ||||||
|  | 	if (!find_first_strong_holder(gl)) { | ||||||
|  | 		if (state == LM_ST_UNLOCKED) | ||||||
|  | 			mock_gh.gh_state = LM_ST_EXCLUSIVE; | ||||||
|  | 		demote_incompat_holders(gl, &mock_gh); | ||||||
|  | 	} | ||||||
| 	handle_callback(gl, state, delay, true); | 	handle_callback(gl, state, delay, true); | ||||||
| 	__gfs2_glock_queue_work(gl, delay); | 	__gfs2_glock_queue_work(gl, delay); | ||||||
| 	spin_unlock(&gl->gl_lockref.lock); | 	spin_unlock(&gl->gl_lockref.lock); | ||||||
| @ -2076,6 +2236,8 @@ static const char *hflags2str(char *buf, u16 flags, unsigned long iflags) | |||||||
| 		*p++ = 'H'; | 		*p++ = 'H'; | ||||||
| 	if (test_bit(HIF_WAIT, &iflags)) | 	if (test_bit(HIF_WAIT, &iflags)) | ||||||
| 		*p++ = 'W'; | 		*p++ = 'W'; | ||||||
|  | 	if (test_bit(HIF_MAY_DEMOTE, &iflags)) | ||||||
|  | 		*p++ = 'D'; | ||||||
| 	*p = 0; | 	*p = 0; | ||||||
| 	return buf; | 	return buf; | ||||||
| } | } | ||||||
|  | |||||||
| @ -150,6 +150,8 @@ static inline struct gfs2_holder *gfs2_glock_is_locked_by_me(struct gfs2_glock * | |||||||
| 	list_for_each_entry(gh, &gl->gl_holders, gh_list) { | 	list_for_each_entry(gh, &gl->gl_holders, gh_list) { | ||||||
| 		if (!test_bit(HIF_HOLDER, &gh->gh_iflags)) | 		if (!test_bit(HIF_HOLDER, &gh->gh_iflags)) | ||||||
| 			break; | 			break; | ||||||
|  | 		if (test_bit(HIF_MAY_DEMOTE, &gh->gh_iflags)) | ||||||
|  | 			continue; | ||||||
| 		if (gh->gh_owner_pid == pid) | 		if (gh->gh_owner_pid == pid) | ||||||
| 			goto out; | 			goto out; | ||||||
| 	} | 	} | ||||||
| @ -325,6 +327,24 @@ static inline void glock_clear_object(struct gfs2_glock *gl, void *object) | |||||||
| 	spin_unlock(&gl->gl_lockref.lock); | 	spin_unlock(&gl->gl_lockref.lock); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static inline void gfs2_holder_allow_demote(struct gfs2_holder *gh) | ||||||
|  | { | ||||||
|  | 	struct gfs2_glock *gl = gh->gh_gl; | ||||||
|  | 
 | ||||||
|  | 	spin_lock(&gl->gl_lockref.lock); | ||||||
|  | 	set_bit(HIF_MAY_DEMOTE, &gh->gh_iflags); | ||||||
|  | 	spin_unlock(&gl->gl_lockref.lock); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void gfs2_holder_disallow_demote(struct gfs2_holder *gh) | ||||||
|  | { | ||||||
|  | 	struct gfs2_glock *gl = gh->gh_gl; | ||||||
|  | 
 | ||||||
|  | 	spin_lock(&gl->gl_lockref.lock); | ||||||
|  | 	clear_bit(HIF_MAY_DEMOTE, &gh->gh_iflags); | ||||||
|  | 	spin_unlock(&gl->gl_lockref.lock); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| extern void gfs2_inode_remember_delete(struct gfs2_glock *gl, u64 generation); | extern void gfs2_inode_remember_delete(struct gfs2_glock *gl, u64 generation); | ||||||
| extern bool gfs2_inode_already_deleted(struct gfs2_glock *gl, u64 generation); | extern bool gfs2_inode_already_deleted(struct gfs2_glock *gl, u64 generation); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -252,6 +252,7 @@ struct gfs2_lkstats { | |||||||
| 
 | 
 | ||||||
| enum { | enum { | ||||||
| 	/* States */ | 	/* States */ | ||||||
|  | 	HIF_MAY_DEMOTE		= 1, | ||||||
| 	HIF_HOLDER		= 6,  /* Set for gh that "holds" the glock */ | 	HIF_HOLDER		= 6,  /* Set for gh that "holds" the glock */ | ||||||
| 	HIF_WAIT		= 10, | 	HIF_WAIT		= 10, | ||||||
| }; | }; | ||||||
| @ -386,9 +387,8 @@ struct gfs2_inode { | |||||||
| 	u64 i_generation; | 	u64 i_generation; | ||||||
| 	u64 i_eattr; | 	u64 i_eattr; | ||||||
| 	unsigned long i_flags;		/* GIF_... */ | 	unsigned long i_flags;		/* GIF_... */ | ||||||
| 	struct gfs2_glock *i_gl; /* Move into i_gh? */ | 	struct gfs2_glock *i_gl; | ||||||
| 	struct gfs2_holder i_iopen_gh; | 	struct gfs2_holder i_iopen_gh; | ||||||
| 	struct gfs2_holder i_gh; /* for prepare/commit_write only */ |  | ||||||
| 	struct gfs2_qadata *i_qadata; /* quota allocation data */ | 	struct gfs2_qadata *i_qadata; /* quota allocation data */ | ||||||
| 	struct gfs2_holder i_rgd_gh; | 	struct gfs2_holder i_rgd_gh; | ||||||
| 	struct gfs2_blkreserv i_res; /* rgrp multi-block reservation */ | 	struct gfs2_blkreserv i_res; /* rgrp multi-block reservation */ | ||||||
|  | |||||||
| @ -750,7 +750,7 @@ again: | |||||||
| 		 * same page as we're writing to, without it being marked | 		 * same page as we're writing to, without it being marked | ||||||
| 		 * up-to-date. | 		 * up-to-date. | ||||||
| 		 */ | 		 */ | ||||||
| 		if (unlikely(iov_iter_fault_in_readable(i, bytes))) { | 		if (unlikely(fault_in_iov_iter_readable(i, bytes))) { | ||||||
| 			status = -EFAULT; | 			status = -EFAULT; | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -31,6 +31,7 @@ struct iomap_dio { | |||||||
| 	atomic_t		ref; | 	atomic_t		ref; | ||||||
| 	unsigned		flags; | 	unsigned		flags; | ||||||
| 	int			error; | 	int			error; | ||||||
|  | 	size_t			done_before; | ||||||
| 	bool			wait_for_completion; | 	bool			wait_for_completion; | ||||||
| 
 | 
 | ||||||
| 	union { | 	union { | ||||||
| @ -114,6 +115,9 @@ ssize_t iomap_dio_complete(struct iomap_dio *dio) | |||||||
| 	if (ret > 0 && (dio->flags & IOMAP_DIO_NEED_SYNC)) | 	if (ret > 0 && (dio->flags & IOMAP_DIO_NEED_SYNC)) | ||||||
| 		ret = generic_write_sync(iocb, ret); | 		ret = generic_write_sync(iocb, ret); | ||||||
| 
 | 
 | ||||||
|  | 	if (ret > 0) | ||||||
|  | 		ret += dio->done_before; | ||||||
|  | 
 | ||||||
| 	kfree(dio); | 	kfree(dio); | ||||||
| 
 | 
 | ||||||
| 	return ret; | 	return ret; | ||||||
| @ -375,6 +379,8 @@ static loff_t iomap_dio_hole_iter(const struct iomap_iter *iter, | |||||||
| 	loff_t length = iov_iter_zero(iomap_length(iter), dio->submit.iter); | 	loff_t length = iov_iter_zero(iomap_length(iter), dio->submit.iter); | ||||||
| 
 | 
 | ||||||
| 	dio->size += length; | 	dio->size += length; | ||||||
|  | 	if (!length) | ||||||
|  | 		return -EFAULT; | ||||||
| 	return length; | 	return length; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -406,6 +412,8 @@ static loff_t iomap_dio_inline_iter(const struct iomap_iter *iomi, | |||||||
| 		copied = copy_to_iter(inline_data, length, iter); | 		copied = copy_to_iter(inline_data, length, iter); | ||||||
| 	} | 	} | ||||||
| 	dio->size += copied; | 	dio->size += copied; | ||||||
|  | 	if (!copied) | ||||||
|  | 		return -EFAULT; | ||||||
| 	return copied; | 	return copied; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -450,13 +458,21 @@ static loff_t iomap_dio_iter(const struct iomap_iter *iter, | |||||||
|  * may be pure data writes. In that case, we still need to do a full data sync |  * may be pure data writes. In that case, we still need to do a full data sync | ||||||
|  * completion. |  * completion. | ||||||
|  * |  * | ||||||
|  |  * When page faults are disabled and @dio_flags includes IOMAP_DIO_PARTIAL, | ||||||
|  |  * __iomap_dio_rw can return a partial result if it encounters a non-resident | ||||||
|  |  * page in @iter after preparing a transfer.  In that case, the non-resident | ||||||
|  |  * pages can be faulted in and the request resumed with @done_before set to the | ||||||
|  |  * number of bytes previously transferred.  The request will then complete with | ||||||
|  |  * the correct total number of bytes transferred; this is essential for | ||||||
|  |  * completing partial requests asynchronously. | ||||||
|  |  * | ||||||
|  * Returns -ENOTBLK In case of a page invalidation invalidation failure for |  * Returns -ENOTBLK In case of a page invalidation invalidation failure for | ||||||
|  * writes.  The callers needs to fall back to buffered I/O in this case. |  * writes.  The callers needs to fall back to buffered I/O in this case. | ||||||
|  */ |  */ | ||||||
| struct iomap_dio * | struct iomap_dio * | ||||||
| __iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | __iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | ||||||
| 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | ||||||
| 		unsigned int dio_flags) | 		unsigned int dio_flags, size_t done_before) | ||||||
| { | { | ||||||
| 	struct address_space *mapping = iocb->ki_filp->f_mapping; | 	struct address_space *mapping = iocb->ki_filp->f_mapping; | ||||||
| 	struct inode *inode = file_inode(iocb->ki_filp); | 	struct inode *inode = file_inode(iocb->ki_filp); | ||||||
| @ -486,6 +502,7 @@ __iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | |||||||
| 	dio->dops = dops; | 	dio->dops = dops; | ||||||
| 	dio->error = 0; | 	dio->error = 0; | ||||||
| 	dio->flags = 0; | 	dio->flags = 0; | ||||||
|  | 	dio->done_before = done_before; | ||||||
| 
 | 
 | ||||||
| 	dio->submit.iter = iter; | 	dio->submit.iter = iter; | ||||||
| 	dio->submit.waiter = current; | 	dio->submit.waiter = current; | ||||||
| @ -587,6 +604,12 @@ __iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | |||||||
| 	if (iov_iter_rw(iter) == READ && iomi.pos >= dio->i_size) | 	if (iov_iter_rw(iter) == READ && iomi.pos >= dio->i_size) | ||||||
| 		iov_iter_revert(iter, iomi.pos - dio->i_size); | 		iov_iter_revert(iter, iomi.pos - dio->i_size); | ||||||
| 
 | 
 | ||||||
|  | 	if (ret == -EFAULT && dio->size && (dio_flags & IOMAP_DIO_PARTIAL)) { | ||||||
|  | 		if (!(iocb->ki_flags & IOCB_NOWAIT)) | ||||||
|  | 			wait_for_completion = true; | ||||||
|  | 		ret = 0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/* magic error code to fall back to buffered I/O */ | 	/* magic error code to fall back to buffered I/O */ | ||||||
| 	if (ret == -ENOTBLK) { | 	if (ret == -ENOTBLK) { | ||||||
| 		wait_for_completion = true; | 		wait_for_completion = true; | ||||||
| @ -649,11 +672,11 @@ EXPORT_SYMBOL_GPL(__iomap_dio_rw); | |||||||
| ssize_t | ssize_t | ||||||
| iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | ||||||
| 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | ||||||
| 		unsigned int dio_flags) | 		unsigned int dio_flags, size_t done_before) | ||||||
| { | { | ||||||
| 	struct iomap_dio *dio; | 	struct iomap_dio *dio; | ||||||
| 
 | 
 | ||||||
| 	dio = __iomap_dio_rw(iocb, iter, ops, dops, dio_flags); | 	dio = __iomap_dio_rw(iocb, iter, ops, dops, dio_flags, done_before); | ||||||
| 	if (IS_ERR_OR_NULL(dio)) | 	if (IS_ERR_OR_NULL(dio)) | ||||||
| 		return PTR_ERR_OR_ZERO(dio); | 		return PTR_ERR_OR_ZERO(dio); | ||||||
| 	return iomap_dio_complete(dio); | 	return iomap_dio_complete(dio); | ||||||
|  | |||||||
| @ -1830,7 +1830,7 @@ again: | |||||||
| 		 * pages being swapped out between us bringing them into memory | 		 * pages being swapped out between us bringing them into memory | ||||||
| 		 * and doing the actual copying. | 		 * and doing the actual copying. | ||||||
| 		 */ | 		 */ | ||||||
| 		if (unlikely(iov_iter_fault_in_readable(i, bytes))) { | 		if (unlikely(fault_in_iov_iter_readable(i, bytes))) { | ||||||
| 			status = -EFAULT; | 			status = -EFAULT; | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -990,7 +990,7 @@ static ssize_t ntfs_compress_write(struct kiocb *iocb, struct iov_iter *from) | |||||||
| 		frame_vbo = pos & ~(frame_size - 1); | 		frame_vbo = pos & ~(frame_size - 1); | ||||||
| 		index = frame_vbo >> PAGE_SHIFT; | 		index = frame_vbo >> PAGE_SHIFT; | ||||||
| 
 | 
 | ||||||
| 		if (unlikely(iov_iter_fault_in_readable(from, bytes))) { | 		if (unlikely(fault_in_iov_iter_readable(from, bytes))) { | ||||||
| 			err = -EFAULT; | 			err = -EFAULT; | ||||||
| 			goto out; | 			goto out; | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -259,7 +259,7 @@ xfs_file_dio_read( | |||||||
| 	ret = xfs_ilock_iocb(iocb, XFS_IOLOCK_SHARED); | 	ret = xfs_ilock_iocb(iocb, XFS_IOLOCK_SHARED); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		return ret; | 		return ret; | ||||||
| 	ret = iomap_dio_rw(iocb, to, &xfs_read_iomap_ops, NULL, 0); | 	ret = iomap_dio_rw(iocb, to, &xfs_read_iomap_ops, NULL, 0, 0); | ||||||
| 	xfs_iunlock(ip, XFS_IOLOCK_SHARED); | 	xfs_iunlock(ip, XFS_IOLOCK_SHARED); | ||||||
| 
 | 
 | ||||||
| 	return ret; | 	return ret; | ||||||
| @ -569,7 +569,7 @@ xfs_file_dio_write_aligned( | |||||||
| 	} | 	} | ||||||
| 	trace_xfs_file_direct_write(iocb, from); | 	trace_xfs_file_direct_write(iocb, from); | ||||||
| 	ret = iomap_dio_rw(iocb, from, &xfs_direct_write_iomap_ops, | 	ret = iomap_dio_rw(iocb, from, &xfs_direct_write_iomap_ops, | ||||||
| 			   &xfs_dio_write_ops, 0); | 			   &xfs_dio_write_ops, 0, 0); | ||||||
| out_unlock: | out_unlock: | ||||||
| 	if (iolock) | 	if (iolock) | ||||||
| 		xfs_iunlock(ip, iolock); | 		xfs_iunlock(ip, iolock); | ||||||
| @ -647,7 +647,7 @@ retry_exclusive: | |||||||
| 
 | 
 | ||||||
| 	trace_xfs_file_direct_write(iocb, from); | 	trace_xfs_file_direct_write(iocb, from); | ||||||
| 	ret = iomap_dio_rw(iocb, from, &xfs_direct_write_iomap_ops, | 	ret = iomap_dio_rw(iocb, from, &xfs_direct_write_iomap_ops, | ||||||
| 			   &xfs_dio_write_ops, flags); | 			   &xfs_dio_write_ops, flags, 0); | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Retry unaligned I/O with exclusive blocking semantics if the DIO | 	 * Retry unaligned I/O with exclusive blocking semantics if the DIO | ||||||
|  | |||||||
| @ -852,7 +852,7 @@ static ssize_t zonefs_file_dio_write(struct kiocb *iocb, struct iov_iter *from) | |||||||
| 		ret = zonefs_file_dio_append(iocb, from); | 		ret = zonefs_file_dio_append(iocb, from); | ||||||
| 	else | 	else | ||||||
| 		ret = iomap_dio_rw(iocb, from, &zonefs_iomap_ops, | 		ret = iomap_dio_rw(iocb, from, &zonefs_iomap_ops, | ||||||
| 				   &zonefs_write_dio_ops, 0); | 				   &zonefs_write_dio_ops, 0, 0); | ||||||
| 	if (zi->i_ztype == ZONEFS_ZTYPE_SEQ && | 	if (zi->i_ztype == ZONEFS_ZTYPE_SEQ && | ||||||
| 	    (ret > 0 || ret == -EIOCBQUEUED)) { | 	    (ret > 0 || ret == -EIOCBQUEUED)) { | ||||||
| 		if (ret > 0) | 		if (ret > 0) | ||||||
| @ -987,7 +987,7 @@ static ssize_t zonefs_file_read_iter(struct kiocb *iocb, struct iov_iter *to) | |||||||
| 		} | 		} | ||||||
| 		file_accessed(iocb->ki_filp); | 		file_accessed(iocb->ki_filp); | ||||||
| 		ret = iomap_dio_rw(iocb, to, &zonefs_iomap_ops, | 		ret = iomap_dio_rw(iocb, to, &zonefs_iomap_ops, | ||||||
| 				   &zonefs_read_dio_ops, 0); | 				   &zonefs_read_dio_ops, 0, 0); | ||||||
| 	} else { | 	} else { | ||||||
| 		ret = generic_file_read_iter(iocb, to); | 		ret = generic_file_read_iter(iocb, to); | ||||||
| 		if (ret == -EIO) | 		if (ret == -EIO) | ||||||
|  | |||||||
| @ -330,12 +330,19 @@ struct iomap_dio_ops { | |||||||
|   */ |   */ | ||||||
| #define IOMAP_DIO_OVERWRITE_ONLY	(1 << 1) | #define IOMAP_DIO_OVERWRITE_ONLY	(1 << 1) | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * When a page fault occurs, return a partial synchronous result and allow | ||||||
|  |  * the caller to retry the rest of the operation after dealing with the page | ||||||
|  |  * fault. | ||||||
|  |  */ | ||||||
|  | #define IOMAP_DIO_PARTIAL		(1 << 2) | ||||||
|  | 
 | ||||||
| ssize_t iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | ssize_t iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | ||||||
| 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | ||||||
| 		unsigned int dio_flags); | 		unsigned int dio_flags, size_t done_before); | ||||||
| struct iomap_dio *__iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | struct iomap_dio *__iomap_dio_rw(struct kiocb *iocb, struct iov_iter *iter, | ||||||
| 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | 		const struct iomap_ops *ops, const struct iomap_dio_ops *dops, | ||||||
| 		unsigned int dio_flags); | 		unsigned int dio_flags, size_t done_before); | ||||||
| ssize_t iomap_dio_complete(struct iomap_dio *dio); | ssize_t iomap_dio_complete(struct iomap_dio *dio); | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_SWAP | #ifdef CONFIG_SWAP | ||||||
|  | |||||||
| @ -2976,7 +2976,8 @@ struct page *follow_page(struct vm_area_struct *vma, unsigned long address, | |||||||
| #define FOLL_FORCE	0x10	/* get_user_pages read/write w/o permission */ | #define FOLL_FORCE	0x10	/* get_user_pages read/write w/o permission */ | ||||||
| #define FOLL_NOWAIT	0x20	/* if a disk transfer is needed, start the IO | #define FOLL_NOWAIT	0x20	/* if a disk transfer is needed, start the IO | ||||||
| 				 * and return without waiting upon it */ | 				 * and return without waiting upon it */ | ||||||
| #define FOLL_POPULATE	0x40	/* fault in page */ | #define FOLL_POPULATE	0x40	/* fault in pages (with FOLL_MLOCK) */ | ||||||
|  | #define FOLL_NOFAULT	0x80	/* do not fault in pages */ | ||||||
| #define FOLL_HWPOISON	0x100	/* check page is hwpoisoned */ | #define FOLL_HWPOISON	0x100	/* check page is hwpoisoned */ | ||||||
| #define FOLL_NUMA	0x200	/* force NUMA hinting page fault */ | #define FOLL_NUMA	0x200	/* force NUMA hinting page fault */ | ||||||
| #define FOLL_MIGRATION	0x400	/* wait for page to replace migration entry */ | #define FOLL_MIGRATION	0x400	/* wait for page to replace migration entry */ | ||||||
|  | |||||||
| @ -824,61 +824,11 @@ int folio_wait_private_2_killable(struct folio *folio); | |||||||
| void folio_add_wait_queue(struct folio *folio, wait_queue_entry_t *waiter); | void folio_add_wait_queue(struct folio *folio, wait_queue_entry_t *waiter); | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Fault everything in given userspace address range in. |  * Fault in userspace address range. | ||||||
|  */ |  */ | ||||||
| static inline int fault_in_pages_writeable(char __user *uaddr, size_t size) | size_t fault_in_writeable(char __user *uaddr, size_t size); | ||||||
| { | size_t fault_in_safe_writeable(const char __user *uaddr, size_t size); | ||||||
| 	char __user *end = uaddr + size - 1; | size_t fault_in_readable(const char __user *uaddr, size_t size); | ||||||
| 
 |  | ||||||
| 	if (unlikely(size == 0)) |  | ||||||
| 		return 0; |  | ||||||
| 
 |  | ||||||
| 	if (unlikely(uaddr > end)) |  | ||||||
| 		return -EFAULT; |  | ||||||
| 	/*
 |  | ||||||
| 	 * Writing zeroes into userspace here is OK, because we know that if |  | ||||||
| 	 * the zero gets there, we'll be overwriting it. |  | ||||||
| 	 */ |  | ||||||
| 	do { |  | ||||||
| 		if (unlikely(__put_user(0, uaddr) != 0)) |  | ||||||
| 			return -EFAULT; |  | ||||||
| 		uaddr += PAGE_SIZE; |  | ||||||
| 	} while (uaddr <= end); |  | ||||||
| 
 |  | ||||||
| 	/* Check whether the range spilled into the next page. */ |  | ||||||
| 	if (((unsigned long)uaddr & PAGE_MASK) == |  | ||||||
| 			((unsigned long)end & PAGE_MASK)) |  | ||||||
| 		return __put_user(0, end); |  | ||||||
| 
 |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static inline int fault_in_pages_readable(const char __user *uaddr, size_t size) |  | ||||||
| { |  | ||||||
| 	volatile char c; |  | ||||||
| 	const char __user *end = uaddr + size - 1; |  | ||||||
| 
 |  | ||||||
| 	if (unlikely(size == 0)) |  | ||||||
| 		return 0; |  | ||||||
| 
 |  | ||||||
| 	if (unlikely(uaddr > end)) |  | ||||||
| 		return -EFAULT; |  | ||||||
| 
 |  | ||||||
| 	do { |  | ||||||
| 		if (unlikely(__get_user(c, uaddr) != 0)) |  | ||||||
| 			return -EFAULT; |  | ||||||
| 		uaddr += PAGE_SIZE; |  | ||||||
| 	} while (uaddr <= end); |  | ||||||
| 
 |  | ||||||
| 	/* Check whether the range spilled into the next page. */ |  | ||||||
| 	if (((unsigned long)uaddr & PAGE_MASK) == |  | ||||||
| 			((unsigned long)end & PAGE_MASK)) { |  | ||||||
| 		return __get_user(c, end); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	(void)c; |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| int add_to_page_cache_locked(struct page *page, struct address_space *mapping, | int add_to_page_cache_locked(struct page *page, struct address_space *mapping, | ||||||
| 		pgoff_t index, gfp_t gfp); | 		pgoff_t index, gfp_t gfp); | ||||||
|  | |||||||
| @ -35,6 +35,7 @@ struct iov_iter_state { | |||||||
| 
 | 
 | ||||||
| struct iov_iter { | struct iov_iter { | ||||||
| 	u8 iter_type; | 	u8 iter_type; | ||||||
|  | 	bool nofault; | ||||||
| 	bool data_source; | 	bool data_source; | ||||||
| 	size_t iov_offset; | 	size_t iov_offset; | ||||||
| 	size_t count; | 	size_t count; | ||||||
| @ -133,7 +134,8 @@ size_t copy_page_from_iter_atomic(struct page *page, unsigned offset, | |||||||
| 				  size_t bytes, struct iov_iter *i); | 				  size_t bytes, struct iov_iter *i); | ||||||
| void iov_iter_advance(struct iov_iter *i, size_t bytes); | void iov_iter_advance(struct iov_iter *i, size_t bytes); | ||||||
| void iov_iter_revert(struct iov_iter *i, size_t bytes); | void iov_iter_revert(struct iov_iter *i, size_t bytes); | ||||||
| int iov_iter_fault_in_readable(const struct iov_iter *i, size_t bytes); | size_t fault_in_iov_iter_readable(const struct iov_iter *i, size_t bytes); | ||||||
|  | size_t fault_in_iov_iter_writeable(const struct iov_iter *i, size_t bytes); | ||||||
| size_t iov_iter_single_seg_count(const struct iov_iter *i); | size_t iov_iter_single_seg_count(const struct iov_iter *i); | ||||||
| size_t copy_page_to_iter(struct page *page, size_t offset, size_t bytes, | size_t copy_page_to_iter(struct page *page, size_t offset, size_t bytes, | ||||||
| 			 struct iov_iter *i); | 			 struct iov_iter *i); | ||||||
|  | |||||||
							
								
								
									
										105
									
								
								lib/iov_iter.c
									
									
									
									
									
								
							
							
						
						
									
										105
									
								
								lib/iov_iter.c
									
									
									
									
									
								
							| @ -191,7 +191,7 @@ static size_t copy_page_to_iter_iovec(struct page *page, size_t offset, size_t b | |||||||
| 	buf = iov->iov_base + skip; | 	buf = iov->iov_base + skip; | ||||||
| 	copy = min(bytes, iov->iov_len - skip); | 	copy = min(bytes, iov->iov_len - skip); | ||||||
| 
 | 
 | ||||||
| 	if (IS_ENABLED(CONFIG_HIGHMEM) && !fault_in_pages_writeable(buf, copy)) { | 	if (IS_ENABLED(CONFIG_HIGHMEM) && !fault_in_writeable(buf, copy)) { | ||||||
| 		kaddr = kmap_atomic(page); | 		kaddr = kmap_atomic(page); | ||||||
| 		from = kaddr + offset; | 		from = kaddr + offset; | ||||||
| 
 | 
 | ||||||
| @ -275,7 +275,7 @@ static size_t copy_page_from_iter_iovec(struct page *page, size_t offset, size_t | |||||||
| 	buf = iov->iov_base + skip; | 	buf = iov->iov_base + skip; | ||||||
| 	copy = min(bytes, iov->iov_len - skip); | 	copy = min(bytes, iov->iov_len - skip); | ||||||
| 
 | 
 | ||||||
| 	if (IS_ENABLED(CONFIG_HIGHMEM) && !fault_in_pages_readable(buf, copy)) { | 	if (IS_ENABLED(CONFIG_HIGHMEM) && !fault_in_readable(buf, copy)) { | ||||||
| 		kaddr = kmap_atomic(page); | 		kaddr = kmap_atomic(page); | ||||||
| 		to = kaddr + offset; | 		to = kaddr + offset; | ||||||
| 
 | 
 | ||||||
| @ -430,35 +430,81 @@ out: | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Fault in one or more iovecs of the given iov_iter, to a maximum length of |  * fault_in_iov_iter_readable - fault in iov iterator for reading | ||||||
|  * bytes.  For each iovec, fault in each page that constitutes the iovec. |  * @i: iterator | ||||||
|  |  * @size: maximum length | ||||||
|  * |  * | ||||||
|  * Return 0 on success, or non-zero if the memory could not be accessed (i.e. |  * Fault in one or more iovecs of the given iov_iter, to a maximum length of | ||||||
|  * because it is an invalid address). |  * @size.  For each iovec, fault in each page that constitutes the iovec. | ||||||
|  |  * | ||||||
|  |  * Returns the number of bytes not faulted in (like copy_to_user() and | ||||||
|  |  * copy_from_user()). | ||||||
|  |  * | ||||||
|  |  * Always returns 0 for non-userspace iterators. | ||||||
|  */ |  */ | ||||||
| int iov_iter_fault_in_readable(const struct iov_iter *i, size_t bytes) | size_t fault_in_iov_iter_readable(const struct iov_iter *i, size_t size) | ||||||
| { | { | ||||||
| 	if (iter_is_iovec(i)) { | 	if (iter_is_iovec(i)) { | ||||||
|  | 		size_t count = min(size, iov_iter_count(i)); | ||||||
| 		const struct iovec *p; | 		const struct iovec *p; | ||||||
| 		size_t skip; | 		size_t skip; | ||||||
| 
 | 
 | ||||||
| 		if (bytes > i->count) | 		size -= count; | ||||||
| 			bytes = i->count; | 		for (p = i->iov, skip = i->iov_offset; count; p++, skip = 0) { | ||||||
| 		for (p = i->iov, skip = i->iov_offset; bytes; p++, skip = 0) { | 			size_t len = min(count, p->iov_len - skip); | ||||||
| 			size_t len = min(bytes, p->iov_len - skip); | 			size_t ret; | ||||||
| 			int err; |  | ||||||
| 
 | 
 | ||||||
| 			if (unlikely(!len)) | 			if (unlikely(!len)) | ||||||
| 				continue; | 				continue; | ||||||
| 			err = fault_in_pages_readable(p->iov_base + skip, len); | 			ret = fault_in_readable(p->iov_base + skip, len); | ||||||
| 			if (unlikely(err)) | 			count -= len - ret; | ||||||
| 				return err; | 			if (ret) | ||||||
| 			bytes -= len; | 				break; | ||||||
| 		} | 		} | ||||||
|  | 		return count + size; | ||||||
| 	} | 	} | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| EXPORT_SYMBOL(iov_iter_fault_in_readable); | EXPORT_SYMBOL(fault_in_iov_iter_readable); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * fault_in_iov_iter_writeable - fault in iov iterator for writing | ||||||
|  |  * @i: iterator | ||||||
|  |  * @size: maximum length | ||||||
|  |  * | ||||||
|  |  * Faults in the iterator using get_user_pages(), i.e., without triggering | ||||||
|  |  * hardware page faults.  This is primarily useful when we already know that | ||||||
|  |  * some or all of the pages in @i aren't in memory. | ||||||
|  |  * | ||||||
|  |  * Returns the number of bytes not faulted in, like copy_to_user() and | ||||||
|  |  * copy_from_user(). | ||||||
|  |  * | ||||||
|  |  * Always returns 0 for non-user-space iterators. | ||||||
|  |  */ | ||||||
|  | size_t fault_in_iov_iter_writeable(const struct iov_iter *i, size_t size) | ||||||
|  | { | ||||||
|  | 	if (iter_is_iovec(i)) { | ||||||
|  | 		size_t count = min(size, iov_iter_count(i)); | ||||||
|  | 		const struct iovec *p; | ||||||
|  | 		size_t skip; | ||||||
|  | 
 | ||||||
|  | 		size -= count; | ||||||
|  | 		for (p = i->iov, skip = i->iov_offset; count; p++, skip = 0) { | ||||||
|  | 			size_t len = min(count, p->iov_len - skip); | ||||||
|  | 			size_t ret; | ||||||
|  | 
 | ||||||
|  | 			if (unlikely(!len)) | ||||||
|  | 				continue; | ||||||
|  | 			ret = fault_in_safe_writeable(p->iov_base + skip, len); | ||||||
|  | 			count -= len - ret; | ||||||
|  | 			if (ret) | ||||||
|  | 				break; | ||||||
|  | 		} | ||||||
|  | 		return count + size; | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(fault_in_iov_iter_writeable); | ||||||
| 
 | 
 | ||||||
| void iov_iter_init(struct iov_iter *i, unsigned int direction, | void iov_iter_init(struct iov_iter *i, unsigned int direction, | ||||||
| 			const struct iovec *iov, unsigned long nr_segs, | 			const struct iovec *iov, unsigned long nr_segs, | ||||||
| @ -467,6 +513,7 @@ void iov_iter_init(struct iov_iter *i, unsigned int direction, | |||||||
| 	WARN_ON(direction & ~(READ | WRITE)); | 	WARN_ON(direction & ~(READ | WRITE)); | ||||||
| 	*i = (struct iov_iter) { | 	*i = (struct iov_iter) { | ||||||
| 		.iter_type = ITER_IOVEC, | 		.iter_type = ITER_IOVEC, | ||||||
|  | 		.nofault = false, | ||||||
| 		.data_source = direction, | 		.data_source = direction, | ||||||
| 		.iov = iov, | 		.iov = iov, | ||||||
| 		.nr_segs = nr_segs, | 		.nr_segs = nr_segs, | ||||||
| @ -1481,14 +1528,18 @@ ssize_t iov_iter_get_pages(struct iov_iter *i, | |||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	if (likely(iter_is_iovec(i))) { | 	if (likely(iter_is_iovec(i))) { | ||||||
|  | 		unsigned int gup_flags = 0; | ||||||
| 		unsigned long addr; | 		unsigned long addr; | ||||||
| 
 | 
 | ||||||
|  | 		if (iov_iter_rw(i) != WRITE) | ||||||
|  | 			gup_flags |= FOLL_WRITE; | ||||||
|  | 		if (i->nofault) | ||||||
|  | 			gup_flags |= FOLL_NOFAULT; | ||||||
|  | 
 | ||||||
| 		addr = first_iovec_segment(i, &len, start, maxsize, maxpages); | 		addr = first_iovec_segment(i, &len, start, maxsize, maxpages); | ||||||
| 		n = DIV_ROUND_UP(len, PAGE_SIZE); | 		n = DIV_ROUND_UP(len, PAGE_SIZE); | ||||||
| 		res = get_user_pages_fast(addr, n, | 		res = get_user_pages_fast(addr, n, gup_flags, pages); | ||||||
| 				iov_iter_rw(i) != WRITE ?  FOLL_WRITE : 0, | 		if (unlikely(res <= 0)) | ||||||
| 				pages); |  | ||||||
| 		if (unlikely(res < 0)) |  | ||||||
| 			return res; | 			return res; | ||||||
| 		return (res == n ? len : res * PAGE_SIZE) - *start; | 		return (res == n ? len : res * PAGE_SIZE) - *start; | ||||||
| 	} | 	} | ||||||
| @ -1603,17 +1654,23 @@ ssize_t iov_iter_get_pages_alloc(struct iov_iter *i, | |||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	if (likely(iter_is_iovec(i))) { | 	if (likely(iter_is_iovec(i))) { | ||||||
|  | 		unsigned int gup_flags = 0; | ||||||
| 		unsigned long addr; | 		unsigned long addr; | ||||||
| 
 | 
 | ||||||
|  | 		if (iov_iter_rw(i) != WRITE) | ||||||
|  | 			gup_flags |= FOLL_WRITE; | ||||||
|  | 		if (i->nofault) | ||||||
|  | 			gup_flags |= FOLL_NOFAULT; | ||||||
|  | 
 | ||||||
| 		addr = first_iovec_segment(i, &len, start, maxsize, ~0U); | 		addr = first_iovec_segment(i, &len, start, maxsize, ~0U); | ||||||
| 		n = DIV_ROUND_UP(len, PAGE_SIZE); | 		n = DIV_ROUND_UP(len, PAGE_SIZE); | ||||||
| 		p = get_pages_array(n); | 		p = get_pages_array(n); | ||||||
| 		if (!p) | 		if (!p) | ||||||
| 			return -ENOMEM; | 			return -ENOMEM; | ||||||
| 		res = get_user_pages_fast(addr, n, | 		res = get_user_pages_fast(addr, n, gup_flags, p); | ||||||
| 				iov_iter_rw(i) != WRITE ?  FOLL_WRITE : 0, p); | 		if (unlikely(res <= 0)) { | ||||||
| 		if (unlikely(res < 0)) { |  | ||||||
| 			kvfree(p); | 			kvfree(p); | ||||||
|  | 			*pages = NULL; | ||||||
| 			return res; | 			return res; | ||||||
| 		} | 		} | ||||||
| 		*pages = p; | 		*pages = p; | ||||||
|  | |||||||
| @ -89,7 +89,7 @@ | |||||||
|  *      ->lock_page		(filemap_fault, access_process_vm) |  *      ->lock_page		(filemap_fault, access_process_vm) | ||||||
|  * |  * | ||||||
|  *  ->i_rwsem			(generic_perform_write) |  *  ->i_rwsem			(generic_perform_write) | ||||||
|  *    ->mmap_lock		(fault_in_pages_readable->do_page_fault) |  *    ->mmap_lock		(fault_in_readable->do_page_fault) | ||||||
|  * |  * | ||||||
|  *  bdi->wb.list_lock |  *  bdi->wb.list_lock | ||||||
|  *    sb_lock			(fs/fs-writeback.c) |  *    sb_lock			(fs/fs-writeback.c) | ||||||
| @ -3733,7 +3733,7 @@ again: | |||||||
| 		 * same page as we're writing to, without it being marked | 		 * same page as we're writing to, without it being marked | ||||||
| 		 * up-to-date. | 		 * up-to-date. | ||||||
| 		 */ | 		 */ | ||||||
| 		if (unlikely(iov_iter_fault_in_readable(i, bytes))) { | 		if (unlikely(fault_in_iov_iter_readable(i, bytes))) { | ||||||
| 			status = -EFAULT; | 			status = -EFAULT; | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
|  | |||||||
							
								
								
									
										139
									
								
								mm/gup.c
									
									
									
									
									
								
							
							
						
						
									
										139
									
								
								mm/gup.c
									
									
									
									
									
								
							| @ -918,6 +918,8 @@ static int faultin_page(struct vm_area_struct *vma, | |||||||
| 	/* mlock all present pages, but do not fault in new pages */ | 	/* mlock all present pages, but do not fault in new pages */ | ||||||
| 	if ((*flags & (FOLL_POPULATE | FOLL_MLOCK)) == FOLL_MLOCK) | 	if ((*flags & (FOLL_POPULATE | FOLL_MLOCK)) == FOLL_MLOCK) | ||||||
| 		return -ENOENT; | 		return -ENOENT; | ||||||
|  | 	if (*flags & FOLL_NOFAULT) | ||||||
|  | 		return -EFAULT; | ||||||
| 	if (*flags & FOLL_WRITE) | 	if (*flags & FOLL_WRITE) | ||||||
| 		fault_flags |= FAULT_FLAG_WRITE; | 		fault_flags |= FAULT_FLAG_WRITE; | ||||||
| 	if (*flags & FOLL_REMOTE) | 	if (*flags & FOLL_REMOTE) | ||||||
| @ -1656,6 +1658,141 @@ finish_or_fault: | |||||||
| } | } | ||||||
| #endif /* !CONFIG_MMU */ | #endif /* !CONFIG_MMU */ | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * fault_in_writeable - fault in userspace address range for writing | ||||||
|  |  * @uaddr: start of address range | ||||||
|  |  * @size: size of address range | ||||||
|  |  * | ||||||
|  |  * Returns the number of bytes not faulted in (like copy_to_user() and | ||||||
|  |  * copy_from_user()). | ||||||
|  |  */ | ||||||
|  | size_t fault_in_writeable(char __user *uaddr, size_t size) | ||||||
|  | { | ||||||
|  | 	char __user *start = uaddr, *end; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(size == 0)) | ||||||
|  | 		return 0; | ||||||
|  | 	if (!PAGE_ALIGNED(uaddr)) { | ||||||
|  | 		if (unlikely(__put_user(0, uaddr) != 0)) | ||||||
|  | 			return size; | ||||||
|  | 		uaddr = (char __user *)PAGE_ALIGN((unsigned long)uaddr); | ||||||
|  | 	} | ||||||
|  | 	end = (char __user *)PAGE_ALIGN((unsigned long)start + size); | ||||||
|  | 	if (unlikely(end < start)) | ||||||
|  | 		end = NULL; | ||||||
|  | 	while (uaddr != end) { | ||||||
|  | 		if (unlikely(__put_user(0, uaddr) != 0)) | ||||||
|  | 			goto out; | ||||||
|  | 		uaddr += PAGE_SIZE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | out: | ||||||
|  | 	if (size > uaddr - start) | ||||||
|  | 		return size - (uaddr - start); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(fault_in_writeable); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * fault_in_safe_writeable - fault in an address range for writing | ||||||
|  |  * @uaddr: start of address range | ||||||
|  |  * @size: length of address range | ||||||
|  |  * | ||||||
|  |  * Faults in an address range using get_user_pages, i.e., without triggering | ||||||
|  |  * hardware page faults.  This is primarily useful when we already know that | ||||||
|  |  * some or all of the pages in the address range aren't in memory. | ||||||
|  |  * | ||||||
|  |  * Other than fault_in_writeable(), this function is non-destructive. | ||||||
|  |  * | ||||||
|  |  * Note that we don't pin or otherwise hold the pages referenced that we fault | ||||||
|  |  * in.  There's no guarantee that they'll stay in memory for any duration of | ||||||
|  |  * time. | ||||||
|  |  * | ||||||
|  |  * Returns the number of bytes not faulted in, like copy_to_user() and | ||||||
|  |  * copy_from_user(). | ||||||
|  |  */ | ||||||
|  | size_t fault_in_safe_writeable(const char __user *uaddr, size_t size) | ||||||
|  | { | ||||||
|  | 	unsigned long start = (unsigned long)untagged_addr(uaddr); | ||||||
|  | 	unsigned long end, nstart, nend; | ||||||
|  | 	struct mm_struct *mm = current->mm; | ||||||
|  | 	struct vm_area_struct *vma = NULL; | ||||||
|  | 	int locked = 0; | ||||||
|  | 
 | ||||||
|  | 	nstart = start & PAGE_MASK; | ||||||
|  | 	end = PAGE_ALIGN(start + size); | ||||||
|  | 	if (end < nstart) | ||||||
|  | 		end = 0; | ||||||
|  | 	for (; nstart != end; nstart = nend) { | ||||||
|  | 		unsigned long nr_pages; | ||||||
|  | 		long ret; | ||||||
|  | 
 | ||||||
|  | 		if (!locked) { | ||||||
|  | 			locked = 1; | ||||||
|  | 			mmap_read_lock(mm); | ||||||
|  | 			vma = find_vma(mm, nstart); | ||||||
|  | 		} else if (nstart >= vma->vm_end) | ||||||
|  | 			vma = vma->vm_next; | ||||||
|  | 		if (!vma || vma->vm_start >= end) | ||||||
|  | 			break; | ||||||
|  | 		nend = end ? min(end, vma->vm_end) : vma->vm_end; | ||||||
|  | 		if (vma->vm_flags & (VM_IO | VM_PFNMAP)) | ||||||
|  | 			continue; | ||||||
|  | 		if (nstart < vma->vm_start) | ||||||
|  | 			nstart = vma->vm_start; | ||||||
|  | 		nr_pages = (nend - nstart) / PAGE_SIZE; | ||||||
|  | 		ret = __get_user_pages_locked(mm, nstart, nr_pages, | ||||||
|  | 					      NULL, NULL, &locked, | ||||||
|  | 					      FOLL_TOUCH | FOLL_WRITE); | ||||||
|  | 		if (ret <= 0) | ||||||
|  | 			break; | ||||||
|  | 		nend = nstart + ret * PAGE_SIZE; | ||||||
|  | 	} | ||||||
|  | 	if (locked) | ||||||
|  | 		mmap_read_unlock(mm); | ||||||
|  | 	if (nstart == end) | ||||||
|  | 		return 0; | ||||||
|  | 	return size - min_t(size_t, nstart - start, size); | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(fault_in_safe_writeable); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * fault_in_readable - fault in userspace address range for reading | ||||||
|  |  * @uaddr: start of user address range | ||||||
|  |  * @size: size of user address range | ||||||
|  |  * | ||||||
|  |  * Returns the number of bytes not faulted in (like copy_to_user() and | ||||||
|  |  * copy_from_user()). | ||||||
|  |  */ | ||||||
|  | size_t fault_in_readable(const char __user *uaddr, size_t size) | ||||||
|  | { | ||||||
|  | 	const char __user *start = uaddr, *end; | ||||||
|  | 	volatile char c; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(size == 0)) | ||||||
|  | 		return 0; | ||||||
|  | 	if (!PAGE_ALIGNED(uaddr)) { | ||||||
|  | 		if (unlikely(__get_user(c, uaddr) != 0)) | ||||||
|  | 			return size; | ||||||
|  | 		uaddr = (const char __user *)PAGE_ALIGN((unsigned long)uaddr); | ||||||
|  | 	} | ||||||
|  | 	end = (const char __user *)PAGE_ALIGN((unsigned long)start + size); | ||||||
|  | 	if (unlikely(end < start)) | ||||||
|  | 		end = NULL; | ||||||
|  | 	while (uaddr != end) { | ||||||
|  | 		if (unlikely(__get_user(c, uaddr) != 0)) | ||||||
|  | 			goto out; | ||||||
|  | 		uaddr += PAGE_SIZE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | out: | ||||||
|  | 	(void)c; | ||||||
|  | 	if (size > uaddr - start) | ||||||
|  | 		return size - (uaddr - start); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(fault_in_readable); | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * get_dump_page() - pin user page in memory while writing it to core dump |  * get_dump_page() - pin user page in memory while writing it to core dump | ||||||
|  * @addr: user address |  * @addr: user address | ||||||
| @ -2708,7 +2845,7 @@ static int internal_get_user_pages_fast(unsigned long start, | |||||||
| 
 | 
 | ||||||
| 	if (WARN_ON_ONCE(gup_flags & ~(FOLL_WRITE | FOLL_LONGTERM | | 	if (WARN_ON_ONCE(gup_flags & ~(FOLL_WRITE | FOLL_LONGTERM | | ||||||
| 				       FOLL_FORCE | FOLL_PIN | FOLL_GET | | 				       FOLL_FORCE | FOLL_PIN | FOLL_GET | | ||||||
| 				       FOLL_FAST_ONLY))) | 				       FOLL_FAST_ONLY | FOLL_NOFAULT))) | ||||||
| 		return -EINVAL; | 		return -EINVAL; | ||||||
| 
 | 
 | ||||||
| 	if (gup_flags & FOLL_PIN) | 	if (gup_flags & FOLL_PIN) | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Linus Torvalds
						Linus Torvalds