[LTP] [PATCH 3/3] fanotify24: add mmap() and user page fault to test

Jan Kara jack@suse.cz
Wed Mar 26 16:08:19 CET 2025


On Wed 19-03-25 20:27:42, Amir Goldstein wrote:
> Commit 066e053fe208a ("fsnotify: add pre-content hooks on mmap()")
> in 6.14-rc7 added a pre-content hooh in mmap() before the final release
> of the fsnotify pre-content event feature.
> 
> To test pre-content hook on mmap(), increase the size of the test file
> to 101 pages use mmap() to setup a buffer from the end of the test file.
> 
> Change some of the test cases to allow the write() and use this buffer
> as the input buffer to write(), which tests deadlock avoidance by
> suppresing pre-content hook on user page faults.
> 
> Signed-off-by: Amir Goldstein <amir73il@gmail.com>

Looks good. Feel free to add:

Reviewed-by: Jan Kara <jack@suse.cz>

								Honza

> ---
>  .../kernel/syscalls/fanotify/fanotify24.c     | 41 ++++++++++++++++---
>  1 file changed, 35 insertions(+), 6 deletions(-)
> 
> diff --git a/testcases/kernel/syscalls/fanotify/fanotify24.c b/testcases/kernel/syscalls/fanotify/fanotify24.c
> index 2d2479b39..3a76f2c1b 100644
> --- a/testcases/kernel/syscalls/fanotify/fanotify24.c
> +++ b/testcases/kernel/syscalls/fanotify/fanotify24.c
> @@ -46,6 +46,7 @@
>  static char fname[BUF_SIZE];
>  static char buf[BUF_SIZE];
>  static volatile int fd_notify;
> +static size_t page_sz;
>  
>  static pid_t child_pid;
>  
> @@ -68,7 +69,8 @@ static struct tcase {
>  		FAN_OPEN_PERM | FAN_PRE_ACCESS,
>  		{
>  			{FAN_OPEN_PERM, FAN_ALLOW},
> -			{FAN_PRE_ACCESS, FAN_DENY},
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
>  			{FAN_PRE_ACCESS, FAN_DENY_ERRNO(EIO)},
>  			{FAN_OPEN_PERM, FAN_DENY_ERRNO(EBUSY)}
>  		}
> @@ -78,6 +80,7 @@ static struct tcase {
>  		INIT_FANOTIFY_MARK_TYPE(INODE),
>  		FAN_PRE_ACCESS | FAN_OPEN_EXEC_PERM,
>  		{
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
>  			{FAN_PRE_ACCESS, FAN_DENY},
>  			{FAN_PRE_ACCESS, FAN_DENY_ERRNO(EIO)},
>  			{FAN_OPEN_EXEC_PERM, FAN_DENY_ERRNO(EBUSY)}
> @@ -89,7 +92,8 @@ static struct tcase {
>  		FAN_OPEN_PERM | FAN_PRE_ACCESS,
>  		{
>  			{FAN_OPEN_PERM, FAN_ALLOW},
> -			{FAN_PRE_ACCESS, FAN_DENY},
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
>  			{FAN_PRE_ACCESS, FAN_DENY_ERRNO(EIO)},
>  			{FAN_OPEN_PERM, FAN_DENY_ERRNO(EBUSY)}
>  		}
> @@ -99,6 +103,7 @@ static struct tcase {
>  		INIT_FANOTIFY_MARK_TYPE(MOUNT),
>  		FAN_PRE_ACCESS | FAN_OPEN_EXEC_PERM,
>  		{
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
>  			{FAN_PRE_ACCESS, FAN_DENY},
>  			{FAN_PRE_ACCESS, FAN_DENY_ERRNO(EIO)},
>  			{FAN_OPEN_EXEC_PERM, FAN_DENY_ERRNO(EBUSY)}
> @@ -110,7 +115,8 @@ static struct tcase {
>  		FAN_OPEN_PERM | FAN_PRE_ACCESS,
>  		{
>  			{FAN_OPEN_PERM, FAN_ALLOW},
> -			{FAN_PRE_ACCESS, FAN_DENY},
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
>  			{FAN_PRE_ACCESS, FAN_DENY_ERRNO(EIO)},
>  			{FAN_OPEN_PERM, FAN_DENY_ERRNO(EBUSY)}
>  		}
> @@ -120,6 +126,7 @@ static struct tcase {
>  		INIT_FANOTIFY_MARK_TYPE(FILESYSTEM),
>  		FAN_PRE_ACCESS | FAN_OPEN_EXEC_PERM,
>  		{
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
>  			{FAN_PRE_ACCESS, FAN_DENY},
>  			{FAN_PRE_ACCESS, FAN_DENY_ERRNO(EIO)},
>  			{FAN_OPEN_EXEC_PERM, FAN_DENY_ERRNO(EBUSY)}
> @@ -130,6 +137,7 @@ static struct tcase {
>  		INIT_FANOTIFY_MARK_TYPE(PARENT),
>  		FAN_PRE_ACCESS | FAN_OPEN_EXEC_PERM | FAN_EVENT_ON_CHILD,
>  		{
> +			{FAN_PRE_ACCESS, FAN_ALLOW},
>  			{FAN_PRE_ACCESS, FAN_DENY},
>  			{FAN_PRE_ACCESS, FAN_DENY},
>  			{FAN_OPEN_EXEC_PERM, FAN_DENY}
> @@ -169,6 +177,7 @@ static int expected_errno(unsigned int response)
>  static void generate_events(struct tcase *tc)
>  {
>  	int fd;
> +	char *addr;
>  	char *const argv[] = {FILE_EXEC_PATH, NULL};
>  	struct event *event = tc->event_set;
>  	int exp_ret, exp_errno = 0;
> @@ -186,8 +195,25 @@ static void generate_events(struct tcase *tc)
>  
>  	exp_ret = exp_errno ? -1 : 1;
>  	errno = 0;
> -	/* FAN_PRE_ACCESS events are reported also on write */
> -	if (write(fd, fname, 1) != exp_ret || errno != exp_errno) {
> +	/*
> +	 * FAN_PRE_ACCESS events are reported on map() and write(), but should
> +	 * not be reported when faulting in the user page at offset page_sz*100
> +	 * that is used as an input buffer to the write() syscall.
> +	 */
> +	addr = SAFE_MMAP(NULL, page_sz, PROT_READ, MAP_PRIVATE, fd, page_sz*100);
> +	if (!addr || errno != exp_errno) {
> +		tst_res(TFAIL, "mmap() got errno %d (expected %d)", errno, exp_errno);
> +		exit(3);
> +	} else if (errno == exp_errno) {
> +		tst_res(TINFO, "mmap() got errno %d as expected", errno);
> +	}
> +
> +	exp_errno = expected_errno(event->response);
> +	event++;
> +
> +	exp_ret = exp_errno ? -1 : 1;
> +	errno = 0;
> +	if (write(fd, addr, 1) != exp_ret || errno != exp_errno) {
>  		tst_res(TFAIL, "write() got errno %d (expected %d)", errno, exp_errno);
>  		exit(3);
>  	} else if (errno == exp_errno) {
> @@ -199,7 +225,7 @@ static void generate_events(struct tcase *tc)
>  	exp_errno = expected_errno(event->response);
>  	event++;
>  
> -	exp_ret = exp_errno ? -1 : 1;
> +	exp_ret = exp_errno ? -1 : BUF_SIZE;
>  	errno = 0;
>  	if (read(fd, buf, BUF_SIZE) != exp_ret || errno != exp_errno) {
>  		tst_res(TFAIL, "read() got errno %d (expected %d)", errno, exp_errno);
> @@ -449,8 +475,11 @@ static void test_fanotify(unsigned int n)
>  
>  static void setup(void)
>  {
> +	page_sz = getpagesize();
> +
>  	sprintf(fname, MOUNT_PATH"/fname_%d", getpid());
>  	SAFE_FILE_PRINTF(fname, "1");
> +	SAFE_TRUNCATE(fname, page_sz*101);
>  
>  	REQUIRE_FANOTIFY_EVENTS_SUPPORTED_ON_FS(FAN_CLASS_PRE_CONTENT, FAN_MARK_FILESYSTEM,
>  						FAN_PRE_ACCESS, fname);
> -- 
> 2.34.1
> 
-- 
Jan Kara <jack@suse.com>
SUSE Labs, CR


More information about the ltp mailing list