254 lines
5.4 KiB
C
254 lines
5.4 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (c) 2018 Matthew Bobrowski. All Rights Reserved.
|
|
*
|
|
* Started by Matthew Bobrowski <mbobrowski@mbobrowski.org>
|
|
*/
|
|
|
|
/*\
|
|
* [Description]
|
|
* Validate that the newly introduced FAN_OPEN_EXEC mask functions as expected.
|
|
* The idea is to generate a sequence of open related actions to ensure that
|
|
* the correct event flags are being set depending on what event mask was
|
|
* requested when the object was marked.
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#include "config.h"
|
|
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
#include "tst_test.h"
|
|
|
|
#ifdef HAVE_SYS_FANOTIFY_H
|
|
#include "fanotify.h"
|
|
|
|
#define EVENT_MAX 1024
|
|
#define EVENT_SIZE (sizeof (struct fanotify_event_metadata))
|
|
#define EVENT_BUF_LEN (EVENT_MAX * EVENT_SIZE)
|
|
#define EVENT_SET_BUF 32
|
|
|
|
#define BUF_SIZE 256
|
|
#define TEST_APP "fanotify_child"
|
|
|
|
static pid_t child_pid;
|
|
static char fname[BUF_SIZE];
|
|
static volatile int fd_notify;
|
|
static volatile int complete;
|
|
static char event_buf[EVENT_BUF_LEN];
|
|
static int exec_events_unsupported;
|
|
|
|
static struct test_case_t {
|
|
const char *tname;
|
|
struct fanotify_mark_type mark;
|
|
unsigned long long mask;
|
|
unsigned long long ignore_mask;
|
|
int event_count;
|
|
unsigned long long event_set[EVENT_SET_BUF];
|
|
} test_cases[] = {
|
|
{
|
|
"inode mark, FAN_OPEN events",
|
|
INIT_FANOTIFY_MARK_TYPE(INODE),
|
|
FAN_OPEN,
|
|
0,
|
|
2,
|
|
{FAN_OPEN, FAN_OPEN}
|
|
},
|
|
{
|
|
"inode mark, FAN_OPEN_EXEC events",
|
|
INIT_FANOTIFY_MARK_TYPE(INODE),
|
|
FAN_OPEN_EXEC,
|
|
0,
|
|
1,
|
|
{FAN_OPEN_EXEC}
|
|
},
|
|
{
|
|
"inode mark, FAN_OPEN | FAN_OPEN_EXEC events",
|
|
INIT_FANOTIFY_MARK_TYPE(INODE),
|
|
FAN_OPEN | FAN_OPEN_EXEC,
|
|
0,
|
|
2,
|
|
{FAN_OPEN, FAN_OPEN | FAN_OPEN_EXEC}
|
|
},
|
|
{
|
|
"inode mark, FAN_OPEN events, ignore FAN_OPEN_EXEC",
|
|
INIT_FANOTIFY_MARK_TYPE(INODE),
|
|
FAN_OPEN,
|
|
FAN_OPEN_EXEC,
|
|
2,
|
|
{FAN_OPEN, FAN_OPEN}
|
|
},
|
|
{
|
|
"inode mark, FAN_OPEN_EXEC events, ignore FAN_OPEN",
|
|
INIT_FANOTIFY_MARK_TYPE(INODE),
|
|
FAN_OPEN_EXEC,
|
|
FAN_OPEN,
|
|
1,
|
|
{FAN_OPEN_EXEC}
|
|
},
|
|
{
|
|
"inode mark, FAN_OPEN | FAN_OPEN_EXEC events, ignore "
|
|
"FAN_OPEN_EXEC",
|
|
INIT_FANOTIFY_MARK_TYPE(INODE),
|
|
FAN_OPEN | FAN_OPEN_EXEC,
|
|
FAN_OPEN_EXEC,
|
|
2,
|
|
{FAN_OPEN, FAN_OPEN}
|
|
}
|
|
};
|
|
|
|
static int generate_events(void)
|
|
{
|
|
int fd, status;
|
|
|
|
child_pid = SAFE_FORK();
|
|
|
|
/*
|
|
* Generate a sequence of events
|
|
*/
|
|
if (child_pid == 0) {
|
|
SAFE_CLOSE(fd_notify);
|
|
|
|
fd = SAFE_OPEN(fname, O_RDONLY);
|
|
|
|
if (fd > 0)
|
|
SAFE_CLOSE(fd);
|
|
|
|
SAFE_EXECL(TEST_APP, TEST_APP, NULL);
|
|
exit(1);
|
|
}
|
|
|
|
SAFE_WAITPID(child_pid, &status, 0);
|
|
|
|
if (WIFEXITED(status) && WEXITSTATUS(status) == 0)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
static int setup_mark(unsigned int n)
|
|
{
|
|
unsigned int i = 0;
|
|
struct test_case_t *tc = &test_cases[n];
|
|
struct fanotify_mark_type *mark = &tc->mark;
|
|
const char *const files[] = {fname, TEST_APP};
|
|
|
|
tst_res(TINFO, "Test #%d: %s", n, tc->tname);
|
|
|
|
if (exec_events_unsupported && ((tc->mask & FAN_OPEN_EXEC) ||
|
|
tc->ignore_mask & FAN_OPEN_EXEC)) {
|
|
tst_res(TCONF, "FAN_OPEN_EXEC not supported in kernel?");
|
|
return -1;
|
|
}
|
|
|
|
fd_notify = SAFE_FANOTIFY_INIT(FAN_CLASS_NOTIF, O_RDONLY);
|
|
|
|
for (; i < ARRAY_SIZE(files); i++) {
|
|
/* Setup normal mark on object */
|
|
SAFE_FANOTIFY_MARK(fd_notify, FAN_MARK_ADD | mark->flag,
|
|
tc->mask, AT_FDCWD, files[i]);
|
|
|
|
/* Setup ignore mark on object */
|
|
if (tc->ignore_mask) {
|
|
SAFE_FANOTIFY_MARK(fd_notify, FAN_MARK_ADD | mark->flag
|
|
| FAN_MARK_IGNORED_MASK,
|
|
tc->ignore_mask, AT_FDCWD,
|
|
files[i]);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void do_test(unsigned int n)
|
|
{
|
|
int len = 0, event_num = 0;
|
|
struct test_case_t *tc = &test_cases[n];
|
|
struct fanotify_event_metadata *event;
|
|
|
|
/* Place a mark on the object */
|
|
if (setup_mark(n) != 0)
|
|
goto cleanup;
|
|
|
|
/* Generate events in child process */
|
|
if (!generate_events())
|
|
goto cleanup;
|
|
|
|
/* Read available events into buffer */
|
|
len = SAFE_READ(0, fd_notify, event_buf, EVENT_BUF_LEN);
|
|
|
|
event = (struct fanotify_event_metadata *) event_buf;
|
|
|
|
/* Process events */
|
|
while (FAN_EVENT_OK(event, len) && event_num < tc->event_count) {
|
|
if (event->mask != *(tc->event_set + event_num)) {
|
|
tst_res(TFAIL,
|
|
"Received event: mask=%llx (expected %llx, "
|
|
"pid=%u, fd=%d",
|
|
(unsigned long long) event->mask,
|
|
*(tc->event_set + event_num),
|
|
(unsigned) event->pid,
|
|
event->fd);
|
|
} else if (event->pid != child_pid) {
|
|
tst_res(TFAIL,
|
|
"Received event: mask=%llx, pid=%u (expected "
|
|
"%u), fd=%d",
|
|
(unsigned long long) event->mask,
|
|
(unsigned) event->pid,
|
|
(unsigned) child_pid,
|
|
event->fd);
|
|
} else {
|
|
tst_res(TPASS,
|
|
"Received event: mask=%llx, pid=%u, fd=%d",
|
|
(unsigned long long) event->mask,
|
|
(unsigned) event->pid,
|
|
event->fd);
|
|
}
|
|
|
|
if (event->fd != FAN_NOFD)
|
|
SAFE_CLOSE(event->fd);
|
|
|
|
event_num++;
|
|
event = FAN_EVENT_NEXT(event, len);
|
|
}
|
|
|
|
cleanup:
|
|
if (fd_notify > 0)
|
|
SAFE_CLOSE(fd_notify);
|
|
}
|
|
|
|
static void do_setup(void)
|
|
{
|
|
exec_events_unsupported = fanotify_events_supported_by_kernel(FAN_OPEN_EXEC);
|
|
|
|
sprintf(fname, "fname_%d", getpid());
|
|
SAFE_FILE_PRINTF(fname, "1");
|
|
}
|
|
|
|
static void do_cleanup(void)
|
|
{
|
|
if (fd_notify > 0)
|
|
SAFE_CLOSE(fd_notify);
|
|
}
|
|
|
|
static const char *const resource_files[] = {
|
|
TEST_APP,
|
|
NULL
|
|
};
|
|
|
|
static struct tst_test test = {
|
|
.setup = do_setup,
|
|
.test = do_test,
|
|
.tcnt = ARRAY_SIZE(test_cases),
|
|
.cleanup = do_cleanup,
|
|
.forks_child = 1,
|
|
.needs_root = 1,
|
|
.resource_files = resource_files
|
|
};
|
|
#else
|
|
TST_TEST_TCONF("System does not contain required fanotify support");
|
|
#endif
|