xref: /dflybsd-src/sys/kern/subr_cpuhelper.c (revision 0e4ac8cf74f44ec920cf9cdb23119f3ff3ede310)
1*0e4ac8cfSSepherosa Ziehau /*
2*0e4ac8cfSSepherosa Ziehau  * Copyright (c) 2017 The DragonFly Project.  All rights reserved.
3*0e4ac8cfSSepherosa Ziehau  *
4*0e4ac8cfSSepherosa Ziehau  * This code is derived from software contributed to The DragonFly Project
5*0e4ac8cfSSepherosa Ziehau  * by Sepherosa Ziehau <sepherosa@gmail.com>
6*0e4ac8cfSSepherosa Ziehau  *
7*0e4ac8cfSSepherosa Ziehau  * Redistribution and use in source and binary forms, with or without
8*0e4ac8cfSSepherosa Ziehau  * modification, are permitted provided that the following conditions
9*0e4ac8cfSSepherosa Ziehau  * are met:
10*0e4ac8cfSSepherosa Ziehau  *
11*0e4ac8cfSSepherosa Ziehau  * 1. Redistributions of source code must retain the above copyright
12*0e4ac8cfSSepherosa Ziehau  *    notice, this list of conditions and the following disclaimer.
13*0e4ac8cfSSepherosa Ziehau  * 2. Redistributions in binary form must reproduce the above copyright
14*0e4ac8cfSSepherosa Ziehau  *    notice, this list of conditions and the following disclaimer in
15*0e4ac8cfSSepherosa Ziehau  *    the documentation and/or other materials provided with the
16*0e4ac8cfSSepherosa Ziehau  *    distribution.
17*0e4ac8cfSSepherosa Ziehau  * 3. Neither the name of The DragonFly Project nor the names of its
18*0e4ac8cfSSepherosa Ziehau  *    contributors may be used to endorse or promote products derived
19*0e4ac8cfSSepherosa Ziehau  *    from this software without specific, prior written permission.
20*0e4ac8cfSSepherosa Ziehau  *
21*0e4ac8cfSSepherosa Ziehau  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22*0e4ac8cfSSepherosa Ziehau  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23*0e4ac8cfSSepherosa Ziehau  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24*0e4ac8cfSSepherosa Ziehau  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
25*0e4ac8cfSSepherosa Ziehau  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26*0e4ac8cfSSepherosa Ziehau  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27*0e4ac8cfSSepherosa Ziehau  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28*0e4ac8cfSSepherosa Ziehau  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29*0e4ac8cfSSepherosa Ziehau  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30*0e4ac8cfSSepherosa Ziehau  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31*0e4ac8cfSSepherosa Ziehau  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32*0e4ac8cfSSepherosa Ziehau  * SUCH DAMAGE.
33*0e4ac8cfSSepherosa Ziehau  */
34*0e4ac8cfSSepherosa Ziehau 
35*0e4ac8cfSSepherosa Ziehau #include <sys/param.h>
36*0e4ac8cfSSepherosa Ziehau #include <sys/systm.h>
37*0e4ac8cfSSepherosa Ziehau #include <sys/kernel.h>
38*0e4ac8cfSSepherosa Ziehau #include <sys/thread.h>
39*0e4ac8cfSSepherosa Ziehau 
40*0e4ac8cfSSepherosa Ziehau #include <sys/cpuhelper.h>
41*0e4ac8cfSSepherosa Ziehau 
42*0e4ac8cfSSepherosa Ziehau static struct thread	*cpuhelper[MAXCPU];
43*0e4ac8cfSSepherosa Ziehau static int		(*cpuhelper_saved_putport)(lwkt_port_t, lwkt_msg_t);
44*0e4ac8cfSSepherosa Ziehau 
45*0e4ac8cfSSepherosa Ziehau static __inline lwkt_port_t
cpuhelper_port(int cpuid)46*0e4ac8cfSSepherosa Ziehau cpuhelper_port(int cpuid)
47*0e4ac8cfSSepherosa Ziehau {
48*0e4ac8cfSSepherosa Ziehau 
49*0e4ac8cfSSepherosa Ziehau 	return (&cpuhelper[cpuid]->td_msgport);
50*0e4ac8cfSSepherosa Ziehau }
51*0e4ac8cfSSepherosa Ziehau 
52*0e4ac8cfSSepherosa Ziehau /*
53*0e4ac8cfSSepherosa Ziehau  * To prevent deadlocking, we must execute these self-referential messages
54*0e4ac8cfSSepherosa Ziehau  * synchronously, effectively turning the message into a direct procedure
55*0e4ac8cfSSepherosa Ziehau  * call.
56*0e4ac8cfSSepherosa Ziehau  */
57*0e4ac8cfSSepherosa Ziehau static int
cpuhelper_putport(lwkt_port_t port,lwkt_msg_t lmsg)58*0e4ac8cfSSepherosa Ziehau cpuhelper_putport(lwkt_port_t port, lwkt_msg_t lmsg)
59*0e4ac8cfSSepherosa Ziehau {
60*0e4ac8cfSSepherosa Ziehau 	struct cpuhelper_msg *msg = (struct cpuhelper_msg *)lmsg;
61*0e4ac8cfSSepherosa Ziehau 
62*0e4ac8cfSSepherosa Ziehau 
63*0e4ac8cfSSepherosa Ziehau 	if ((lmsg->ms_flags & MSGF_SYNC) && port == &curthread->td_msgport) {
64*0e4ac8cfSSepherosa Ziehau 		msg->ch_cb(msg);
65*0e4ac8cfSSepherosa Ziehau 		return (EASYNC);
66*0e4ac8cfSSepherosa Ziehau 	} else {
67*0e4ac8cfSSepherosa Ziehau 		return (cpuhelper_saved_putport(port, lmsg));
68*0e4ac8cfSSepherosa Ziehau 	}
69*0e4ac8cfSSepherosa Ziehau }
70*0e4ac8cfSSepherosa Ziehau 
71*0e4ac8cfSSepherosa Ziehau static void
cpuhelper_mainloop(void * arg __unused)72*0e4ac8cfSSepherosa Ziehau cpuhelper_mainloop(void *arg __unused)
73*0e4ac8cfSSepherosa Ziehau {
74*0e4ac8cfSSepherosa Ziehau 	struct cpuhelper_msg *msg;
75*0e4ac8cfSSepherosa Ziehau 
76*0e4ac8cfSSepherosa Ziehau 
77*0e4ac8cfSSepherosa Ziehau 	while ((msg = lwkt_waitport(&curthread->td_msgport, 0))) {
78*0e4ac8cfSSepherosa Ziehau 		KASSERT(msg->ch_cb != NULL, ("cpuhelper%d: badmsg", mycpuid));
79*0e4ac8cfSSepherosa Ziehau 		msg->ch_cb(msg);
80*0e4ac8cfSSepherosa Ziehau 	}
81*0e4ac8cfSSepherosa Ziehau 	/* NEVER REACHED */
82*0e4ac8cfSSepherosa Ziehau }
83*0e4ac8cfSSepherosa Ziehau 
84*0e4ac8cfSSepherosa Ziehau void
cpuhelper_initmsg(struct cpuhelper_msg * msg,lwkt_port_t rport,cpuhelper_cb_t cb,void * cbarg,int flags)85*0e4ac8cfSSepherosa Ziehau cpuhelper_initmsg(struct cpuhelper_msg *msg, lwkt_port_t rport,
86*0e4ac8cfSSepherosa Ziehau     cpuhelper_cb_t cb, void *cbarg, int flags)
87*0e4ac8cfSSepherosa Ziehau {
88*0e4ac8cfSSepherosa Ziehau 
89*0e4ac8cfSSepherosa Ziehau 	lwkt_initmsg(&msg->ch_lmsg, rport, flags);
90*0e4ac8cfSSepherosa Ziehau 	msg->ch_cb = cb;
91*0e4ac8cfSSepherosa Ziehau 	msg->ch_cbarg = cbarg;
92*0e4ac8cfSSepherosa Ziehau 	msg->ch_cbarg1 = 0;
93*0e4ac8cfSSepherosa Ziehau }
94*0e4ac8cfSSepherosa Ziehau 
95*0e4ac8cfSSepherosa Ziehau void
cpuhelper_replymsg(struct cpuhelper_msg * msg,int error)96*0e4ac8cfSSepherosa Ziehau cpuhelper_replymsg(struct cpuhelper_msg *msg, int error)
97*0e4ac8cfSSepherosa Ziehau {
98*0e4ac8cfSSepherosa Ziehau 
99*0e4ac8cfSSepherosa Ziehau 	lwkt_replymsg(&msg->ch_lmsg, error);
100*0e4ac8cfSSepherosa Ziehau }
101*0e4ac8cfSSepherosa Ziehau 
102*0e4ac8cfSSepherosa Ziehau int
cpuhelper_domsg(struct cpuhelper_msg * msg,int cpuid)103*0e4ac8cfSSepherosa Ziehau cpuhelper_domsg(struct cpuhelper_msg *msg, int cpuid)
104*0e4ac8cfSSepherosa Ziehau {
105*0e4ac8cfSSepherosa Ziehau 
106*0e4ac8cfSSepherosa Ziehau 	KASSERT(cpuid >= 0 && cpuid < ncpus, ("invalid cpuid"));
107*0e4ac8cfSSepherosa Ziehau 	return (lwkt_domsg(cpuhelper_port(cpuid), &msg->ch_lmsg, 0));
108*0e4ac8cfSSepherosa Ziehau }
109*0e4ac8cfSSepherosa Ziehau 
110*0e4ac8cfSSepherosa Ziehau void
cpuhelper_assert(int cpuid,bool in)111*0e4ac8cfSSepherosa Ziehau cpuhelper_assert(int cpuid, bool in)
112*0e4ac8cfSSepherosa Ziehau {
113*0e4ac8cfSSepherosa Ziehau 
114*0e4ac8cfSSepherosa Ziehau #ifdef INVARIANTS
115*0e4ac8cfSSepherosa Ziehau 	KASSERT(cpuid >= 0 && cpuid < ncpus, ("invalid cpuid"));
116*0e4ac8cfSSepherosa Ziehau 	if (in) {
117*0e4ac8cfSSepherosa Ziehau 		KASSERT(&curthread->td_msgport == cpuhelper_port(cpuid),
118*0e4ac8cfSSepherosa Ziehau 		    ("not in cpuhelper%d", cpuid));
119*0e4ac8cfSSepherosa Ziehau 	} else {
120*0e4ac8cfSSepherosa Ziehau 		KASSERT(&curthread->td_msgport != cpuhelper_port(cpuid),
121*0e4ac8cfSSepherosa Ziehau 		    ("in cpuhelper%d", cpuid));
122*0e4ac8cfSSepherosa Ziehau 	}
123*0e4ac8cfSSepherosa Ziehau #endif
124*0e4ac8cfSSepherosa Ziehau }
125*0e4ac8cfSSepherosa Ziehau 
126*0e4ac8cfSSepherosa Ziehau static void
cpuhelper_init(void)127*0e4ac8cfSSepherosa Ziehau cpuhelper_init(void)
128*0e4ac8cfSSepherosa Ziehau {
129*0e4ac8cfSSepherosa Ziehau 	int i;
130*0e4ac8cfSSepherosa Ziehau 
131*0e4ac8cfSSepherosa Ziehau 	/*
132*0e4ac8cfSSepherosa Ziehau 	 * Create per-cpu helper threads.
133*0e4ac8cfSSepherosa Ziehau 	 */
134*0e4ac8cfSSepherosa Ziehau 	for (i = 0; i < ncpus; ++i) {
135*0e4ac8cfSSepherosa Ziehau 		lwkt_port_t port;
136*0e4ac8cfSSepherosa Ziehau 
137*0e4ac8cfSSepherosa Ziehau 		lwkt_create(cpuhelper_mainloop, NULL, &cpuhelper[i], NULL,
138*0e4ac8cfSSepherosa Ziehau 		    TDF_NOSTART|TDF_FORCE_SPINPORT|TDF_FIXEDCPU, i,
139*0e4ac8cfSSepherosa Ziehau 		    "cpuhelper %d", i);
140*0e4ac8cfSSepherosa Ziehau 
141*0e4ac8cfSSepherosa Ziehau 		/*
142*0e4ac8cfSSepherosa Ziehau 		 * Override the putport function.  Our custom function checks
143*0e4ac8cfSSepherosa Ziehau 		 * for self-references.
144*0e4ac8cfSSepherosa Ziehau 		 */
145*0e4ac8cfSSepherosa Ziehau 		port = cpuhelper_port(i);
146*0e4ac8cfSSepherosa Ziehau 		if (cpuhelper_saved_putport == NULL)
147*0e4ac8cfSSepherosa Ziehau 			cpuhelper_saved_putport = port->mp_putport;
148*0e4ac8cfSSepherosa Ziehau 		KKASSERT(cpuhelper_saved_putport == port->mp_putport);
149*0e4ac8cfSSepherosa Ziehau 		port->mp_putport = cpuhelper_putport;
150*0e4ac8cfSSepherosa Ziehau 
151*0e4ac8cfSSepherosa Ziehau 		lwkt_schedule(cpuhelper[i]);
152*0e4ac8cfSSepherosa Ziehau 	}
153*0e4ac8cfSSepherosa Ziehau }
154*0e4ac8cfSSepherosa Ziehau SYSINIT(cpuhelper, SI_SUB_PRE_DRIVERS, SI_ORDER_FIRST, cpuhelper_init, NULL);
155