1f08c3bdfSopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 2f08c3bdfSopenharmony_ci/* 3f08c3bdfSopenharmony_ci * Copyright (c) Bull S.A.S. 2008 4f08c3bdfSopenharmony_ci * 01/12/08 Nadia Derbey <Nadia.Derbey@bull.net> 5f08c3bdfSopenharmony_ci * Copyright (C) 2023 SUSE LLC Andrea Cervesato <andrea.cervesato@suse.com> 6f08c3bdfSopenharmony_ci */ 7f08c3bdfSopenharmony_ci 8f08c3bdfSopenharmony_ci/*\ 9f08c3bdfSopenharmony_ci * [Description] 10f08c3bdfSopenharmony_ci * 11f08c3bdfSopenharmony_ci * Clone a process with CLONE_NEWPID flag, register notification on a posix 12f08c3bdfSopenharmony_ci * mqueue and send a mqueue message from the child. Then check if signal 13f08c3bdfSopenharmony_ci * notification contains si_pid of the child. 14f08c3bdfSopenharmony_ci */ 15f08c3bdfSopenharmony_ci 16f08c3bdfSopenharmony_ci#define _GNU_SOURCE 1 17f08c3bdfSopenharmony_ci#include <signal.h> 18f08c3bdfSopenharmony_ci#include <mqueue.h> 19f08c3bdfSopenharmony_ci#include "tst_test.h" 20f08c3bdfSopenharmony_ci#include "tst_safe_posix_ipc.h" 21f08c3bdfSopenharmony_ci#include "lapi/sched.h" 22f08c3bdfSopenharmony_ci 23f08c3bdfSopenharmony_ci#define MQNAME "/LTP_PIDNS30_MQ" 24f08c3bdfSopenharmony_ci 25f08c3bdfSopenharmony_cistatic mqd_t mqd = -1; 26f08c3bdfSopenharmony_cistatic volatile int received; 27f08c3bdfSopenharmony_cistatic siginfo_t info; 28f08c3bdfSopenharmony_ci 29f08c3bdfSopenharmony_cistatic void remove_mqueue(mqd_t mqd) 30f08c3bdfSopenharmony_ci{ 31f08c3bdfSopenharmony_ci if (mqd != -1) 32f08c3bdfSopenharmony_ci SAFE_MQ_CLOSE(mqd); 33f08c3bdfSopenharmony_ci 34f08c3bdfSopenharmony_ci mq_unlink(MQNAME); 35f08c3bdfSopenharmony_ci} 36f08c3bdfSopenharmony_ci 37f08c3bdfSopenharmony_cistatic void signal_handler(LTP_ATTRIBUTE_UNUSED int sig, siginfo_t *si, LTP_ATTRIBUTE_UNUSED void *unused) 38f08c3bdfSopenharmony_ci{ 39f08c3bdfSopenharmony_ci memcpy(&info, si, sizeof(info)); 40f08c3bdfSopenharmony_ci received++; 41f08c3bdfSopenharmony_ci} 42f08c3bdfSopenharmony_ci 43f08c3bdfSopenharmony_cistatic void child_func(void) 44f08c3bdfSopenharmony_ci{ 45f08c3bdfSopenharmony_ci pid_t cpid, ppid; 46f08c3bdfSopenharmony_ci mqd_t mqd_child; 47f08c3bdfSopenharmony_ci 48f08c3bdfSopenharmony_ci cpid = tst_getpid(); 49f08c3bdfSopenharmony_ci ppid = getppid(); 50f08c3bdfSopenharmony_ci 51f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(cpid, 1); 52f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(ppid, 0); 53f08c3bdfSopenharmony_ci 54f08c3bdfSopenharmony_ci TST_CHECKPOINT_WAIT(0); 55f08c3bdfSopenharmony_ci 56f08c3bdfSopenharmony_ci tst_res(TINFO, "Send mqueue message from child"); 57f08c3bdfSopenharmony_ci 58f08c3bdfSopenharmony_ci mqd_child = SAFE_MQ_OPEN(MQNAME, O_WRONLY, 0, NULL); 59f08c3bdfSopenharmony_ci SAFE_MQ_SEND(mqd_child, "pippo", 5, 1); 60f08c3bdfSopenharmony_ci 61f08c3bdfSopenharmony_ci TST_CHECKPOINT_WAKE(0); 62f08c3bdfSopenharmony_ci} 63f08c3bdfSopenharmony_ci 64f08c3bdfSopenharmony_cistatic void cleanup(void) 65f08c3bdfSopenharmony_ci{ 66f08c3bdfSopenharmony_ci remove_mqueue(mqd); 67f08c3bdfSopenharmony_ci} 68f08c3bdfSopenharmony_ci 69f08c3bdfSopenharmony_cistatic void run(void) 70f08c3bdfSopenharmony_ci{ 71f08c3bdfSopenharmony_ci pid_t cpid; 72f08c3bdfSopenharmony_ci struct sigaction sa; 73f08c3bdfSopenharmony_ci struct sigevent notif; 74f08c3bdfSopenharmony_ci const struct tst_clone_args args = { 75f08c3bdfSopenharmony_ci .flags = CLONE_NEWPID, 76f08c3bdfSopenharmony_ci .exit_signal = SIGCHLD, 77f08c3bdfSopenharmony_ci }; 78f08c3bdfSopenharmony_ci 79f08c3bdfSopenharmony_ci remove_mqueue(mqd); 80f08c3bdfSopenharmony_ci received = 0; 81f08c3bdfSopenharmony_ci 82f08c3bdfSopenharmony_ci cpid = SAFE_CLONE(&args); 83f08c3bdfSopenharmony_ci if (!cpid) { 84f08c3bdfSopenharmony_ci child_func(); 85f08c3bdfSopenharmony_ci return; 86f08c3bdfSopenharmony_ci } 87f08c3bdfSopenharmony_ci 88f08c3bdfSopenharmony_ci tst_res(TINFO, "Register notification on posix mqueue"); 89f08c3bdfSopenharmony_ci 90f08c3bdfSopenharmony_ci mqd = SAFE_MQ_OPEN(MQNAME, O_RDWR | O_CREAT | O_EXCL, 0777, NULL); 91f08c3bdfSopenharmony_ci 92f08c3bdfSopenharmony_ci notif.sigev_notify = SIGEV_SIGNAL; 93f08c3bdfSopenharmony_ci notif.sigev_signo = SIGUSR1; 94f08c3bdfSopenharmony_ci 95f08c3bdfSopenharmony_ci SAFE_MQ_NOTIFY(mqd, ¬if); 96f08c3bdfSopenharmony_ci 97f08c3bdfSopenharmony_ci sa.sa_flags = SA_SIGINFO; 98f08c3bdfSopenharmony_ci SAFE_SIGEMPTYSET(&sa.sa_mask); 99f08c3bdfSopenharmony_ci sa.sa_sigaction = signal_handler; 100f08c3bdfSopenharmony_ci SAFE_SIGACTION(SIGUSR1, &sa, NULL); 101f08c3bdfSopenharmony_ci 102f08c3bdfSopenharmony_ci TST_CHECKPOINT_WAKE_AND_WAIT(0); 103f08c3bdfSopenharmony_ci 104f08c3bdfSopenharmony_ci tst_reap_children(); 105f08c3bdfSopenharmony_ci 106f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(received, 1); 107f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(info.si_signo, SIGUSR1); 108f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(info.si_code, SI_MESGQ); 109f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(info.si_pid, cpid); 110f08c3bdfSopenharmony_ci} 111f08c3bdfSopenharmony_ci 112f08c3bdfSopenharmony_cistatic struct tst_test test = { 113f08c3bdfSopenharmony_ci .test_all = run, 114f08c3bdfSopenharmony_ci .cleanup = cleanup, 115f08c3bdfSopenharmony_ci .forks_child = 1, 116f08c3bdfSopenharmony_ci .needs_root = 1, 117f08c3bdfSopenharmony_ci .needs_checkpoints = 1, 118f08c3bdfSopenharmony_ci}; 119