* [LTP] [PATCH 0/3] syscalls: add tgkill test-cases
@ 2019-03-07 7:28 Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 1/3] syscalls/tgkill01: add new test Sumit Garg
` (2 more replies)
0 siblings, 3 replies; 8+ messages in thread
From: Sumit Garg @ 2019-03-07 7:28 UTC (permalink / raw)
To: ltp
This patch-set is inherited from Greg which adds test-cases for tgkill
syscall.
Greg Hackmann (3):
syscalls/tgkill01: add new test
syscalls/tgkill02: add new test
syscalls/tgkill03: add new test
runtest/syscalls | 4 +
testcases/kernel/syscalls/tgkill/.gitignore | 3 +
testcases/kernel/syscalls/tgkill/Makefile | 10 ++
testcases/kernel/syscalls/tgkill/tgkill.h | 22 +++++
testcases/kernel/syscalls/tgkill/tgkill01.c | 138 ++++++++++++++++++++++++++++
testcases/kernel/syscalls/tgkill/tgkill02.c | 80 ++++++++++++++++
testcases/kernel/syscalls/tgkill/tgkill03.c | 122 ++++++++++++++++++++++++
7 files changed, 379 insertions(+)
create mode 100644 testcases/kernel/syscalls/tgkill/.gitignore
create mode 100644 testcases/kernel/syscalls/tgkill/Makefile
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill.h
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill01.c
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill02.c
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill03.c
--
2.7.4
^ permalink raw reply [flat|nested] 8+ messages in thread
* [LTP] [PATCH 1/3] syscalls/tgkill01: add new test
2019-03-07 7:28 [LTP] [PATCH 0/3] syscalls: add tgkill test-cases Sumit Garg
@ 2019-03-07 7:28 ` Sumit Garg
2019-03-12 13:29 ` Li Wang
2019-03-14 14:04 ` Cyril Hrubis
2019-03-07 7:28 ` [LTP] [PATCH 2/3] syscalls/tgkill02: " Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 3/3] syscalls/tgkill03: " Sumit Garg
2 siblings, 2 replies; 8+ messages in thread
From: Sumit Garg @ 2019-03-07 7:28 UTC (permalink / raw)
To: ltp
From: Greg Hackmann <ghackmann@google.com>
tgkill() delivers a signal to a specific thread. Test this by
installing a SIGUSR1 handler which records the current pthread ID.
Start a number of threads in parallel, then one-by-one call tgkill(...,
tid, SIGUSR1) and check that the expected pthread ID was recorded.
Signed-off-by: Greg Hackmann <ghackmann@google.com>
Signed-off-by: Sumit Garg <sumit.garg@linaro.org>
---
runtest/syscalls | 2 +
testcases/kernel/syscalls/tgkill/.gitignore | 1 +
testcases/kernel/syscalls/tgkill/Makefile | 10 ++
testcases/kernel/syscalls/tgkill/tgkill.h | 22 +++++
testcases/kernel/syscalls/tgkill/tgkill01.c | 138 ++++++++++++++++++++++++++++
5 files changed, 173 insertions(+)
create mode 100644 testcases/kernel/syscalls/tgkill/.gitignore
create mode 100644 testcases/kernel/syscalls/tgkill/Makefile
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill.h
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill01.c
diff --git a/runtest/syscalls b/runtest/syscalls
index b43de7a..9ed1214 100644
--- a/runtest/syscalls
+++ b/runtest/syscalls
@@ -1395,6 +1395,8 @@ syslog10 syslog10
syslog11 syslog11
syslog12 syslog12
+tgkill01 tgkill01
+
time01 time01
time02 time02
diff --git a/testcases/kernel/syscalls/tgkill/.gitignore b/testcases/kernel/syscalls/tgkill/.gitignore
new file mode 100644
index 0000000..f4566fd
--- /dev/null
+++ b/testcases/kernel/syscalls/tgkill/.gitignore
@@ -0,0 +1 @@
+tgkill01
diff --git a/testcases/kernel/syscalls/tgkill/Makefile b/testcases/kernel/syscalls/tgkill/Makefile
new file mode 100644
index 0000000..a51080c
--- /dev/null
+++ b/testcases/kernel/syscalls/tgkill/Makefile
@@ -0,0 +1,10 @@
+# SPDX-License-Identifier: GPL-2.0-or-later
+# Copyright (c) 2018 Google, Inc.
+
+top_srcdir ?= ../../../..
+
+include $(top_srcdir)/include/mk/testcases.mk
+
+include $(top_srcdir)/include/mk/generic_leaf_target.mk
+
+LDLIBS += -pthread
diff --git a/testcases/kernel/syscalls/tgkill/tgkill.h b/testcases/kernel/syscalls/tgkill/tgkill.h
new file mode 100644
index 0000000..a7d96f4
--- /dev/null
+++ b/testcases/kernel/syscalls/tgkill/tgkill.h
@@ -0,0 +1,22 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Copyright (c) 2018 Google, Inc.
+ */
+
+#ifndef TGKILL_H
+#define TGKILL_H
+
+#include "config.h"
+#include "lapi/syscalls.h"
+
+static inline int sys_tgkill(int tgid, int tid, int sig)
+{
+ return tst_syscall(__NR_tgkill, tgid, tid, sig);
+}
+
+static inline pid_t sys_gettid(void)
+{
+ return tst_syscall(__NR_gettid);
+}
+
+#endif /* TGKILL_H */
diff --git a/testcases/kernel/syscalls/tgkill/tgkill01.c b/testcases/kernel/syscalls/tgkill/tgkill01.c
new file mode 100644
index 0000000..1fb1171
--- /dev/null
+++ b/testcases/kernel/syscalls/tgkill/tgkill01.c
@@ -0,0 +1,138 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Copyright (c) 2018 Google, Inc.
+ *
+ * tgkill() delivers a signal to a specific thread. Test this by installing
+ * a SIGUSR1 handler which records the current pthread ID. Start a number
+ * of threads in parallel, then one-by-one call tgkill(..., tid, SIGUSR1)
+ * and check that the expected pthread ID was recorded.
+ */
+
+#include <pthread.h>
+
+#include "tst_safe_pthread.h"
+#include "tst_test.h"
+#include "tgkill.h"
+
+struct thread_state {
+ pthread_t thread;
+ pid_t tid;
+};
+
+static char *str_threads;
+static int n_threads = 10;
+static struct thread_state *threads;
+
+static pthread_t sigusr1_thread;
+
+static int test_running;
+static pthread_cond_t test_running_cond = PTHREAD_COND_INITIALIZER;
+static pthread_mutex_t test_running_mutex = PTHREAD_MUTEX_INITIALIZER;
+
+static void sigusr1_handler(int signum __attribute__((unused)))
+{
+ sigusr1_thread = pthread_self();
+}
+
+static void *thread_func(void *arg)
+{
+ struct thread_state *thread = arg;
+
+ /**
+ * There is no standard way to map pthread -> tid, so we will have the
+ * child stash its own tid then notify the parent that the stashed tid
+ * is available.
+ */
+ thread->tid = sys_gettid();
+
+ TST_CHECKPOINT_WAKE(0);
+
+ pthread_mutex_lock(&test_running_mutex);
+ while (test_running)
+ pthread_cond_wait(&test_running_cond, &test_running_mutex);
+ pthread_mutex_unlock(&test_running_mutex);
+
+ return arg;
+}
+
+static void start_thread(struct thread_state *thread)
+{
+ SAFE_PTHREAD_CREATE(&thread->thread, NULL, thread_func, thread);
+
+ TST_CHECKPOINT_WAIT(0);
+}
+
+static void stop_threads(void)
+{
+ int i;
+
+ test_running = 0;
+ pthread_cond_broadcast(&test_running_cond);
+
+ for (i = 0; i < n_threads; i++) {
+ if (threads[i].tid == -1)
+ continue;
+
+ SAFE_PTHREAD_JOIN(threads[i].thread, NULL);
+ threads[i].tid = -1;
+ }
+}
+
+static void run(void)
+{
+ int i;
+
+ for (i = 0; i < n_threads; i++)
+ threads[i].tid = -1;
+
+ test_running = 1;
+ for (i = 0; i < n_threads; i++)
+ start_thread(&threads[i]);
+
+ for (i = 0; i < n_threads; i++) {
+ sigusr1_thread = pthread_self();
+
+ TEST(sys_tgkill(getpid(), threads[i].tid, SIGUSR1));
+ if (TST_RET) {
+ tst_res(TFAIL | TTERRNO, "tgkill() failed");
+ return;
+ }
+
+ while (pthread_equal(sigusr1_thread, pthread_self()))
+ usleep(1000);
+
+ if (!pthread_equal(sigusr1_thread, threads[i].thread)) {
+ tst_res(TFAIL, "SIGUSR1 delivered to wrong thread");
+ return;
+ }
+ }
+
+ stop_threads();
+ tst_res(TPASS, "SIGUSR1 delivered to correct threads");
+}
+
+static void setup(void)
+{
+ if (tst_parse_int(str_threads, &n_threads, 1, INT_MAX))
+ tst_brk(TBROK, "Invalid number of threads '%s'", str_threads);
+
+ threads = SAFE_MALLOC(sizeof(*threads) * n_threads);
+
+ struct sigaction sigusr1 = {
+ .sa_handler = sigusr1_handler,
+ };
+ SAFE_SIGACTION(SIGUSR1, &sigusr1, NULL);
+}
+
+static struct tst_option options[] = {
+ {"t:", &str_threads, "-t Number of threads (default 10)"},
+ {NULL, NULL, NULL},
+};
+
+static struct tst_test test = {
+ .options = options,
+ .needs_checkpoints = 1,
+ .setup = setup,
+ .test_all = run,
+ .cleanup = stop_threads,
+};
--
2.7.4
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [LTP] [PATCH 2/3] syscalls/tgkill02: add new test
2019-03-07 7:28 [LTP] [PATCH 0/3] syscalls: add tgkill test-cases Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 1/3] syscalls/tgkill01: add new test Sumit Garg
@ 2019-03-07 7:28 ` Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 3/3] syscalls/tgkill03: " Sumit Garg
2 siblings, 0 replies; 8+ messages in thread
From: Sumit Garg @ 2019-03-07 7:28 UTC (permalink / raw)
To: ltp
From: Greg Hackmann <ghackmann@google.com>
tgkill() should fail with EAGAIN when RLIMIT_SIGPENDING is reached with
a real-time signal. Test this by starting a child thread with SIGRTMIN
blocked and a limit of 0 pending signals, then attempting to deliver
SIGRTMIN from the parent thread.
Signed-off-by: Greg Hackmann <ghackmann@google.com>
Signed-off-by: Sumit Garg <sumit.garg@linaro.org>
---
runtest/syscalls | 1 +
testcases/kernel/syscalls/tgkill/.gitignore | 1 +
testcases/kernel/syscalls/tgkill/tgkill02.c | 80 +++++++++++++++++++++++++++++
3 files changed, 82 insertions(+)
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill02.c
diff --git a/runtest/syscalls b/runtest/syscalls
index 9ed1214..fef45fd 100644
--- a/runtest/syscalls
+++ b/runtest/syscalls
@@ -1396,6 +1396,7 @@ syslog11 syslog11
syslog12 syslog12
tgkill01 tgkill01
+tgkill02 tgkill02
time01 time01
time02 time02
diff --git a/testcases/kernel/syscalls/tgkill/.gitignore b/testcases/kernel/syscalls/tgkill/.gitignore
index f4566fd..42be2bb 100644
--- a/testcases/kernel/syscalls/tgkill/.gitignore
+++ b/testcases/kernel/syscalls/tgkill/.gitignore
@@ -1 +1,2 @@
tgkill01
+tgkill02
diff --git a/testcases/kernel/syscalls/tgkill/tgkill02.c b/testcases/kernel/syscalls/tgkill/tgkill02.c
new file mode 100644
index 0000000..a121992
--- /dev/null
+++ b/testcases/kernel/syscalls/tgkill/tgkill02.c
@@ -0,0 +1,80 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Copyright (c) 2018 Google, Inc.
+ *
+ * tgkill() should fail with EAGAIN when RLIMIT_SIGPENDING is reached with a
+ * real-time signal. Test this by starting a child thread with SIGRTMIN
+ * blocked and a limit of 0 pending signals, then attempting to deliver
+ * SIGRTMIN from the parent thread.
+ */
+
+#include <pthread.h>
+#include <signal.h>
+
+#include "tst_safe_pthread.h"
+#include "tst_test.h"
+#include "tgkill.h"
+
+static int test_running;
+static pthread_cond_t test_running_cond = PTHREAD_COND_INITIALIZER;
+static pthread_mutex_t test_running_mutex = PTHREAD_MUTEX_INITIALIZER;
+
+static void *thread_func(void *arg)
+{
+ const struct rlimit sigpending = {
+ .rlim_cur = 0,
+ .rlim_max = 0,
+ };
+ sigset_t sigrtmin;
+ int err;
+ pid_t *tid = arg;
+
+ sigemptyset(&sigrtmin);
+ sigaddset(&sigrtmin, SIGRTMIN);
+
+ err = pthread_sigmask(SIG_BLOCK, &sigrtmin, NULL);
+ if (err)
+ tst_brk(TBROK, "pthread_sigmask() failed: %s",
+ tst_strerrno(err));
+
+ SAFE_SETRLIMIT(RLIMIT_SIGPENDING, &sigpending);
+ *tid = sys_gettid();
+
+ TST_CHECKPOINT_WAKE(0);
+
+ pthread_mutex_lock(&test_running_mutex);
+ while (test_running)
+ pthread_cond_wait(&test_running_cond, &test_running_mutex);
+ pthread_mutex_unlock(&test_running_mutex);
+
+ return arg;
+}
+
+static void run(void)
+{
+ pthread_t thread;
+ pid_t tid = -1;
+
+ test_running = 1;
+
+ SAFE_PTHREAD_CREATE(&thread, NULL, thread_func, &tid);
+
+ TST_CHECKPOINT_WAIT(0);
+
+ TEST(sys_tgkill(getpid(), tid, SIGRTMIN));
+ if (TST_RET && TST_ERR == EAGAIN)
+ tst_res(TPASS, "tgkill() failed with EAGAIN as expected");
+ else
+ tst_res(TFAIL | TTERRNO,
+ "tgkill() should have failed with EAGAIN");
+
+ test_running = 0;
+ pthread_cond_broadcast(&test_running_cond);
+
+ SAFE_PTHREAD_JOIN(thread, NULL);
+}
+
+static struct tst_test test = {
+ .needs_checkpoints = 1,
+ .test_all = run,
+};
--
2.7.4
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [LTP] [PATCH 3/3] syscalls/tgkill03: add new test
2019-03-07 7:28 [LTP] [PATCH 0/3] syscalls: add tgkill test-cases Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 1/3] syscalls/tgkill01: add new test Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 2/3] syscalls/tgkill02: " Sumit Garg
@ 2019-03-07 7:28 ` Sumit Garg
2 siblings, 0 replies; 8+ messages in thread
From: Sumit Garg @ 2019-03-07 7:28 UTC (permalink / raw)
To: ltp
From: Greg Hackmann <ghackmann@google.com>
Test simple tgkill() error cases.
Signed-off-by: Greg Hackmann <ghackmann@google.com>
Signed-off-by: Sumit Garg <sumit.garg@linaro.org>
---
runtest/syscalls | 1 +
testcases/kernel/syscalls/tgkill/.gitignore | 1 +
testcases/kernel/syscalls/tgkill/tgkill03.c | 122 ++++++++++++++++++++++++++++
3 files changed, 124 insertions(+)
create mode 100644 testcases/kernel/syscalls/tgkill/tgkill03.c
diff --git a/runtest/syscalls b/runtest/syscalls
index fef45fd..13096f3 100644
--- a/runtest/syscalls
+++ b/runtest/syscalls
@@ -1397,6 +1397,7 @@ syslog12 syslog12
tgkill01 tgkill01
tgkill02 tgkill02
+tgkill03 tgkill03
time01 time01
time02 time02
diff --git a/testcases/kernel/syscalls/tgkill/.gitignore b/testcases/kernel/syscalls/tgkill/.gitignore
index 42be2bb..a6d2299 100644
--- a/testcases/kernel/syscalls/tgkill/.gitignore
+++ b/testcases/kernel/syscalls/tgkill/.gitignore
@@ -1,2 +1,3 @@
tgkill01
tgkill02
+tgkill03
diff --git a/testcases/kernel/syscalls/tgkill/tgkill03.c b/testcases/kernel/syscalls/tgkill/tgkill03.c
new file mode 100644
index 0000000..3597d1b
--- /dev/null
+++ b/testcases/kernel/syscalls/tgkill/tgkill03.c
@@ -0,0 +1,122 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Copyright (c) 2018 Google, Inc.
+ *
+ * Test simple tgkill() error cases.
+ */
+
+#include <pthread.h>
+#include <pwd.h>
+#include <sys/types.h>
+
+#include "tst_safe_pthread.h"
+#include "tst_test.h"
+#include "tgkill.h"
+
+static pthread_t child_thread;
+
+static pid_t parent_tgid;
+static pid_t parent_tid;
+static pid_t child_tid;
+static pid_t defunct_tid;
+
+static const int invalid_pid = -1;
+
+static int test_running;
+static pthread_cond_t test_running_cond = PTHREAD_COND_INITIALIZER;
+static pthread_mutex_t test_running_mutex = PTHREAD_MUTEX_INITIALIZER;
+
+static void *child_thread_func(void *arg)
+{
+ child_tid = sys_gettid();
+
+ TST_CHECKPOINT_WAKE(0);
+
+ pthread_mutex_lock(&test_running_mutex);
+ while (test_running)
+ pthread_cond_wait(&test_running_cond, &test_running_mutex);
+ pthread_mutex_unlock(&test_running_mutex);
+
+ return arg;
+}
+
+static void *defunct_thread_func(void *arg)
+{
+ defunct_tid = sys_gettid();
+
+ return arg;
+}
+
+static void setup(void)
+{
+ sigset_t sigusr1;
+ pthread_t defunct_thread;
+
+ sigemptyset(&sigusr1);
+ sigaddset(&sigusr1, SIGUSR1);
+ pthread_sigmask(SIG_BLOCK, &sigusr1, NULL);
+
+ parent_tgid = getpid();
+ parent_tid = sys_gettid();
+
+ test_running = 1;
+
+ SAFE_PTHREAD_CREATE(&child_thread, NULL, child_thread_func, NULL);
+
+ TST_CHECKPOINT_WAIT(0);
+
+ SAFE_PTHREAD_CREATE(&defunct_thread, NULL, defunct_thread_func, NULL);
+
+ SAFE_PTHREAD_JOIN(defunct_thread, NULL);
+}
+
+static void cleanup(void)
+{
+ test_running = 0;
+ pthread_cond_broadcast(&test_running_cond);
+
+ SAFE_PTHREAD_JOIN(child_thread, NULL);
+}
+
+static const struct testcase {
+ const char *desc;
+ const int *tgid;
+ const int *tid;
+ const int sig;
+ const int err;
+} testcases[] = {
+ { "Invalid tgid", &invalid_pid, &parent_tid, SIGUSR1, EINVAL },
+ { "Invalid tid", &parent_tgid, &invalid_pid, SIGUSR1, EINVAL },
+ { "Invalid signal", &parent_tgid, &parent_tid, -1, EINVAL },
+ { "Defunct thread ID", &parent_tgid, &defunct_tid, SIGUSR1, ESRCH },
+ { "Valid tgkill call", &parent_tgid, &child_tid, SIGUSR1, 0 },
+};
+
+static void run(unsigned int i)
+{
+ const struct testcase *tc = &testcases[i];
+
+ TEST(sys_tgkill(*tc->tgid, *tc->tid, tc->sig));
+ if (tc->err) {
+ if (TST_RET < 0 && TST_ERR == tc->err)
+ tst_res(TPASS | TTERRNO, "%s failed as expected",
+ tc->desc);
+ else
+ tst_res(TFAIL | TTERRNO,
+ "%s should have failed with %s", tc->desc,
+ tst_strerrno(tc->err));
+ } else {
+ if (TST_RET == 0)
+ tst_res(TPASS, "%s succeeded", tc->desc);
+ else
+ tst_res(TFAIL | TTERRNO, "%s failed", tc->desc);
+ }
+}
+
+static struct tst_test test = {
+ .tcnt = ARRAY_SIZE(testcases),
+ .needs_checkpoints = 1,
+ .setup = setup,
+ .cleanup = cleanup,
+ .test = run,
+};
--
2.7.4
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [LTP] [PATCH 1/3] syscalls/tgkill01: add new test
2019-03-07 7:28 ` [LTP] [PATCH 1/3] syscalls/tgkill01: add new test Sumit Garg
@ 2019-03-12 13:29 ` Li Wang
2019-03-12 13:49 ` Sumit Garg
2019-03-14 14:04 ` Cyril Hrubis
1 sibling, 1 reply; 8+ messages in thread
From: Li Wang @ 2019-03-12 13:29 UTC (permalink / raw)
To: ltp
On Thu, Mar 7, 2019 at 3:29 PM Sumit Garg <sumit.garg@linaro.org> wrote:
> From: Greg Hackmann <ghackmann@google.com>
>
> tgkill() delivers a signal to a specific thread. Test this by
> installing a SIGUSR1 handler which records the current pthread ID.
> Start a number of threads in parallel, then one-by-one call tgkill(...,
> tid, SIGUSR1) and check that the expected pthread ID was recorded.
>
> Signed-off-by: Greg Hackmann <ghackmann@google.com>
> Signed-off-by: Sumit Garg <sumit.garg@linaro.org>
>
Reviewed-by: Li Wang <liwang@redhat.com>
This patch set looks pretty good to me. Just have a tiny comments for PATCH
1/3 in below.
> ---
> runtest/syscalls | 2 +
> testcases/kernel/syscalls/tgkill/.gitignore | 1 +
> testcases/kernel/syscalls/tgkill/Makefile | 10 ++
> testcases/kernel/syscalls/tgkill/tgkill.h | 22 +++++
> testcases/kernel/syscalls/tgkill/tgkill01.c | 138
> ++++++++++++++++++++++++++++
> 5 files changed, 173 insertions(+)
> create mode 100644 testcases/kernel/syscalls/tgkill/.gitignore
> create mode 100644 testcases/kernel/syscalls/tgkill/Makefile
> create mode 100644 testcases/kernel/syscalls/tgkill/tgkill.h
> create mode 100644 testcases/kernel/syscalls/tgkill/tgkill01.c
>
> diff --git a/runtest/syscalls b/runtest/syscalls
> index b43de7a..9ed1214 100644
> --- a/runtest/syscalls
> +++ b/runtest/syscalls
> @@ -1395,6 +1395,8 @@ syslog10 syslog10
> syslog11 syslog11
> syslog12 syslog12
>
> +tgkill01 tgkill01
> +
> time01 time01
> time02 time02
>
> diff --git a/testcases/kernel/syscalls/tgkill/.gitignore
> b/testcases/kernel/syscalls/tgkill/.gitignore
> new file mode 100644
> index 0000000..f4566fd
> --- /dev/null
> +++ b/testcases/kernel/syscalls/tgkill/.gitignore
> @@ -0,0 +1 @@
> +tgkill01
> diff --git a/testcases/kernel/syscalls/tgkill/Makefile
> b/testcases/kernel/syscalls/tgkill/Makefile
> new file mode 100644
> index 0000000..a51080c
> --- /dev/null
> +++ b/testcases/kernel/syscalls/tgkill/Makefile
> @@ -0,0 +1,10 @@
> +# SPDX-License-Identifier: GPL-2.0-or-later
> +# Copyright (c) 2018 Google, Inc.
> +
> +top_srcdir ?= ../../../..
> +
> +include $(top_srcdir)/include/mk/testcases.mk
> +
> +include $(top_srcdir)/include/mk/generic_leaf_target.mk
> +
> +LDLIBS += -pthread
> diff --git a/testcases/kernel/syscalls/tgkill/tgkill.h
> b/testcases/kernel/syscalls/tgkill/tgkill.h
> new file mode 100644
> index 0000000..a7d96f4
> --- /dev/null
> +++ b/testcases/kernel/syscalls/tgkill/tgkill.h
> @@ -0,0 +1,22 @@
> +// SPDX-License-Identifier: GPL-2.0-or-later
> +/*
> + * Copyright (c) 2018 Google, Inc.
> + */
> +
> +#ifndef TGKILL_H
> +#define TGKILL_H
> +
> +#include "config.h"
> +#include "lapi/syscalls.h"
> +
> +static inline int sys_tgkill(int tgid, int tid, int sig)
> +{
> + return tst_syscall(__NR_tgkill, tgid, tid, sig);
> +}
> +
> +static inline pid_t sys_gettid(void)
> +{
> + return tst_syscall(__NR_gettid);
> +}
> +
> +#endif /* TGKILL_H */
> diff --git a/testcases/kernel/syscalls/tgkill/tgkill01.c
> b/testcases/kernel/syscalls/tgkill/tgkill01.c
> new file mode 100644
> index 0000000..1fb1171
> --- /dev/null
> +++ b/testcases/kernel/syscalls/tgkill/tgkill01.c
> @@ -0,0 +1,138 @@
> +// SPDX-License-Identifier: GPL-2.0-or-later
> +/*
> + * Copyright (c) 2018 Google, Inc.
> + *
> + * tgkill() delivers a signal to a specific thread. Test this by
> installing
> + * a SIGUSR1 handler which records the current pthread ID. Start a number
> + * of threads in parallel, then one-by-one call tgkill(..., tid, SIGUSR1)
> + * and check that the expected pthread ID was recorded.
> + */
> +
> +#include <pthread.h>
> +
> +#include "tst_safe_pthread.h"
> +#include "tst_test.h"
> +#include "tgkill.h"
> +
> +struct thread_state {
> + pthread_t thread;
> + pid_t tid;
> +};
> +
> +static char *str_threads;
> +static int n_threads = 10;
> +static struct thread_state *threads;
> +
> +static pthread_t sigusr1_thread;
> +
> +static int test_running;
> +static pthread_cond_t test_running_cond = PTHREAD_COND_INITIALIZER;
> +static pthread_mutex_t test_running_mutex = PTHREAD_MUTEX_INITIALIZER;
> +
> +static void sigusr1_handler(int signum __attribute__((unused)))
> +{
> + sigusr1_thread = pthread_self();
> +}
> +
> +static void *thread_func(void *arg)
> +{
> + struct thread_state *thread = arg;
> +
> + /**
> + * There is no standard way to map pthread -> tid, so we will have
> the
> + * child stash its own tid then notify the parent that the stashed
> tid
> + * is available.
> + */
> + thread->tid = sys_gettid();
> +
> + TST_CHECKPOINT_WAKE(0);
> +
> + pthread_mutex_lock(&test_running_mutex);
> + while (test_running)
> + pthread_cond_wait(&test_running_cond, &test_running_mutex);
> + pthread_mutex_unlock(&test_running_mutex);
> +
> + return arg;
> +}
> +
> +static void start_thread(struct thread_state *thread)
> +{
> + SAFE_PTHREAD_CREATE(&thread->thread, NULL, thread_func, thread);
> +
> + TST_CHECKPOINT_WAIT(0);
> +}
> +
> +static void stop_threads(void)
> +{
> + int i;
> +
> + test_running = 0;
> + pthread_cond_broadcast(&test_running_cond);
> +
> + for (i = 0; i < n_threads; i++) {
> + if (threads[i].tid == -1)
> + continue;
> +
> + SAFE_PTHREAD_JOIN(threads[i].thread, NULL);
> + threads[i].tid = -1;
> + }
>
Maybe we can add free() to release the allocated memory here.
if (threads)
free(threads);
> +}
> +
> +static void run(void)
> +{
> + int i;
> +
> + for (i = 0; i < n_threads; i++)
> + threads[i].tid = -1;
> +
> + test_running = 1;
> + for (i = 0; i < n_threads; i++)
> + start_thread(&threads[i]);
> +
> + for (i = 0; i < n_threads; i++) {
> + sigusr1_thread = pthread_self();
> +
> + TEST(sys_tgkill(getpid(), threads[i].tid, SIGUSR1));
> + if (TST_RET) {
> + tst_res(TFAIL | TTERRNO, "tgkill() failed");
> + return;
> + }
> +
> + while (pthread_equal(sigusr1_thread, pthread_self()))
> + usleep(1000);
> +
> + if (!pthread_equal(sigusr1_thread, threads[i].thread)) {
> + tst_res(TFAIL, "SIGUSR1 delivered to wrong
> thread");
> + return;
> + }
> + }
> +
> + stop_threads();
>
Since we have set the .cleanup function as stop_threads() in tst_test
struct, I think
we can remove it here.
> + tst_res(TPASS, "SIGUSR1 delivered to correct threads");
> +}
> +
> +static void setup(void)
> +{
> + if (tst_parse_int(str_threads, &n_threads, 1, INT_MAX))
> + tst_brk(TBROK, "Invalid number of threads '%s'",
> str_threads);
> +
> + threads = SAFE_MALLOC(sizeof(*threads) * n_threads);
> +
> + struct sigaction sigusr1 = {
> + .sa_handler = sigusr1_handler,
> + };
> + SAFE_SIGACTION(SIGUSR1, &sigusr1, NULL);
> +}
> +
> +static struct tst_option options[] = {
> + {"t:", &str_threads, "-t Number of threads (default 10)"},
> + {NULL, NULL, NULL},
> +};
> +
> +static struct tst_test test = {
> + .options = options,
> + .needs_checkpoints = 1,
> + .setup = setup,
> + .test_all = run,
> + .cleanup = stop_threads,
> +};
> --
> 2.7.4
>
>
> --
> Mailing list info: https://lists.linux.it/listinfo/ltp
>
--
Regards,
Li Wang
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.linux.it/pipermail/ltp/attachments/20190312/8ac27a72/attachment.html>
^ permalink raw reply [flat|nested] 8+ messages in thread
* [LTP] [PATCH 1/3] syscalls/tgkill01: add new test
2019-03-12 13:29 ` Li Wang
@ 2019-03-12 13:49 ` Sumit Garg
0 siblings, 0 replies; 8+ messages in thread
From: Sumit Garg @ 2019-03-12 13:49 UTC (permalink / raw)
To: ltp
On Tue, 12 Mar 2019 at 19:00, Li Wang <liwang@redhat.com> wrote:
>
>
>
> On Thu, Mar 7, 2019 at 3:29 PM Sumit Garg <sumit.garg@linaro.org> wrote:
>>
>> From: Greg Hackmann <ghackmann@google.com>
>>
>> tgkill() delivers a signal to a specific thread. Test this by
>> installing a SIGUSR1 handler which records the current pthread ID.
>> Start a number of threads in parallel, then one-by-one call tgkill(...,
>> tid, SIGUSR1) and check that the expected pthread ID was recorded.
>>
>> Signed-off-by: Greg Hackmann <ghackmann@google.com>
>> Signed-off-by: Sumit Garg <sumit.garg@linaro.org>
>
> Reviewed-by: Li Wang <liwang@redhat.com>
>
> This patch set looks pretty good to me. Just have a tiny comments for PATCH 1/3 in below.
Thanks for your review.
>
>>
>> ---
>> runtest/syscalls | 2 +
>> testcases/kernel/syscalls/tgkill/.gitignore | 1 +
>> testcases/kernel/syscalls/tgkill/Makefile | 10 ++
>> testcases/kernel/syscalls/tgkill/tgkill.h | 22 +++++
>> testcases/kernel/syscalls/tgkill/tgkill01.c | 138 ++++++++++++++++++++++++++++
>> 5 files changed, 173 insertions(+)
>> create mode 100644 testcases/kernel/syscalls/tgkill/.gitignore
>> create mode 100644 testcases/kernel/syscalls/tgkill/Makefile
>> create mode 100644 testcases/kernel/syscalls/tgkill/tgkill.h
>> create mode 100644 testcases/kernel/syscalls/tgkill/tgkill01.c
>>
>> diff --git a/runtest/syscalls b/runtest/syscalls
>> index b43de7a..9ed1214 100644
>> --- a/runtest/syscalls
>> +++ b/runtest/syscalls
>> @@ -1395,6 +1395,8 @@ syslog10 syslog10
>> syslog11 syslog11
>> syslog12 syslog12
>>
>> +tgkill01 tgkill01
>> +
>> time01 time01
>> time02 time02
>>
>> diff --git a/testcases/kernel/syscalls/tgkill/.gitignore b/testcases/kernel/syscalls/tgkill/.gitignore
>> new file mode 100644
>> index 0000000..f4566fd
>> --- /dev/null
>> +++ b/testcases/kernel/syscalls/tgkill/.gitignore
>> @@ -0,0 +1 @@
>> +tgkill01
>> diff --git a/testcases/kernel/syscalls/tgkill/Makefile b/testcases/kernel/syscalls/tgkill/Makefile
>> new file mode 100644
>> index 0000000..a51080c
>> --- /dev/null
>> +++ b/testcases/kernel/syscalls/tgkill/Makefile
>> @@ -0,0 +1,10 @@
>> +# SPDX-License-Identifier: GPL-2.0-or-later
>> +# Copyright (c) 2018 Google, Inc.
>> +
>> +top_srcdir ?= ../../../..
>> +
>> +include $(top_srcdir)/include/mk/testcases.mk
>> +
>> +include $(top_srcdir)/include/mk/generic_leaf_target.mk
>> +
>> +LDLIBS += -pthread
>> diff --git a/testcases/kernel/syscalls/tgkill/tgkill.h b/testcases/kernel/syscalls/tgkill/tgkill.h
>> new file mode 100644
>> index 0000000..a7d96f4
>> --- /dev/null
>> +++ b/testcases/kernel/syscalls/tgkill/tgkill.h
>> @@ -0,0 +1,22 @@
>> +// SPDX-License-Identifier: GPL-2.0-or-later
>> +/*
>> + * Copyright (c) 2018 Google, Inc.
>> + */
>> +
>> +#ifndef TGKILL_H
>> +#define TGKILL_H
>> +
>> +#include "config.h"
>> +#include "lapi/syscalls.h"
>> +
>> +static inline int sys_tgkill(int tgid, int tid, int sig)
>> +{
>> + return tst_syscall(__NR_tgkill, tgid, tid, sig);
>> +}
>> +
>> +static inline pid_t sys_gettid(void)
>> +{
>> + return tst_syscall(__NR_gettid);
>> +}
>> +
>> +#endif /* TGKILL_H */
>> diff --git a/testcases/kernel/syscalls/tgkill/tgkill01.c b/testcases/kernel/syscalls/tgkill/tgkill01.c
>> new file mode 100644
>> index 0000000..1fb1171
>> --- /dev/null
>> +++ b/testcases/kernel/syscalls/tgkill/tgkill01.c
>> @@ -0,0 +1,138 @@
>> +// SPDX-License-Identifier: GPL-2.0-or-later
>> +/*
>> + * Copyright (c) 2018 Google, Inc.
>> + *
>> + * tgkill() delivers a signal to a specific thread. Test this by installing
>> + * a SIGUSR1 handler which records the current pthread ID. Start a number
>> + * of threads in parallel, then one-by-one call tgkill(..., tid, SIGUSR1)
>> + * and check that the expected pthread ID was recorded.
>> + */
>> +
>> +#include <pthread.h>
>> +
>> +#include "tst_safe_pthread.h"
>> +#include "tst_test.h"
>> +#include "tgkill.h"
>> +
>> +struct thread_state {
>> + pthread_t thread;
>> + pid_t tid;
>> +};
>> +
>> +static char *str_threads;
>> +static int n_threads = 10;
>> +static struct thread_state *threads;
>> +
>> +static pthread_t sigusr1_thread;
>> +
>> +static int test_running;
>> +static pthread_cond_t test_running_cond = PTHREAD_COND_INITIALIZER;
>> +static pthread_mutex_t test_running_mutex = PTHREAD_MUTEX_INITIALIZER;
>> +
>> +static void sigusr1_handler(int signum __attribute__((unused)))
>> +{
>> + sigusr1_thread = pthread_self();
>> +}
>> +
>> +static void *thread_func(void *arg)
>> +{
>> + struct thread_state *thread = arg;
>> +
>> + /**
>> + * There is no standard way to map pthread -> tid, so we will have the
>> + * child stash its own tid then notify the parent that the stashed tid
>> + * is available.
>> + */
>> + thread->tid = sys_gettid();
>> +
>> + TST_CHECKPOINT_WAKE(0);
>> +
>> + pthread_mutex_lock(&test_running_mutex);
>> + while (test_running)
>> + pthread_cond_wait(&test_running_cond, &test_running_mutex);
>> + pthread_mutex_unlock(&test_running_mutex);
>> +
>> + return arg;
>> +}
>> +
>> +static void start_thread(struct thread_state *thread)
>> +{
>> + SAFE_PTHREAD_CREATE(&thread->thread, NULL, thread_func, thread);
>> +
>> + TST_CHECKPOINT_WAIT(0);
>> +}
>> +
>> +static void stop_threads(void)
>> +{
>> + int i;
>> +
>> + test_running = 0;
>> + pthread_cond_broadcast(&test_running_cond);
>> +
>> + for (i = 0; i < n_threads; i++) {
>> + if (threads[i].tid == -1)
>> + continue;
>> +
>> + SAFE_PTHREAD_JOIN(threads[i].thread, NULL);
>> + threads[i].tid = -1;
>> + }
>
>
> Maybe we can add free() to release the allocated memory here.
>
> if (threads)
> free(threads);
>
Ok, will add this here.
>
>>
>> +}
>> +
>> +static void run(void)
>> +{
>> + int i;
>> +
>> + for (i = 0; i < n_threads; i++)
>> + threads[i].tid = -1;
>> +
>> + test_running = 1;
>> + for (i = 0; i < n_threads; i++)
>> + start_thread(&threads[i]);
>> +
>> + for (i = 0; i < n_threads; i++) {
>> + sigusr1_thread = pthread_self();
>> +
>> + TEST(sys_tgkill(getpid(), threads[i].tid, SIGUSR1));
>> + if (TST_RET) {
>> + tst_res(TFAIL | TTERRNO, "tgkill() failed");
>> + return;
>> + }
>> +
>> + while (pthread_equal(sigusr1_thread, pthread_self()))
>> + usleep(1000);
>> +
>> + if (!pthread_equal(sigusr1_thread, threads[i].thread)) {
>> + tst_res(TFAIL, "SIGUSR1 delivered to wrong thread");
>> + return;
>> + }
>> + }
>> +
>> + stop_threads();
>
>
> Since we have set the .cleanup function as stop_threads() in tst_test struct, I think
> we can remove it here.
Makes sense, will remove it.
-Sumit
>>
>> + tst_res(TPASS, "SIGUSR1 delivered to correct threads");
>> +}
>> +
>> +static void setup(void)
>> +{
>> + if (tst_parse_int(str_threads, &n_threads, 1, INT_MAX))
>> + tst_brk(TBROK, "Invalid number of threads '%s'", str_threads);
>> +
>> + threads = SAFE_MALLOC(sizeof(*threads) * n_threads);
>> +
>> + struct sigaction sigusr1 = {
>> + .sa_handler = sigusr1_handler,
>> + };
>> + SAFE_SIGACTION(SIGUSR1, &sigusr1, NULL);
>> +}
>> +
>> +static struct tst_option options[] = {
>> + {"t:", &str_threads, "-t Number of threads (default 10)"},
>> + {NULL, NULL, NULL},
>> +};
>> +
>> +static struct tst_test test = {
>> + .options = options,
>> + .needs_checkpoints = 1,
>> + .setup = setup,
>> + .test_all = run,
>> + .cleanup = stop_threads,
>> +};
>> --
>> 2.7.4
>>
>>
>> --
>> Mailing list info: https://lists.linux.it/listinfo/ltp
>
>
>
> --
> Regards,
> Li Wang
^ permalink raw reply [flat|nested] 8+ messages in thread
* [LTP] [PATCH 1/3] syscalls/tgkill01: add new test
2019-03-07 7:28 ` [LTP] [PATCH 1/3] syscalls/tgkill01: add new test Sumit Garg
2019-03-12 13:29 ` Li Wang
@ 2019-03-14 14:04 ` Cyril Hrubis
2019-03-15 9:48 ` Sumit Garg
1 sibling, 1 reply; 8+ messages in thread
From: Cyril Hrubis @ 2019-03-14 14:04 UTC (permalink / raw)
To: ltp
Hi!
> +static void run(void)
> +{
> + int i;
> +
> + for (i = 0; i < n_threads; i++)
> + threads[i].tid = -1;
> +
> + test_running = 1;
> + for (i = 0; i < n_threads; i++)
> + start_thread(&threads[i]);
Just a minor nit these threads should actully be started in the test
setup() since we stop them in the test cleanup(), otherwise the test
breaks with the looping -i option.
--
Cyril Hrubis
chrubis@suse.cz
^ permalink raw reply [flat|nested] 8+ messages in thread
* [LTP] [PATCH 1/3] syscalls/tgkill01: add new test
2019-03-14 14:04 ` Cyril Hrubis
@ 2019-03-15 9:48 ` Sumit Garg
0 siblings, 0 replies; 8+ messages in thread
From: Sumit Garg @ 2019-03-15 9:48 UTC (permalink / raw)
To: ltp
On Thu, 14 Mar 2019 at 19:34, Cyril Hrubis <chrubis@suse.cz> wrote:
>
> Hi!
> > +static void run(void)
> > +{
> > + int i;
> > +
> > + for (i = 0; i < n_threads; i++)
> > + threads[i].tid = -1;
> > +
> > + test_running = 1;
> > + for (i = 0; i < n_threads; i++)
> > + start_thread(&threads[i]);
>
> Just a minor nit these threads should actully be started in the test
> setup() since we stop them in the test cleanup(), otherwise the test
> breaks with the looping -i option.
>
Makes sense, will fix it.
-Sumit
> --
> Cyril Hrubis
> chrubis@suse.cz
^ permalink raw reply [flat|nested] 8+ messages in thread
end of thread, other threads:[~2019-03-15 9:48 UTC | newest]
Thread overview: 8+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2019-03-07 7:28 [LTP] [PATCH 0/3] syscalls: add tgkill test-cases Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 1/3] syscalls/tgkill01: add new test Sumit Garg
2019-03-12 13:29 ` Li Wang
2019-03-12 13:49 ` Sumit Garg
2019-03-14 14:04 ` Cyril Hrubis
2019-03-15 9:48 ` Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 2/3] syscalls/tgkill02: " Sumit Garg
2019-03-07 7:28 ` [LTP] [PATCH 3/3] syscalls/tgkill03: " Sumit Garg
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.