Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/external/cddl/osnet This implementation of Solaris taskq API...
details: https://anonhg.NetBSD.org/src/rev/825e8dfdc0f4
branches: trunk
changeset: 456408:825e8dfdc0f4
user: hannken <hannken%NetBSD.org@localhost>
date: Tue May 07 08:49:59 2019 +0000
description:
This implementation of Solaris taskq API is incomplete and doesn't track
Solaris upstream. FreeBSD already replaced it with a glue to their
taskqueue API.
Replace it with a glue component that queues Solaris taskq requests to
threadpool jobs.
diffstat:
external/cddl/osnet/dist/uts/common/fs/zfs/spa.c | 2 +-
external/cddl/osnet/dist/uts/common/fs/zfs/zio.c | 8 +-
external/cddl/osnet/dist/uts/common/sys/taskq.h | 12 +-
external/cddl/osnet/sys/kern/taskq.c | 1743 ++-------------------
external/cddl/osnet/sys/sys/zfs_context.h | 9 +-
5 files changed, 255 insertions(+), 1519 deletions(-)
diffs (truncated from 1932 to 300 lines):
diff -r 87c8bff4a355 -r 825e8dfdc0f4 external/cddl/osnet/dist/uts/common/fs/zfs/spa.c
--- a/external/cddl/osnet/dist/uts/common/fs/zfs/spa.c Tue May 07 08:14:59 2019 +0000
+++ b/external/cddl/osnet/dist/uts/common/fs/zfs/spa.c Tue May 07 08:49:59 2019 +0000
@@ -924,7 +924,7 @@
* then a difference between them is insignificant.
*/
if (t == ZIO_TYPE_WRITE && q == ZIO_TASKQ_ISSUE)
-#ifdef illumos
+#if defined(illumos) || defined(__NetBSD__)
pri--;
#else
pri += 4;
diff -r 87c8bff4a355 -r 825e8dfdc0f4 external/cddl/osnet/dist/uts/common/fs/zfs/zio.c
--- a/external/cddl/osnet/dist/uts/common/fs/zfs/zio.c Tue May 07 08:14:59 2019 +0000
+++ b/external/cddl/osnet/dist/uts/common/fs/zfs/zio.c Tue May 07 08:49:59 2019 +0000
@@ -1537,8 +1537,10 @@
* to a single taskq at a time. It would be a grievous error
* to dispatch the zio to another taskq at the same time.
*/
-#if defined(illumos) || defined(__NetBSD__) || !defined(_KERNEL)
+#if defined(illumos) || !defined(_KERNEL)
ASSERT(zio->io_tqent.tqent_next == NULL);
+#elif defined(__NetBSD__)
+ ASSERT(zio->io_tqent.tqent_queued == 0);
#else
ASSERT(zio->io_tqent.tqent_task.ta_pending == 0);
#endif
@@ -3842,8 +3844,10 @@
* Reexecution is potentially a huge amount of work.
* Hand it off to the otherwise-unused claim taskq.
*/
-#if defined(illumos) || defined(__NetBSD__) || !defined(_KERNEL)
+#if defined(illumos) || !defined(_KERNEL)
ASSERT(zio->io_tqent.tqent_next == NULL);
+#elif defined(__NetBSD__)
+ ASSERT(zio->io_tqent.tqent_queued == 0);
#else
ASSERT(zio->io_tqent.tqent_task.ta_pending == 0);
#endif
diff -r 87c8bff4a355 -r 825e8dfdc0f4 external/cddl/osnet/dist/uts/common/sys/taskq.h
--- a/external/cddl/osnet/dist/uts/common/sys/taskq.h Tue May 07 08:14:59 2019 +0000
+++ b/external/cddl/osnet/dist/uts/common/sys/taskq.h Tue May 07 08:49:59 2019 +0000
@@ -54,6 +54,16 @@
typedef uintptr_t taskqid_t;
typedef void (task_func_t)(void *);
+#ifdef __NetBSD__
+typedef struct taskq_ent {
+ SIMPLEQ_ENTRY(taskq_ent) tqent_list; /* Task queue. */
+ task_func_t *tqent_func; /* Function to run. */
+ void *tqent_arg; /* Argument to function above. */
+ unsigned tqent_dynamic:1; /* Must kmem_free() if true. */
+ unsigned tqent_queued:1; /* Queued and waiting to run if true. */
+} taskq_ent_t;
+#endif
+
struct proc;
/*
@@ -89,7 +99,7 @@
taskq_t *taskq_create_sysdc(const char *, int, int, int,
struct proc *, uint_t, uint_t);
taskqid_t taskq_dispatch(taskq_t *, task_func_t, void *, uint_t);
-#ifdef __FreeBSD__
+#if defined(__FreeBSD__) || defined(__NetBSD__)
void taskq_dispatch_ent(taskq_t *, task_func_t, void *, uint_t,
taskq_ent_t *);
#endif
diff -r 87c8bff4a355 -r 825e8dfdc0f4 external/cddl/osnet/sys/kern/taskq.c
--- a/external/cddl/osnet/sys/kern/taskq.c Tue May 07 08:14:59 2019 +0000
+++ b/external/cddl/osnet/sys/kern/taskq.c Tue May 07 08:49:59 2019 +0000
@@ -1,1594 +1,323 @@
-/* $NetBSD: taskq.c,v 1.8 2019/01/12 10:42:40 hannken Exp $ */
+/* $NetBSD: taskq.c,v 1.9 2019/05/07 08:49:59 hannken Exp $ */
-/*
- * CDDL HEADER START
+/*-
+ * Copyright (c) 2019 The NetBSD Foundation, Inc.
+ * All rights reserved.
*
- * The contents of this file are subject to the terms of the
- * Common Development and Distribution License, Version 1.0 only
- * (the "License"). You may not use this file except in compliance
- * with the License.
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Juergen Hannken-Illjes.
*
- * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
- * or http://www.opensolaris.org/os/licensing.
- * See the License for the specific language governing permissions
- * and limitations under the License.
+ * 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.
*
- * When distributing Covered Code, include this CDDL HEADER in each
- * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
- * If applicable, add the following below this CDDL HEADER, with the
- * fields enclosed by brackets "[]" replaced with your own identifying
- * information: Portions Copyright [yyyy] [name of copyright owner]
- *
- * CDDL HEADER END
- */
-/*
- * Copyright 2005 Sun Microsystems, Inc. All rights reserved.
- * Use is subject to license terms.
+ * 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.
*/
-/*
- * Copyright 2015 Nexenta Systems, Inc. All rights reserved.
- */
+#include <sys/types.h>
+#include <sys/mutex.h>
+#include <sys/kcondvar.h>
+#include <sys/kmem.h>
+#include <sys/proc.h>
+#include <sys/threadpool.h>
+
+#include <sys/taskq.h>
+
+struct taskq_executor {
+ struct threadpool_job te_job; /* Threadpool job serving the queue. */
+ taskq_t *te_self; /* Backpointer to the queue. */
+ unsigned te_running:1; /* True if the job is running. */
+};
+
+struct taskq {
+ int tq_nthreads; /* # of threads serving queue. */
+ pri_t tq_pri; /* Scheduling priority. */
+ uint_t tq_flags; /* Saved flags from taskq_create. */
+ int tq_active; /* # of tasks (queued or running). */
+ int tq_running; /* # of jobs currently running. */
+ int tq_waiting; /* # of jobs currently idle. */
+ unsigned tq_destroyed:1; /* True if queue gets destroyed. */
+ kmutex_t tq_lock; /* Queue and job lock. */
+ kcondvar_t tq_cv; /* Queue condvar. */
+ struct taskq_executor *tq_executor; /* Array of jobs. */
+ struct threadpool *tq_threadpool; /* Pool backing the jobs. */
+ SIMPLEQ_HEAD(, taskq_ent) tq_list; /* Queue of tasks waiting. */
+};
+
+taskq_t *system_taskq; /* General purpose task queue. */
+
+static specificdata_key_t taskq_lwp_key; /* Null or taskq this thread runs. */
/*
- * Kernel task queues: general-purpose asynchronous task scheduling.
- *
- * A common problem in kernel programming is the need to schedule tasks
- * to be performed later, by another thread. There are several reasons
- * you may want or need to do this:
- *
- * (1) The task isn't time-critical, but your current code path is.
- *
- * (2) The task may require grabbing locks that you already hold.
- *
- * (3) The task may need to block (e.g. to wait for memory), but you
- * cannot block in your current context.
- *
- * (4) Your code path can't complete because of some condition, but you can't
- * sleep or fail, so you queue the task for later execution when condition
- * disappears.
- *
- * (5) You just want a simple way to launch multiple tasks in parallel.
- *
- * Task queues provide such a facility. In its simplest form (used when
- * performance is not a critical consideration) a task queue consists of a
- * single list of tasks, together with one or more threads to service the
- * list. There are some cases when this simple queue is not sufficient:
- *
- * (1) The task queues are very hot and there is a need to avoid data and lock
- * contention over global resources.
- *
- * (2) Some tasks may depend on other tasks to complete, so they can't be put in
- * the same list managed by the same thread.
- *
- * (3) Some tasks may block for a long time, and this should not block other
- * tasks in the queue.
- *
- * To provide useful service in such cases we define a "dynamic task queue"
- * which has an individual thread for each of the tasks. These threads are
- * dynamically created as they are needed and destroyed when they are not in
- * use. The API for managing task pools is the same as for managing task queues
- * with the exception of a taskq creation flag TASKQ_DYNAMIC which tells that
- * dynamic task pool behavior is desired.
- *
- * Dynamic task queues may also place tasks in the normal queue (called "backing
- * queue") when task pool runs out of resources. Users of task queues may
- * disallow such queued scheduling by specifying TQ_NOQUEUE in the dispatch
- * flags.
- *
- * The backing task queue is also used for scheduling internal tasks needed for
- * dynamic task queue maintenance.
- *
- * INTERFACES ==================================================================
- *
- * taskq_t *taskq_create(name, nthreads, pri, minalloc, maxalloc, flags);
- *
- * Create a taskq with specified properties.
- * Possible 'flags':
- *
- * TASKQ_DYNAMIC: Create task pool for task management. If this flag is
- * specified, 'nthreads' specifies the maximum number of threads in
- * the task queue. Task execution order for dynamic task queues is
- * not predictable.
- *
- * If this flag is not specified (default case) a
- * single-list task queue is created with 'nthreads' threads
- * servicing it. Entries in this queue are managed by
- * taskq_ent_alloc() and taskq_ent_free() which try to keep the
- * task population between 'minalloc' and 'maxalloc', but the
- * latter limit is only advisory for TQ_SLEEP dispatches and the
- * former limit is only advisory for TQ_NOALLOC dispatches. If
- * TASKQ_PREPOPULATE is set in 'flags', the taskq will be
- * prepopulated with 'minalloc' task structures.
- *
- * Since non-DYNAMIC taskqs are queues, tasks are guaranteed to be
- * executed in the order they are scheduled if nthreads == 1.
- * If nthreads > 1, task execution order is not predictable.
- *
- * TASKQ_PREPOPULATE: Prepopulate task queue with threads.
- * Also prepopulate the task queue with 'minalloc' task structures.
- *
- * TASKQ_THREADS_CPU_PCT: This flag specifies that 'nthreads' should be
- * interpreted as a percentage of the # of online CPUs on the
- * system. The taskq subsystem will automatically adjust the
- * number of threads in the taskq in response to CPU online
- * and offline events, to keep the ratio. nthreads must be in
- * the range [0,100].
- *
- * The calculation used is:
- *
- * MAX((ncpus_online * percentage)/100, 1)
- *
- * This flag is not supported for DYNAMIC task queues.
- * This flag is not compatible with TASKQ_CPR_SAFE.
- *
- * TASKQ_CPR_SAFE: This flag specifies that users of the task queue will
- * use their own protocol for handling CPR issues. This flag is not
- * supported for DYNAMIC task queues. This flag is not compatible
- * with TASKQ_THREADS_CPU_PCT.
- *
- * The 'pri' field specifies the default priority for the threads that
- * service all scheduled tasks.
- *
- * taskq_t *taskq_create_instance(name, instance, nthreads, pri, minalloc,
- * maxalloc, flags);
- *
- * Like taskq_create(), but takes an instance number (or -1 to indicate
- * no instance).
- *
- * taskq_t *taskq_create_proc(name, nthreads, pri, minalloc, maxalloc, proc,
- * flags);
- *
- * Like taskq_create(), but creates the taskq threads in the specified
- * system process. If proc != &p0, this must be called from a thread
- * in that process.
- *
- * taskq_t *taskq_create_sysdc(name, nthreads, minalloc, maxalloc, proc,
- * dc, flags);
- *
- * Like taskq_create_proc(), but the taskq threads will use the
- * System Duty Cycle (SDC) scheduling class with a duty cycle of dc.
- *
- * void taskq_destroy(tap):
- *
- * Waits for any scheduled tasks to complete, then destroys the taskq.
- * Caller should guarantee that no new tasks are scheduled in the closing
- * taskq.
- *
- * taskqid_t taskq_dispatch(tq, func, arg, flags):
- *
- * Dispatches the task "func(arg)" to taskq. The 'flags' indicates whether
- * the caller is willing to block for memory. The function returns an
- * opaque value which is zero iff dispatch fails. If flags is TQ_NOSLEEP
- * or TQ_NOALLOC and the task can't be dispatched, taskq_dispatch() fails
- * and returns (taskqid_t)0.
- *
- * ASSUMES: func != NULL.
- *
- * Possible flags:
- * TQ_NOSLEEP: Do not wait for resources; may fail.
- *
Home |
Main Index |
Thread Index |
Old Index