Module Name:    src
Committed By:   thorpej
Date:           Mon Dec 24 21:42:05 UTC 2018

Modified Files:
        src/distrib/sets/lists/tests: mi
        src/tests/rump/kernspace: Makefile kernspace.h
        src/tests/rump/rumpkern: Makefile
Added Files:
        src/tests/rump/kernspace: threadpool.c
        src/tests/rump/rumpkern: t_threadpool.c

Log Message:
Add rump-based test cases for threadpool(9).


To generate a diff of this commit:
cvs rdiff -u -r1.799 -r1.800 src/distrib/sets/lists/tests/mi
cvs rdiff -u -r1.6 -r1.7 src/tests/rump/kernspace/Makefile \
    src/tests/rump/kernspace/kernspace.h
cvs rdiff -u -r0 -r1.1 src/tests/rump/kernspace/threadpool.c
cvs rdiff -u -r1.16 -r1.17 src/tests/rump/rumpkern/Makefile
cvs rdiff -u -r0 -r1.1 src/tests/rump/rumpkern/t_threadpool.c

Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.

Modified files:

Index: src/distrib/sets/lists/tests/mi
diff -u src/distrib/sets/lists/tests/mi:1.799 src/distrib/sets/lists/tests/mi:1.800
--- src/distrib/sets/lists/tests/mi:1.799	Mon Dec 24 16:58:54 2018
+++ src/distrib/sets/lists/tests/mi	Mon Dec 24 21:42:05 2018
@@ -1,4 +1,4 @@
-# $NetBSD: mi,v 1.799 2018/12/24 16:58:54 thorpej Exp $
+# $NetBSD: mi,v 1.800 2018/12/24 21:42:05 thorpej Exp $
 #
 # Note: don't delete entries from here - mark them as "obsolete" instead.
 #
@@ -3468,6 +3468,7 @@
 ./usr/tests/rump/rumpkern/t_modlinkset		tests-rump-tests	atf,rump
 ./usr/tests/rump/rumpkern/t_signals		tests-rump-tests	atf,rump
 ./usr/tests/rump/rumpkern/t_sp			tests-rump-tests	atf,rump
+./usr/tests/rump/rumpkern/t_threadpool		tests-rump-tests	atf,rump
 ./usr/tests/rump/rumpkern/t_threads		tests-rump-tests	atf,rump
 ./usr/tests/rump/rumpkern/t_tsleep		tests-rump-tests	atf,rump
 ./usr/tests/rump/rumpkern/t_vm			tests-rump-tests	atf,rump

Index: src/tests/rump/kernspace/Makefile
diff -u src/tests/rump/kernspace/Makefile:1.6 src/tests/rump/kernspace/Makefile:1.7
--- src/tests/rump/kernspace/Makefile:1.6	Fri Sep 29 12:42:36 2017
+++ src/tests/rump/kernspace/Makefile	Mon Dec 24 21:42:05 2018
@@ -1,10 +1,11 @@
-#	$NetBSD: Makefile,v 1.6 2017/09/29 12:42:36 maya Exp $
+#	$NetBSD: Makefile,v 1.7 2018/12/24 21:42:05 thorpej Exp $
 #
 
 .include <bsd.own.mk>
 
 LIB=	kernspace
-SRCS=	thread.c busypage.c tsleep.c alloc.c lockme.c workqueue.c sendsig.c
+SRCS=	thread.c threadpool.c busypage.c tsleep.c alloc.c lockme.c \
+	workqueue.c sendsig.c
 
 RUMPTOP=${NETBSDSRCDIR}/sys/rump
 
Index: src/tests/rump/kernspace/kernspace.h
diff -u src/tests/rump/kernspace/kernspace.h:1.6 src/tests/rump/kernspace/kernspace.h:1.7
--- src/tests/rump/kernspace/kernspace.h:1.6	Thu Dec 28 07:10:25 2017
+++ src/tests/rump/kernspace/kernspace.h	Mon Dec 24 21:42:05 2018
@@ -1,7 +1,7 @@
-/*	$NetBSD: kernspace.h,v 1.6 2017/12/28 07:10:25 ozaki-r Exp $	*/
+/*	$NetBSD: kernspace.h,v 1.7 2018/12/24 21:42:05 thorpej Exp $	*/
 
 /*-
- * Copyright (c) 2010 The NetBSD Foundation, Inc.
+ * Copyright (c) 2010, 2018 The NetBSD Foundation, Inc.
  * All rights reserved.
  *
  * Redistribution and use in source and binary forms, with or without
@@ -46,4 +46,10 @@ void rumptest_workqueue_wait(void);
 void rumptest_sendsig(char *);
 void rumptest_localsig(int);
 
+void rumptest_threadpool_unbound_lifecycle(void);
+void rumptest_threadpool_percpu_lifecycle(void);
+void rumptest_threadpool_unbound_schedule(void);
+void rumptest_threadpool_percpu_schedule(void);
+void rumptest_threadpool_job_cancel(void);
+
 #endif /* _TESTS_RUMP_KERNSPACE_KERNSPACE_H_ */

