xref: /freebsd/sys/contrib/openzfs/lib/libspl/condvar.c (revision 8ac904ce090b1c2e355da8aa122ca2252183f4e1)
1*8ac904ceSMartin Matuska // SPDX-License-Identifier: CDDL-1.0
2*8ac904ceSMartin Matuska /*
3*8ac904ceSMartin Matuska  * CDDL HEADER START
4*8ac904ceSMartin Matuska  *
5*8ac904ceSMartin Matuska  * The contents of this file are subject to the terms of the
6*8ac904ceSMartin Matuska  * Common Development and Distribution License (the "License").
7*8ac904ceSMartin Matuska  * You may not use this file except in compliance with the License.
8*8ac904ceSMartin Matuska  *
9*8ac904ceSMartin Matuska  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10*8ac904ceSMartin Matuska  * or https://opensource.org/licenses/CDDL-1.0.
11*8ac904ceSMartin Matuska  * See the License for the specific language governing permissions
12*8ac904ceSMartin Matuska  * and limitations under the License.
13*8ac904ceSMartin Matuska  *
14*8ac904ceSMartin Matuska  * When distributing Covered Code, include this CDDL HEADER in each
15*8ac904ceSMartin Matuska  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16*8ac904ceSMartin Matuska  * If applicable, add the following below this CDDL HEADER, with the
17*8ac904ceSMartin Matuska  * fields enclosed by brackets "[]" replaced with your own identifying
18*8ac904ceSMartin Matuska  * information: Portions Copyright [yyyy] [name of copyright owner]
19*8ac904ceSMartin Matuska  *
20*8ac904ceSMartin Matuska  * CDDL HEADER END
21*8ac904ceSMartin Matuska  */
22*8ac904ceSMartin Matuska /*
23*8ac904ceSMartin Matuska  * Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
24*8ac904ceSMartin Matuska  * Copyright (c) 2012, 2018 by Delphix. All rights reserved.
25*8ac904ceSMartin Matuska  * Copyright (c) 2016 Actifio, Inc. All rights reserved.
26*8ac904ceSMartin Matuska  * Copyright (c) 2025, Klara, Inc.
27*8ac904ceSMartin Matuska  */
28*8ac904ceSMartin Matuska 
29*8ac904ceSMartin Matuska #include <assert.h>
30*8ac904ceSMartin Matuska #include <pthread.h>
31*8ac904ceSMartin Matuska #include <errno.h>
32*8ac904ceSMartin Matuska #include <string.h>
33*8ac904ceSMartin Matuska #include <sys/timer.h>
34*8ac904ceSMartin Matuska #include <sys/condvar.h>
35*8ac904ceSMartin Matuska 
36*8ac904ceSMartin Matuska /*
37*8ac904ceSMartin Matuska  * =========================================================================
38*8ac904ceSMartin Matuska  * condition variables
39*8ac904ceSMartin Matuska  * =========================================================================
40*8ac904ceSMartin Matuska  */
41*8ac904ceSMartin Matuska 
42*8ac904ceSMartin Matuska void
cv_init(kcondvar_t * cv,char * name,int type,void * arg)43*8ac904ceSMartin Matuska cv_init(kcondvar_t *cv, char *name, int type, void *arg)
44*8ac904ceSMartin Matuska {
45*8ac904ceSMartin Matuska 	(void) name, (void) type, (void) arg;
46*8ac904ceSMartin Matuska 	VERIFY0(pthread_cond_init(cv, NULL));
47*8ac904ceSMartin Matuska }
48*8ac904ceSMartin Matuska 
49*8ac904ceSMartin Matuska void
cv_destroy(kcondvar_t * cv)50*8ac904ceSMartin Matuska cv_destroy(kcondvar_t *cv)
51*8ac904ceSMartin Matuska {
52*8ac904ceSMartin Matuska 	VERIFY0(pthread_cond_destroy(cv));
53*8ac904ceSMartin Matuska }
54*8ac904ceSMartin Matuska 
55*8ac904ceSMartin Matuska void
cv_wait(kcondvar_t * cv,kmutex_t * mp)56*8ac904ceSMartin Matuska cv_wait(kcondvar_t *cv, kmutex_t *mp)
57*8ac904ceSMartin Matuska {
58*8ac904ceSMartin Matuska 	memset(&mp->m_owner, 0, sizeof (pthread_t));
59*8ac904ceSMartin Matuska 	VERIFY0(pthread_cond_wait(cv, &mp->m_lock));
60*8ac904ceSMartin Matuska 	mp->m_owner = pthread_self();
61*8ac904ceSMartin Matuska }
62*8ac904ceSMartin Matuska 
63*8ac904ceSMartin Matuska int
cv_wait_sig(kcondvar_t * cv,kmutex_t * mp)64*8ac904ceSMartin Matuska cv_wait_sig(kcondvar_t *cv, kmutex_t *mp)
65*8ac904ceSMartin Matuska {
66*8ac904ceSMartin Matuska 	cv_wait(cv, mp);
67*8ac904ceSMartin Matuska 	return (1);
68*8ac904ceSMartin Matuska }
69*8ac904ceSMartin Matuska 
70*8ac904ceSMartin Matuska int
cv_timedwait(kcondvar_t * cv,kmutex_t * mp,clock_t abstime)71*8ac904ceSMartin Matuska cv_timedwait(kcondvar_t *cv, kmutex_t *mp, clock_t abstime)
72*8ac904ceSMartin Matuska {
73*8ac904ceSMartin Matuska 	int error;
74*8ac904ceSMartin Matuska 	struct timeval tv;
75*8ac904ceSMartin Matuska 	struct timespec ts;
76*8ac904ceSMartin Matuska 	clock_t delta;
77*8ac904ceSMartin Matuska 
78*8ac904ceSMartin Matuska 	delta = abstime - ddi_get_lbolt();
79*8ac904ceSMartin Matuska 	if (delta <= 0)
80*8ac904ceSMartin Matuska 		return (-1);
81*8ac904ceSMartin Matuska 
82*8ac904ceSMartin Matuska 	VERIFY0(gettimeofday(&tv, NULL));
83*8ac904ceSMartin Matuska 
84*8ac904ceSMartin Matuska 	ts.tv_sec = tv.tv_sec + delta / hz;
85*8ac904ceSMartin Matuska 	ts.tv_nsec = tv.tv_usec * NSEC_PER_USEC + (delta % hz) * (NANOSEC / hz);
86*8ac904ceSMartin Matuska 	if (ts.tv_nsec >= NANOSEC) {
87*8ac904ceSMartin Matuska 		ts.tv_sec++;
88*8ac904ceSMartin Matuska 		ts.tv_nsec -= NANOSEC;
89*8ac904ceSMartin Matuska 	}
90*8ac904ceSMartin Matuska 
91*8ac904ceSMartin Matuska 	memset(&mp->m_owner, 0, sizeof (pthread_t));
92*8ac904ceSMartin Matuska 	error = pthread_cond_timedwait(cv, &mp->m_lock, &ts);
93*8ac904ceSMartin Matuska 	mp->m_owner = pthread_self();
94*8ac904ceSMartin Matuska 
95*8ac904ceSMartin Matuska 	if (error == ETIMEDOUT)
96*8ac904ceSMartin Matuska 		return (-1);
97*8ac904ceSMartin Matuska 
98*8ac904ceSMartin Matuska 	VERIFY0(error);
99*8ac904ceSMartin Matuska 
100*8ac904ceSMartin Matuska 	return (1);
101*8ac904ceSMartin Matuska }
102*8ac904ceSMartin Matuska 
103*8ac904ceSMartin Matuska int
cv_timedwait_hires(kcondvar_t * cv,kmutex_t * mp,hrtime_t tim,hrtime_t res,int flag)104*8ac904ceSMartin Matuska cv_timedwait_hires(kcondvar_t *cv, kmutex_t *mp, hrtime_t tim, hrtime_t res,
105*8ac904ceSMartin Matuska     int flag)
106*8ac904ceSMartin Matuska {
107*8ac904ceSMartin Matuska 	(void) res;
108*8ac904ceSMartin Matuska 	int error;
109*8ac904ceSMartin Matuska 	struct timeval tv;
110*8ac904ceSMartin Matuska 	struct timespec ts;
111*8ac904ceSMartin Matuska 	hrtime_t delta;
112*8ac904ceSMartin Matuska 
113*8ac904ceSMartin Matuska 	ASSERT(flag == 0 || flag == CALLOUT_FLAG_ABSOLUTE);
114*8ac904ceSMartin Matuska 
115*8ac904ceSMartin Matuska 	delta = tim;
116*8ac904ceSMartin Matuska 	if (flag & CALLOUT_FLAG_ABSOLUTE)
117*8ac904ceSMartin Matuska 		delta -= gethrtime();
118*8ac904ceSMartin Matuska 
119*8ac904ceSMartin Matuska 	if (delta <= 0)
120*8ac904ceSMartin Matuska 		return (-1);
121*8ac904ceSMartin Matuska 
122*8ac904ceSMartin Matuska 	VERIFY0(gettimeofday(&tv, NULL));
123*8ac904ceSMartin Matuska 
124*8ac904ceSMartin Matuska 	ts.tv_sec = tv.tv_sec + delta / NANOSEC;
125*8ac904ceSMartin Matuska 	ts.tv_nsec = tv.tv_usec * NSEC_PER_USEC + (delta % NANOSEC);
126*8ac904ceSMartin Matuska 	if (ts.tv_nsec >= NANOSEC) {
127*8ac904ceSMartin Matuska 		ts.tv_sec++;
128*8ac904ceSMartin Matuska 		ts.tv_nsec -= NANOSEC;
129*8ac904ceSMartin Matuska 	}
130*8ac904ceSMartin Matuska 
131*8ac904ceSMartin Matuska 	memset(&mp->m_owner, 0, sizeof (pthread_t));
132*8ac904ceSMartin Matuska 	error = pthread_cond_timedwait(cv, &mp->m_lock, &ts);
133*8ac904ceSMartin Matuska 	mp->m_owner = pthread_self();
134*8ac904ceSMartin Matuska 
135*8ac904ceSMartin Matuska 	if (error == ETIMEDOUT)
136*8ac904ceSMartin Matuska 		return (-1);
137*8ac904ceSMartin Matuska 
138*8ac904ceSMartin Matuska 	VERIFY0(error);
139*8ac904ceSMartin Matuska 
140*8ac904ceSMartin Matuska 	return (1);
141*8ac904ceSMartin Matuska }
142*8ac904ceSMartin Matuska 
143*8ac904ceSMartin Matuska void
cv_signal(kcondvar_t * cv)144*8ac904ceSMartin Matuska cv_signal(kcondvar_t *cv)
145*8ac904ceSMartin Matuska {
146*8ac904ceSMartin Matuska 	VERIFY0(pthread_cond_signal(cv));
147*8ac904ceSMartin Matuska }
148*8ac904ceSMartin Matuska 
149*8ac904ceSMartin Matuska void
cv_broadcast(kcondvar_t * cv)150*8ac904ceSMartin Matuska cv_broadcast(kcondvar_t *cv)
151*8ac904ceSMartin Matuska {
152*8ac904ceSMartin Matuska 	VERIFY0(pthread_cond_broadcast(cv));
153*8ac904ceSMartin Matuska }
154