// SPDX-License-Identifier: GPL-2.0 #define _GNU_SOURCE #include #include #include #include #include #include #include #include #include #include #include #include "../kselftest_harness.h" static const struct { struct nlmsghdr nlhdr; struct genlmsghdr genlhdr; struct nlattr ahdr; __u16 val; __u16 pad; } dump_policies = { .nlhdr = { .nlmsg_len = sizeof(dump_policies), .nlmsg_type = GENL_ID_CTRL, .nlmsg_flags = NLM_F_REQUEST | NLM_F_ACK | NLM_F_DUMP, .nlmsg_seq = 1, }, .genlhdr = { .cmd = CTRL_CMD_GETPOLICY, .version = 2, }, .ahdr = { .nla_len = 6, .nla_type = CTRL_ATTR_FAMILY_ID, }, .val = GENL_ID_CTRL, .pad = 0, }; // Sanity check for the test itself, make sure the dump doesn't fit in one msg TEST(test_sanity) { int netlink_sock; char buf[8192]; ssize_t n; netlink_sock = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC); ASSERT_GE(netlink_sock, 0); n = send(netlink_sock, &dump_policies, sizeof(dump_policies), 0); ASSERT_EQ(n, sizeof(dump_policies)); n = recv(netlink_sock, buf, sizeof(buf), MSG_DONTWAIT); ASSERT_GE(n, sizeof(struct nlmsghdr)); n = recv(netlink_sock, buf, sizeof(buf), MSG_DONTWAIT); ASSERT_GE(n, sizeof(struct nlmsghdr)); close(netlink_sock); } TEST(close_in_progress) { int netlink_sock; ssize_t n; netlink_sock = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC); ASSERT_GE(netlink_sock, 0); n = send(netlink_sock, &dump_policies, sizeof(dump_policies), 0); ASSERT_EQ(n, sizeof(dump_policies)); close(netlink_sock); } TEST(close_with_ref) { char cookie[NOTIFY_COOKIE_LEN] = {}; int netlink_sock, mq_fd; struct sigevent sigev; ssize_t n; netlink_sock = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC); ASSERT_GE(netlink_sock, 0); n = send(netlink_sock, &dump_policies, sizeof(dump_policies), 0); ASSERT_EQ(n, sizeof(dump_policies)); mq_fd = syscall(__NR_mq_open, "sed", O_CREAT | O_WRONLY, 0600, 0); ASSERT_GE(mq_fd, 0); memset(&sigev, 0, sizeof(sigev)); sigev.sigev_notify = SIGEV_THREAD; sigev.sigev_value.sival_ptr = cookie; sigev.sigev_signo = netlink_sock; syscall(__NR_mq_notify, mq_fd, &sigev); close(netlink_sock); // give mqueue time to fire usleep(100 * 1000); } TEST_HARNESS_MAIN