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 parent. Then check if signal 13f08c3bdfSopenharmony_ci * notification contains si_pid of the parent. 14f08c3bdfSopenharmony_ci */ 15f08c3bdfSopenharmony_ci 16f08c3bdfSopenharmony_ci#define _GNU_SOURCE 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 siginfo_t info; 27f08c3bdfSopenharmony_cistatic volatile int received; 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 child_signal_handler(LTP_ATTRIBUTE_UNUSED int sig, siginfo_t *si, LTP_ATTRIBUTE_UNUSED void *unused) 38f08c3bdfSopenharmony_ci{ 39f08c3bdfSopenharmony_ci received = 1; 40f08c3bdfSopenharmony_ci memcpy(&info, si, sizeof(info)); 41f08c3bdfSopenharmony_ci} 42f08c3bdfSopenharmony_ci 43f08c3bdfSopenharmony_cistatic void child_func(void) 44f08c3bdfSopenharmony_ci{ 45f08c3bdfSopenharmony_ci pid_t cpid, ppid; 46f08c3bdfSopenharmony_ci struct sigaction sa; 47f08c3bdfSopenharmony_ci struct sigevent notif; 48f08c3bdfSopenharmony_ci mqd_t mqd_child; 49f08c3bdfSopenharmony_ci 50f08c3bdfSopenharmony_ci cpid = tst_getpid(); 51f08c3bdfSopenharmony_ci ppid = getppid(); 52f08c3bdfSopenharmony_ci 53f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(cpid, 1); 54f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(ppid, 0); 55f08c3bdfSopenharmony_ci 56f08c3bdfSopenharmony_ci TST_CHECKPOINT_WAIT(0); 57f08c3bdfSopenharmony_ci 58f08c3bdfSopenharmony_ci tst_res(TINFO, "Register notification on posix mqueue"); 59f08c3bdfSopenharmony_ci 60f08c3bdfSopenharmony_ci mqd_child = SAFE_MQ_OPEN(MQNAME, O_RDONLY, 0, NULL); 61f08c3bdfSopenharmony_ci notif.sigev_notify = SIGEV_SIGNAL; 62f08c3bdfSopenharmony_ci notif.sigev_signo = SIGUSR1; 63f08c3bdfSopenharmony_ci notif.sigev_value.sival_int = mqd_child; 64f08c3bdfSopenharmony_ci 65f08c3bdfSopenharmony_ci SAFE_MQ_NOTIFY(mqd_child, ¬if); 66f08c3bdfSopenharmony_ci 67f08c3bdfSopenharmony_ci sa.sa_flags = SA_SIGINFO; 68f08c3bdfSopenharmony_ci SAFE_SIGEMPTYSET(&sa.sa_mask); 69f08c3bdfSopenharmony_ci sa.sa_sigaction = child_signal_handler; 70f08c3bdfSopenharmony_ci SAFE_SIGACTION(SIGUSR1, &sa, NULL); 71f08c3bdfSopenharmony_ci 72f08c3bdfSopenharmony_ci TST_CHECKPOINT_WAKE_AND_WAIT(0); 73f08c3bdfSopenharmony_ci 74f08c3bdfSopenharmony_ci if (received) 75f08c3bdfSopenharmony_ci tst_res(TPASS, "Signal notification has been received"); 76f08c3bdfSopenharmony_ci else 77f08c3bdfSopenharmony_ci tst_res(TFAIL, "Signal notification has not been received"); 78f08c3bdfSopenharmony_ci 79f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(info.si_signo, SIGUSR1); 80f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(info.si_code, SI_MESGQ); 81f08c3bdfSopenharmony_ci TST_EXP_EQ_LI(info.si_pid, 0); 82f08c3bdfSopenharmony_ci} 83f08c3bdfSopenharmony_ci 84f08c3bdfSopenharmony_cistatic void cleanup(void) 85f08c3bdfSopenharmony_ci{ 86f08c3bdfSopenharmony_ci remove_mqueue(mqd); 87f08c3bdfSopenharmony_ci} 88f08c3bdfSopenharmony_ci 89f08c3bdfSopenharmony_cistatic void run(void) 90f08c3bdfSopenharmony_ci{ 91f08c3bdfSopenharmony_ci const struct tst_clone_args args = { 92f08c3bdfSopenharmony_ci .flags = CLONE_NEWPID, 93f08c3bdfSopenharmony_ci .exit_signal = SIGCHLD, 94f08c3bdfSopenharmony_ci }; 95f08c3bdfSopenharmony_ci 96f08c3bdfSopenharmony_ci remove_mqueue(mqd); 97f08c3bdfSopenharmony_ci received = 0; 98f08c3bdfSopenharmony_ci 99f08c3bdfSopenharmony_ci if (!SAFE_CLONE(&args)) { 100f08c3bdfSopenharmony_ci child_func(); 101f08c3bdfSopenharmony_ci return; 102f08c3bdfSopenharmony_ci } 103f08c3bdfSopenharmony_ci 104f08c3bdfSopenharmony_ci mqd = SAFE_MQ_OPEN(MQNAME, O_RDWR | O_CREAT | O_EXCL, 0777, 0); 105f08c3bdfSopenharmony_ci 106f08c3bdfSopenharmony_ci TST_CHECKPOINT_WAKE_AND_WAIT(0); 107f08c3bdfSopenharmony_ci 108f08c3bdfSopenharmony_ci tst_res(TINFO, "Send mqueue message"); 109f08c3bdfSopenharmony_ci 110f08c3bdfSopenharmony_ci SAFE_MQ_SEND(mqd, "pippo", 5, 1); 111f08c3bdfSopenharmony_ci 112f08c3bdfSopenharmony_ci TST_CHECKPOINT_WAKE(0); 113f08c3bdfSopenharmony_ci} 114f08c3bdfSopenharmony_ci 115f08c3bdfSopenharmony_cistatic struct tst_test test = { 116f08c3bdfSopenharmony_ci .test_all = run, 117f08c3bdfSopenharmony_ci .cleanup = cleanup, 118f08c3bdfSopenharmony_ci .forks_child = 1, 119f08c3bdfSopenharmony_ci .needs_root = 1, 120f08c3bdfSopenharmony_ci .needs_checkpoints = 1, 121f08c3bdfSopenharmony_ci}; 122