Index: src/tests/rump/rumpkern/Makefile
diff -u src/tests/rump/rumpkern/Makefile:1.16 src/tests/rump/rumpkern/Makefile:1.17
--- src/tests/rump/rumpkern/Makefile:1.16	Fri Sep 29 12:42:37 2017
+++ src/tests/rump/rumpkern/Makefile	Mon Dec 24 21:42:05 2018
@@ -1,4 +1,4 @@
-# $NetBSD: Makefile,v 1.16 2017/09/29 12:42:37 maya Exp $
+# $NetBSD: Makefile,v 1.17 2018/12/24 21:42:05 thorpej Exp $
 
 .include <bsd.own.mk>
 
@@ -11,6 +11,7 @@ TESTS_C+=	t_modcmd
 TESTS_C+=	t_modlinkset
 TESTS_C+=	t_signals
 TESTS_C+=	t_threads
+TESTS_C+=	t_threadpool
 TESTS_C+=	t_tsleep
 TESTS_C+=	t_workqueue
 TESTS_C+=	t_vm

Added files:

Index: src/tests/rump/kernspace/threadpool.c
diff -u /dev/null src/tests/rump/kernspace/threadpool.c:1.1
--- /dev/null	Mon Dec 24 21:42:05 2018
+++ src/tests/rump/kernspace/threadpool.c	Mon Dec 24 21:42:05 2018
@@ -0,0 +1,234 @@
+/*	$NetBSD: threadpool.c,v 1.1 2018/12/24 21:42:05 thorpej Exp $	*/
+
+/*-
+ * Copyright (c) 2018 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Jason R. Thorpe.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
+ * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
+ * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
+ * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
+ * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
+ * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
+ * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <sys/cdefs.h>
+#if !defined(lint)
+__RCSID("$NetBSD: threadpool.c,v 1.1 2018/12/24 21:42:05 thorpej Exp $");
+#endif /* !lint */
+
+#include <sys/param.h>
+#include <sys/condvar.h>
+#include <sys/kernel.h>
+#include <sys/kmem.h>
+#include <sys/mutex.h>
+#include <sys/threadpool.h>
+
+#include "kernspace.h"
+
+void
+rumptest_threadpool_unbound_lifecycle(void)
+{
+	threadpool_t *pool0, *pool1, *pool2;
+	int error;
+
+	error = threadpool_get(&pool0, PRI_NONE);
+	KASSERT(error == 0);
+
+	error = threadpool_get(&pool1, PRI_NONE);
+	KASSERT(error == 0);
+
+	KASSERT(pool0 == pool1);
+
+	error = threadpool_get(&pool2, PRI_KERNEL_RT);
+	KASSERT(error == 0);
+
+	KASSERT(pool0 != pool2);
+
+	threadpool_put(pool0, PRI_NONE);
+	threadpool_put(pool1, PRI_NONE);
+	threadpool_put(pool2, PRI_KERNEL_RT);
+}
+
+void
+rumptest_threadpool_percpu_lifecycle(void)
+{
+	threadpool_percpu_t *pcpu0, *pcpu1, *pcpu2;
+	int error;
+
+	error = threadpool_percpu_get(&pcpu0, PRI_NONE);
+	KASSERT(error == 0);
+
+	error = threadpool_percpu_get(&pcpu1, PRI_NONE);
+	KASSERT(error == 0);
+
+	KASSERT(pcpu0 == pcpu1);
+
+	error = threadpool_percpu_get(&pcpu2, PRI_KERNEL_RT);
+	KASSERT(error == 0);
+
+	KASSERT(pcpu0 != pcpu2);
+
+	threadpool_percpu_put(pcpu0, PRI_NONE);
+	threadpool_percpu_put(pcpu1, PRI_NONE);
+	threadpool_percpu_put(pcpu2, PRI_KERNEL_RT);
+}
+
+struct test_job_data {
+	kmutex_t mutex;
+	kcondvar_t cond;
+	unsigned int count;
+	threadpool_job_t job;
+};
+
+#define	FINAL_COUNT	12345
+
+static void
+test_job_func_schedule(threadpool_job_t *job)
+{
+	struct test_job_data *data =
+	    container_of(job, struct test_job_data, job);
+	
+	mutex_enter(&data->mutex);
+	KASSERT(data->count != FINAL_COUNT);
+	data->count++;
+	cv_broadcast(&data->cond);
+	threadpool_job_done(job);
+	mutex_exit(&data->mutex);
+}
+
+static void
+test_job_func_cancel(threadpool_job_t *job)
+{
+	struct test_job_data *data =
+	    container_of(job, struct test_job_data, job);
+	
+	mutex_enter(&data->mutex);
+	data->count = 1;
+	cv_broadcast(&data->cond);
+	while (data->count != FINAL_COUNT - 1)
+		cv_wait(&data->cond, &data->mutex);
+	data->count = FINAL_COUNT;
+	cv_broadcast(&data->cond);
+	threadpool_job_done(job);
+	mutex_exit(&data->mutex);
+}
+
+static void
+init_test_job_data(struct test_job_data *data, threadpool_job_fn_t fn)
+{
+	mutex_init(&data->mutex, MUTEX_DEFAULT, IPL_NONE);
+	cv_init(&data->cond, "testjob");
+	threadpool_job_init(&data->job, fn, &data->mutex, "testjob");
+	data->count = 0;
+}
+
+static void
+fini_test_job_data(struct test_job_data *data)
+{
+	threadpool_job_destroy(&data->job);
+	cv_destroy(&data->cond);
+	mutex_destroy(&data->mutex);
+}
+
+void
+rumptest_threadpool_unbound_schedule(void)
+{
+	struct test_job_data data;
+	threadpool_t *pool;
+	int error;
+
+	error = threadpool_get(&pool, PRI_NONE);
+	KASSERT(error == 0);
+
+	init_test_job_data(&data, test_job_func_schedule);
+
+	mutex_enter(&data.mutex);
+	while (data.count != FINAL_COUNT) {
+		threadpool_schedule_job(pool, &data.job);
+		error = cv_timedwait(&data.cond, &data.mutex, hz * 2);
+		KASSERT(error != EWOULDBLOCK);
+	}
+	mutex_exit(&data.mutex);
+
+	fini_test_job_data(&data);
+
+	threadpool_put(pool, PRI_NONE);
+}
+
+void
+rumptest_threadpool_percpu_schedule(void)
+{
+	struct test_job_data data;
+	threadpool_percpu_t *pcpu;
+	threadpool_t *pool;
+	int error;
+
+	error = threadpool_percpu_get(&pcpu, PRI_NONE);
+	KASSERT(error == 0);
+
+	pool = threadpool_percpu_ref(pcpu);
+
+	init_test_job_data(&data, test_job_func_schedule);
+
+	mutex_enter(&data.mutex);
+	while (data.count != FINAL_COUNT) {
+		threadpool_schedule_job(pool, &data.job);
+		error = cv_timedwait(&data.cond, &data.mutex, hz * 2);
+		KASSERT(error != EWOULDBLOCK);
+	}
+	mutex_exit(&data.mutex);
+
+	fini_test_job_data(&data);
+
+	threadpool_percpu_put(pcpu, PRI_NONE);
+}
+
+void
+rumptest_threadpool_job_cancel(void)
+{
+	struct test_job_data data;
+	threadpool_t *pool;
+	int error;
+	bool rv;
+
+	error = threadpool_get(&pool, PRI_NONE);
+	KASSERT(error == 0);
+
+	init_test_job_data(&data, test_job_func_cancel);
+
+	mutex_enter(&data.mutex);
+	threadpool_schedule_job(pool, &data.job);
+	while (data.count == 0)
+		cv_wait(&data.cond, &data.mutex);
+	KASSERT(data.count == 1);
+
+	/* Job is already running (and is not finished); this shold fail. */
+	rv = threadpool_cancel_job_async(pool, &data.job);
+	KASSERT(rv == false);
+
+	data.count = FINAL_COUNT - 1;
+	cv_broadcast(&data.cond);
+	
+	/* Now wait for the job to finish. */
+	threadpool_cancel_job(pool, &data.job);
+	KASSERT(data.count == FINAL_COUNT);
+}

