]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - lib/libthr/thread/thr_sleepq.c
disable BIND_NOW in libc, libthr, and rtld
[FreeBSD/FreeBSD.git] / lib / libthr / thread / thr_sleepq.c
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2010 David Xu <davidxu@freebsd.org>
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice unmodified, this list of conditions, and the following
12  *    disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28
29 #include <sys/cdefs.h>
30 __FBSDID("$FreeBSD$");
31
32 #include <stdlib.h>
33 #include "thr_private.h"
34
35 #define HASHSHIFT       9
36 #define HASHSIZE        (1 << HASHSHIFT)
37 #define SC_HASH(wchan) ((unsigned)                              \
38         ((((uintptr_t)(wchan) >> 3)                             \
39         ^ ((uintptr_t)(wchan) >> (HASHSHIFT + 3)))              \
40         & (HASHSIZE - 1)))
41 #define SC_LOOKUP(wc)   &sc_table[SC_HASH(wc)]
42
43 struct sleepqueue_chain {
44         struct umutex           sc_lock;
45         int                     sc_enqcnt;
46         LIST_HEAD(, sleepqueue) sc_queues;
47         int                     sc_type;
48 };
49
50 static struct sleepqueue_chain  sc_table[HASHSIZE];
51
52 void
53 _sleepq_init(void)
54 {
55         int     i;
56
57         for (i = 0; i < HASHSIZE; ++i) {
58                 LIST_INIT(&sc_table[i].sc_queues);
59                 _thr_umutex_init(&sc_table[i].sc_lock);
60         }
61 }
62
63 struct sleepqueue *
64 _sleepq_alloc(void)
65 {
66         struct sleepqueue *sq;
67
68         sq = calloc(1, sizeof(struct sleepqueue));
69         TAILQ_INIT(&sq->sq_blocked);
70         SLIST_INIT(&sq->sq_freeq);
71         return (sq);
72 }
73
74 void
75 _sleepq_free(struct sleepqueue *sq)
76 {
77         free(sq);
78 }
79
80 void
81 _sleepq_lock(void *wchan)
82 {
83         struct pthread *curthread = _get_curthread();
84         struct sleepqueue_chain *sc;
85
86         sc = SC_LOOKUP(wchan);
87         THR_LOCK_ACQUIRE_SPIN(curthread, &sc->sc_lock);
88 }
89
90 void
91 _sleepq_unlock(void *wchan)
92 {
93         struct sleepqueue_chain *sc;
94         struct pthread *curthread = _get_curthread();
95                     
96         sc = SC_LOOKUP(wchan);
97         THR_LOCK_RELEASE(curthread, &sc->sc_lock);
98 }
99
100 static inline struct sleepqueue *
101 lookup(struct sleepqueue_chain *sc, void *wchan)
102 {
103         struct sleepqueue *sq;
104
105         LIST_FOREACH(sq, &sc->sc_queues, sq_hash)
106                 if (sq->sq_wchan == wchan)
107                         return (sq);
108         return (NULL);
109 }
110
111 struct sleepqueue *
112 _sleepq_lookup(void *wchan)
113 {
114         return (lookup(SC_LOOKUP(wchan), wchan));
115 }
116
117 void
118 _sleepq_add(void *wchan, struct pthread *td)
119 {
120         struct sleepqueue_chain *sc;
121         struct sleepqueue *sq;
122
123         sc = SC_LOOKUP(wchan);
124         sq = lookup(sc, wchan);
125         if (sq != NULL) {
126                 SLIST_INSERT_HEAD(&sq->sq_freeq, td->sleepqueue, sq_flink);
127         } else {
128                 sq = td->sleepqueue;
129                 LIST_INSERT_HEAD(&sc->sc_queues, sq, sq_hash);
130                 sq->sq_wchan = wchan;
131                 /* sq->sq_type = type; */
132         }
133         td->sleepqueue = NULL;
134         td->wchan = wchan;
135         if (((++sc->sc_enqcnt << _thr_queuefifo) & 0xff) != 0)
136                 TAILQ_INSERT_HEAD(&sq->sq_blocked, td, wle);
137         else
138                 TAILQ_INSERT_TAIL(&sq->sq_blocked, td, wle);
139 }
140
141 int
142 _sleepq_remove(struct sleepqueue *sq, struct pthread *td)
143 {
144         int rc;
145
146         TAILQ_REMOVE(&sq->sq_blocked, td, wle);
147         if (TAILQ_EMPTY(&sq->sq_blocked)) {
148                 LIST_REMOVE(sq, sq_hash);
149                 td->sleepqueue = sq;
150                 rc = 0;
151         } else {
152                 td->sleepqueue = SLIST_FIRST(&sq->sq_freeq);
153                 SLIST_REMOVE_HEAD(&sq->sq_freeq, sq_flink);
154                 rc = 1;
155         }
156         td->wchan = NULL;
157         return (rc);
158 }
159
160 void
161 _sleepq_drop(struct sleepqueue *sq,
162         void (*cb)(struct pthread *, void *arg), void *arg)
163 {
164         struct pthread *td;
165         struct sleepqueue *sq2;
166
167         td = TAILQ_FIRST(&sq->sq_blocked);
168         if (td == NULL)
169                 return;
170         LIST_REMOVE(sq, sq_hash);
171         TAILQ_REMOVE(&sq->sq_blocked, td, wle);
172         if (cb != NULL)
173                 cb(td, arg);
174         td->sleepqueue = sq;
175         td->wchan = NULL;
176         sq2 = SLIST_FIRST(&sq->sq_freeq);
177         TAILQ_FOREACH(td, &sq->sq_blocked, wle) {
178                 if (cb != NULL)
179                         cb(td, arg);
180                 td->sleepqueue = sq2;
181                 td->wchan = NULL;
182                 sq2 = SLIST_NEXT(sq2, sq_flink);
183         }
184         TAILQ_INIT(&sq->sq_blocked);
185         SLIST_INIT(&sq->sq_freeq);
186 }