Index: src/tests/rump/rumpkern/t_threadpool.c
diff -u /dev/null src/tests/rump/rumpkern/t_threadpool.c:1.1
--- /dev/null	Mon Dec 24 21:42:05 2018
+++ src/tests/rump/rumpkern/t_threadpool.c	Mon Dec 24 21:42:05 2018
@@ -0,0 +1,138 @@
+/*	$NetBSD: t_threadpool.c,v 1.1 2018/12/24 21:42:05 thorpej Exp $	*/
+
+/*-
+ * Copyright (c) 2018 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
+ * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
+ * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
+ * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
+ * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
+ * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
+ * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <sys/types.h>
+#include <sys/mount.h>
+#include <sys/sysctl.h>
+
+#include <rump/rump.h>
+
+#include <atf-c.h>
+
+#include "h_macros.h"
+#include "../kernspace/kernspace.h"
+
+ATF_TC(threadpool_unbound_lifecycle);
+ATF_TC_HEAD(threadpool_unbound_lifecycle, tc)
+{
+
+	atf_tc_set_md_var(tc, "descr", "Tests unbound threadpool lifecycle");
+}
+
+ATF_TC_BODY(threadpool_unbound_lifecycle, tc)
+{
+
+	rump_init();
+
+	rump_schedule();
+	rumptest_threadpool_unbound_lifecycle(); /* panics if fails */
+	rump_unschedule();
+}
+
+ATF_TC(threadpool_percpu_lifecycle);
+ATF_TC_HEAD(threadpool_percpu_lifecycle, tc)
+{
+
+	atf_tc_set_md_var(tc, "descr", "Tests percpu threadpool lifecycle");
+}
+
+ATF_TC_BODY(threadpool_percpu_lifecycle, tc)
+{
+
+	rump_init();
+
+	rump_schedule();
+	rumptest_threadpool_percpu_lifecycle(); /* panics if fails */
+	rump_unschedule();
+}
+
+ATF_TC(threadpool_unbound_schedule);
+ATF_TC_HEAD(threadpool_unbound_schedule, tc)
+{
+
+	atf_tc_set_md_var(tc, "descr",
+	    "Tests scheduling on unbound threadpools");
+}
+
+ATF_TC_BODY(threadpool_unbound_schedule, tc)
+{
+
+	rump_init();
+
+	rump_schedule();
+	rumptest_threadpool_unbound_schedule(); /* panics if fails */
+	rump_unschedule();
+}
+
+ATF_TC(threadpool_percpu_schedule);
+ATF_TC_HEAD(threadpool_percpu_schedule, tc)
+{
+
+	atf_tc_set_md_var(tc, "descr",
+	    "Tests scheduling on percpu threadpools");
+}
+
+ATF_TC_BODY(threadpool_percpu_schedule, tc)
+{
+
+	rump_init();
+
+	rump_schedule();
+	rumptest_threadpool_percpu_schedule(); /* panics if fails */
+	rump_unschedule();
+}
+
+ATF_TC(threadpool_job_cancel);
+ATF_TC_HEAD(threadpool_job_cancel, tc)
+{
+
+	atf_tc_set_md_var(tc, "descr",
+	    "Tests synchronizing with job cancellation");
+}
+
+ATF_TC_BODY(threadpool_job_cancel, tc)
+{
+
+	rump_init();
+
+	rump_schedule();
+	rumptest_threadpool_job_cancel(); /* panics if fails */
+	rump_unschedule();
+}
+
+ATF_TP_ADD_TCS(tp)
+{
+	ATF_TP_ADD_TC(tp, threadpool_unbound_lifecycle);
+	ATF_TP_ADD_TC(tp, threadpool_percpu_lifecycle);
+	ATF_TP_ADD_TC(tp, threadpool_unbound_schedule);
+	ATF_TP_ADD_TC(tp, threadpool_percpu_schedule);
+	ATF_TP_ADD_TC(tp, threadpool_job_cancel);
+
+	return atf_no_error();
+}

Reply via email to