1d30ea906Sjfb8856606 /* SPDX-License-Identifier: BSD-3-Clause
2d30ea906Sjfb8856606 * Copyright(c) 2015 Intel Corporation
3a9643ea8Slogwang */
4a9643ea8Slogwang
5a9643ea8Slogwang #include <stdio.h>
6a9643ea8Slogwang #include <stdlib.h>
7a9643ea8Slogwang #include <sys/types.h>
8a9643ea8Slogwang #include <errno.h>
9a9643ea8Slogwang #include <sched.h>
10a9643ea8Slogwang #include <dlfcn.h>
11a9643ea8Slogwang
12a9643ea8Slogwang #include <rte_log.h>
13a9643ea8Slogwang
14a9643ea8Slogwang #include "lthread_api.h"
15a9643ea8Slogwang #include "pthread_shim.h"
16a9643ea8Slogwang
17a9643ea8Slogwang #define RTE_LOGTYPE_PTHREAD_SHIM RTE_LOGTYPE_USER3
18a9643ea8Slogwang
19a9643ea8Slogwang #define POSIX_ERRNO(x) (x)
20a9643ea8Slogwang
212bfe3f2eSlogwang /* some releases of FreeBSD 10, e.g. 10.0, don't have CPU_COUNT macro */
222bfe3f2eSlogwang #ifndef CPU_COUNT
232bfe3f2eSlogwang #define CPU_COUNT(x) __cpu_count(x)
242bfe3f2eSlogwang
252bfe3f2eSlogwang static inline unsigned int
__cpu_count(const rte_cpuset_t * cpuset)262bfe3f2eSlogwang __cpu_count(const rte_cpuset_t *cpuset)
272bfe3f2eSlogwang {
282bfe3f2eSlogwang unsigned int i, count = 0;
292bfe3f2eSlogwang for (i = 0; i < RTE_MAX_LCORE; i++)
302bfe3f2eSlogwang if (CPU_ISSET(i, cpuset))
312bfe3f2eSlogwang count++;
322bfe3f2eSlogwang return count;
332bfe3f2eSlogwang }
342bfe3f2eSlogwang #endif
352bfe3f2eSlogwang
36a9643ea8Slogwang /*
37a9643ea8Slogwang * this flag determines at run time if we override pthread
38a9643ea8Slogwang * calls and map then to equivalent lthread calls
39a9643ea8Slogwang * or of we call the standard pthread function
40a9643ea8Slogwang */
41a9643ea8Slogwang static __thread int override;
42a9643ea8Slogwang
43a9643ea8Slogwang
44a9643ea8Slogwang /*
45a9643ea8Slogwang * this structures contains function pointers that will be
46a9643ea8Slogwang * initialised to the loaded address of the real
47a9643ea8Slogwang * pthread library API functions
48a9643ea8Slogwang */
49a9643ea8Slogwang struct pthread_lib_funcs {
50a9643ea8Slogwang int (*f_pthread_barrier_destroy)
51a9643ea8Slogwang (pthread_barrier_t *);
52a9643ea8Slogwang int (*f_pthread_barrier_init)
53a9643ea8Slogwang (pthread_barrier_t *, const pthread_barrierattr_t *, unsigned);
54a9643ea8Slogwang int (*f_pthread_barrier_wait)
55a9643ea8Slogwang (pthread_barrier_t *);
56a9643ea8Slogwang int (*f_pthread_cond_broadcast)
57a9643ea8Slogwang (pthread_cond_t *);
58a9643ea8Slogwang int (*f_pthread_cond_destroy)
59a9643ea8Slogwang (pthread_cond_t *);
60a9643ea8Slogwang int (*f_pthread_cond_init)
61a9643ea8Slogwang (pthread_cond_t *, const pthread_condattr_t *);
62a9643ea8Slogwang int (*f_pthread_cond_signal)
63a9643ea8Slogwang (pthread_cond_t *);
64a9643ea8Slogwang int (*f_pthread_cond_timedwait)
65a9643ea8Slogwang (pthread_cond_t *, pthread_mutex_t *, const struct timespec *);
66a9643ea8Slogwang int (*f_pthread_cond_wait)
67a9643ea8Slogwang (pthread_cond_t *, pthread_mutex_t *);
68a9643ea8Slogwang int (*f_pthread_create)
69a9643ea8Slogwang (pthread_t *, const pthread_attr_t *, void *(*)(void *), void *);
70a9643ea8Slogwang int (*f_pthread_detach)
71a9643ea8Slogwang (pthread_t);
72a9643ea8Slogwang int (*f_pthread_equal)
73a9643ea8Slogwang (pthread_t, pthread_t);
74a9643ea8Slogwang void (*f_pthread_exit)
75a9643ea8Slogwang (void *);
76a9643ea8Slogwang void * (*f_pthread_getspecific)
77a9643ea8Slogwang (pthread_key_t);
78a9643ea8Slogwang int (*f_pthread_getcpuclockid)
79a9643ea8Slogwang (pthread_t, clockid_t *);
80a9643ea8Slogwang int (*f_pthread_join)
81a9643ea8Slogwang (pthread_t, void **);
82a9643ea8Slogwang int (*f_pthread_key_create)
83a9643ea8Slogwang (pthread_key_t *, void (*) (void *));
84a9643ea8Slogwang int (*f_pthread_key_delete)
85a9643ea8Slogwang (pthread_key_t);
86a9643ea8Slogwang int (*f_pthread_mutex_destroy)
87a9643ea8Slogwang (pthread_mutex_t *__mutex);
88a9643ea8Slogwang int (*f_pthread_mutex_init)
89a9643ea8Slogwang (pthread_mutex_t *__mutex, const pthread_mutexattr_t *);
90a9643ea8Slogwang int (*f_pthread_mutex_lock)
91a9643ea8Slogwang (pthread_mutex_t *__mutex);
92a9643ea8Slogwang int (*f_pthread_mutex_trylock)
93a9643ea8Slogwang (pthread_mutex_t *__mutex);
94a9643ea8Slogwang int (*f_pthread_mutex_timedlock)
95a9643ea8Slogwang (pthread_mutex_t *__mutex, const struct timespec *);
96a9643ea8Slogwang int (*f_pthread_mutex_unlock)
97a9643ea8Slogwang (pthread_mutex_t *__mutex);
98a9643ea8Slogwang int (*f_pthread_once)
99a9643ea8Slogwang (pthread_once_t *, void (*) (void));
100a9643ea8Slogwang int (*f_pthread_rwlock_destroy)
101a9643ea8Slogwang (pthread_rwlock_t *__rwlock);
102a9643ea8Slogwang int (*f_pthread_rwlock_init)
103a9643ea8Slogwang (pthread_rwlock_t *__rwlock, const pthread_rwlockattr_t *);
104a9643ea8Slogwang int (*f_pthread_rwlock_rdlock)
105a9643ea8Slogwang (pthread_rwlock_t *__rwlock);
106a9643ea8Slogwang int (*f_pthread_rwlock_timedrdlock)
107a9643ea8Slogwang (pthread_rwlock_t *__rwlock, const struct timespec *);
108a9643ea8Slogwang int (*f_pthread_rwlock_timedwrlock)
109a9643ea8Slogwang (pthread_rwlock_t *__rwlock, const struct timespec *);
110a9643ea8Slogwang int (*f_pthread_rwlock_tryrdlock)
111a9643ea8Slogwang (pthread_rwlock_t *__rwlock);
112a9643ea8Slogwang int (*f_pthread_rwlock_trywrlock)
113a9643ea8Slogwang (pthread_rwlock_t *__rwlock);
114a9643ea8Slogwang int (*f_pthread_rwlock_unlock)
115a9643ea8Slogwang (pthread_rwlock_t *__rwlock);
116a9643ea8Slogwang int (*f_pthread_rwlock_wrlock)
117a9643ea8Slogwang (pthread_rwlock_t *__rwlock);
118a9643ea8Slogwang pthread_t (*f_pthread_self)
119a9643ea8Slogwang (void);
120a9643ea8Slogwang int (*f_pthread_setspecific)
121a9643ea8Slogwang (pthread_key_t, const void *);
122a9643ea8Slogwang int (*f_pthread_spin_init)
123a9643ea8Slogwang (pthread_spinlock_t *__spin, int);
124a9643ea8Slogwang int (*f_pthread_spin_destroy)
125a9643ea8Slogwang (pthread_spinlock_t *__spin);
126a9643ea8Slogwang int (*f_pthread_spin_lock)
127a9643ea8Slogwang (pthread_spinlock_t *__spin);
128a9643ea8Slogwang int (*f_pthread_spin_trylock)
129a9643ea8Slogwang (pthread_spinlock_t *__spin);
130a9643ea8Slogwang int (*f_pthread_spin_unlock)
131a9643ea8Slogwang (pthread_spinlock_t *__spin);
132a9643ea8Slogwang int (*f_pthread_cancel)
133a9643ea8Slogwang (pthread_t);
134a9643ea8Slogwang int (*f_pthread_setcancelstate)
135a9643ea8Slogwang (int, int *);
136a9643ea8Slogwang int (*f_pthread_setcanceltype)
137a9643ea8Slogwang (int, int *);
138a9643ea8Slogwang void (*f_pthread_testcancel)
139a9643ea8Slogwang (void);
140a9643ea8Slogwang int (*f_pthread_getschedparam)
141a9643ea8Slogwang (pthread_t pthread, int *, struct sched_param *);
142a9643ea8Slogwang int (*f_pthread_setschedparam)
143a9643ea8Slogwang (pthread_t, int, const struct sched_param *);
144a9643ea8Slogwang int (*f_pthread_yield)
145a9643ea8Slogwang (void);
146a9643ea8Slogwang int (*f_pthread_setaffinity_np)
1472bfe3f2eSlogwang (pthread_t thread, size_t cpusetsize, const rte_cpuset_t *cpuset);
148a9643ea8Slogwang int (*f_nanosleep)
149a9643ea8Slogwang (const struct timespec *req, struct timespec *rem);
150a9643ea8Slogwang } _sys_pthread_funcs = {
151a9643ea8Slogwang .f_pthread_barrier_destroy = NULL,
152a9643ea8Slogwang };
153a9643ea8Slogwang
154a9643ea8Slogwang
155a9643ea8Slogwang /*
156a9643ea8Slogwang * this macro obtains the loaded address of a library function
157a9643ea8Slogwang * and saves it.
158a9643ea8Slogwang */
159a9643ea8Slogwang static void *__libc_dl_handle = RTLD_NEXT;
160a9643ea8Slogwang
161a9643ea8Slogwang #define get_addr_of_loaded_symbol(name) do { \
162a9643ea8Slogwang char *error_str; \
163a9643ea8Slogwang _sys_pthread_funcs.f_##name = dlsym(__libc_dl_handle, (#name)); \
164a9643ea8Slogwang error_str = dlerror(); \
165a9643ea8Slogwang if (error_str != NULL) { \
166a9643ea8Slogwang fprintf(stderr, "%s\n", error_str); \
167a9643ea8Slogwang } \
168a9643ea8Slogwang } while (0)
169a9643ea8Slogwang
170a9643ea8Slogwang
171a9643ea8Slogwang /*
172a9643ea8Slogwang * The constructor function initialises the
173a9643ea8Slogwang * function pointers for pthread library functions
174a9643ea8Slogwang */
RTE_INIT(pthread_intercept_ctor)1752bfe3f2eSlogwang RTE_INIT(pthread_intercept_ctor)
176a9643ea8Slogwang {
177a9643ea8Slogwang override = 0;
178a9643ea8Slogwang /*
179a9643ea8Slogwang * Get the original functions
180a9643ea8Slogwang */
181a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_barrier_destroy);
182a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_barrier_init);
183a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_barrier_wait);
184a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_cond_broadcast);
185a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_cond_destroy);
186a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_cond_init);
187a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_cond_signal);
188a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_cond_timedwait);
189a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_cond_wait);
190a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_create);
191a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_detach);
192a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_equal);
193a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_exit);
194a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_getspecific);
195a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_getcpuclockid);
196a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_join);
197a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_key_create);
198a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_key_delete);
199a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_mutex_destroy);
200a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_mutex_init);
201a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_mutex_lock);
202a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_mutex_trylock);
203a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_mutex_timedlock);
204a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_mutex_unlock);
205a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_once);
206a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_destroy);
207a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_init);
208a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_rdlock);
209a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_timedrdlock);
210a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_timedwrlock);
211a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_tryrdlock);
212a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_trywrlock);
213a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_unlock);
214a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_rwlock_wrlock);
215a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_self);
216a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_setspecific);
217a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_spin_init);
218a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_spin_destroy);
219a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_spin_lock);
220a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_spin_trylock);
221a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_spin_unlock);
222a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_cancel);
223a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_setcancelstate);
224a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_setcanceltype);
225a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_testcancel);
226a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_getschedparam);
227a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_setschedparam);
228a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_yield);
229a9643ea8Slogwang get_addr_of_loaded_symbol(pthread_setaffinity_np);
230a9643ea8Slogwang get_addr_of_loaded_symbol(nanosleep);
231a9643ea8Slogwang }
232a9643ea8Slogwang
233a9643ea8Slogwang
234a9643ea8Slogwang /*
235a9643ea8Slogwang * Enable/Disable pthread override
236a9643ea8Slogwang * state
237a9643ea8Slogwang * 0 disable
238a9643ea8Slogwang * 1 enable
239a9643ea8Slogwang */
pthread_override_set(int state)240a9643ea8Slogwang void pthread_override_set(int state)
241a9643ea8Slogwang {
242a9643ea8Slogwang override = state;
243a9643ea8Slogwang }
244a9643ea8Slogwang
245a9643ea8Slogwang
246a9643ea8Slogwang /*
247a9643ea8Slogwang * Return pthread override state
248a9643ea8Slogwang * return
249a9643ea8Slogwang * 0 disable
250a9643ea8Slogwang * 1 enable
251a9643ea8Slogwang */
pthread_override_get(void)252a9643ea8Slogwang int pthread_override_get(void)
253a9643ea8Slogwang {
254a9643ea8Slogwang return override;
255a9643ea8Slogwang }
256a9643ea8Slogwang
257a9643ea8Slogwang /*
258a9643ea8Slogwang * This macro is used to catch and log
259a9643ea8Slogwang * invocation of stubs for unimplemented pthread
260a9643ea8Slogwang * API functions.
261a9643ea8Slogwang */
262a9643ea8Slogwang #define NOT_IMPLEMENTED do { \
263a9643ea8Slogwang if (override) { \
264a9643ea8Slogwang RTE_LOG(WARNING, \
265a9643ea8Slogwang PTHREAD_SHIM, \
266a9643ea8Slogwang "WARNING %s NOT IMPLEMENTED\n", \
267a9643ea8Slogwang __func__); \
268a9643ea8Slogwang } \
269a9643ea8Slogwang } while (0)
270a9643ea8Slogwang
271a9643ea8Slogwang /*
272a9643ea8Slogwang * pthread API override functions follow
273a9643ea8Slogwang * Note in this example code only a subset of functions are
274a9643ea8Slogwang * implemented.
275a9643ea8Slogwang *
276a9643ea8Slogwang * The stub functions provided will issue a warning log
277a9643ea8Slogwang * message if an unimplemented function is invoked
278a9643ea8Slogwang *
279a9643ea8Slogwang */
280a9643ea8Slogwang
pthread_barrier_destroy(pthread_barrier_t * a)281a9643ea8Slogwang int pthread_barrier_destroy(pthread_barrier_t *a)
282a9643ea8Slogwang {
283a9643ea8Slogwang NOT_IMPLEMENTED;
284a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_barrier_destroy(a);
285a9643ea8Slogwang }
286a9643ea8Slogwang
287a9643ea8Slogwang int
pthread_barrier_init(pthread_barrier_t * a,const pthread_barrierattr_t * b,unsigned c)288a9643ea8Slogwang pthread_barrier_init(pthread_barrier_t *a,
289a9643ea8Slogwang const pthread_barrierattr_t *b, unsigned c)
290a9643ea8Slogwang {
291a9643ea8Slogwang NOT_IMPLEMENTED;
292a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_barrier_init(a, b, c);
293a9643ea8Slogwang }
294a9643ea8Slogwang
pthread_barrier_wait(pthread_barrier_t * a)295a9643ea8Slogwang int pthread_barrier_wait(pthread_barrier_t *a)
296a9643ea8Slogwang {
297a9643ea8Slogwang NOT_IMPLEMENTED;
298a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_barrier_wait(a);
299a9643ea8Slogwang }
300a9643ea8Slogwang
pthread_cond_broadcast(pthread_cond_t * cond)301a9643ea8Slogwang int pthread_cond_broadcast(pthread_cond_t *cond)
302a9643ea8Slogwang {
303a9643ea8Slogwang if (override) {
304a9643ea8Slogwang
305a9643ea8Slogwang lthread_cond_broadcast(*(struct lthread_cond **)cond);
306a9643ea8Slogwang return 0;
307a9643ea8Slogwang }
308a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_cond_broadcast(cond);
309a9643ea8Slogwang }
310a9643ea8Slogwang
pthread_mutex_destroy(pthread_mutex_t * mutex)311a9643ea8Slogwang int pthread_mutex_destroy(pthread_mutex_t *mutex)
312a9643ea8Slogwang {
313a9643ea8Slogwang if (override)
314a9643ea8Slogwang return lthread_mutex_destroy(*(struct lthread_mutex **)mutex);
315a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_mutex_destroy(mutex);
316a9643ea8Slogwang }
317a9643ea8Slogwang
pthread_cond_destroy(pthread_cond_t * cond)318a9643ea8Slogwang int pthread_cond_destroy(pthread_cond_t *cond)
319a9643ea8Slogwang {
320a9643ea8Slogwang if (override)
321a9643ea8Slogwang return lthread_cond_destroy(*(struct lthread_cond **)cond);
322a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_cond_destroy(cond);
323a9643ea8Slogwang }
324a9643ea8Slogwang
pthread_cond_init(pthread_cond_t * cond,const pthread_condattr_t * attr)325a9643ea8Slogwang int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr)
326a9643ea8Slogwang {
327a9643ea8Slogwang if (override)
328a9643ea8Slogwang return lthread_cond_init(NULL,
329a9643ea8Slogwang (struct lthread_cond **)cond,
330a9643ea8Slogwang (const struct lthread_condattr *) attr);
331a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_cond_init(cond, attr);
332a9643ea8Slogwang }
333a9643ea8Slogwang
pthread_cond_signal(pthread_cond_t * cond)334a9643ea8Slogwang int pthread_cond_signal(pthread_cond_t *cond)
335a9643ea8Slogwang {
336a9643ea8Slogwang if (override) {
337a9643ea8Slogwang lthread_cond_signal(*(struct lthread_cond **)cond);
338a9643ea8Slogwang return 0;
339a9643ea8Slogwang }
340a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_cond_signal(cond);
341a9643ea8Slogwang }
342a9643ea8Slogwang
343a9643ea8Slogwang int
pthread_cond_timedwait(pthread_cond_t * __rte_restrict cond,pthread_mutex_t * __rte_restrict mutex,const struct timespec * __rte_restrict time)344*2d9fd380Sjfb8856606 pthread_cond_timedwait(pthread_cond_t *__rte_restrict cond,
345*2d9fd380Sjfb8856606 pthread_mutex_t *__rte_restrict mutex,
346*2d9fd380Sjfb8856606 const struct timespec *__rte_restrict time)
347a9643ea8Slogwang {
348a9643ea8Slogwang NOT_IMPLEMENTED;
349a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_cond_timedwait(cond, mutex, time);
350a9643ea8Slogwang }
351a9643ea8Slogwang
pthread_cond_wait(pthread_cond_t * cond,pthread_mutex_t * mutex)352a9643ea8Slogwang int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex)
353a9643ea8Slogwang {
354a9643ea8Slogwang if (override) {
355a9643ea8Slogwang pthread_mutex_unlock(mutex);
356a9643ea8Slogwang int rv = lthread_cond_wait(*(struct lthread_cond **)cond, 0);
357a9643ea8Slogwang
358a9643ea8Slogwang pthread_mutex_lock(mutex);
359a9643ea8Slogwang return rv;
360a9643ea8Slogwang }
361a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_cond_wait(cond, mutex);
362a9643ea8Slogwang }
363a9643ea8Slogwang
364a9643ea8Slogwang int
pthread_create(pthread_t * __rte_restrict tid,const pthread_attr_t * __rte_restrict attr,lthread_func_t func,void * __rte_restrict arg)365*2d9fd380Sjfb8856606 pthread_create(pthread_t *__rte_restrict tid,
366*2d9fd380Sjfb8856606 const pthread_attr_t *__rte_restrict attr,
367579bf1e2Sjfb8856606 lthread_func_t func,
368*2d9fd380Sjfb8856606 void *__rte_restrict arg)
369a9643ea8Slogwang {
370a9643ea8Slogwang if (override) {
371a9643ea8Slogwang int lcore = -1;
372a9643ea8Slogwang
373a9643ea8Slogwang if (attr != NULL) {
374a9643ea8Slogwang /* determine CPU being requested */
3752bfe3f2eSlogwang rte_cpuset_t cpuset;
376a9643ea8Slogwang
377a9643ea8Slogwang CPU_ZERO(&cpuset);
378a9643ea8Slogwang pthread_attr_getaffinity_np(attr,
3792bfe3f2eSlogwang sizeof(rte_cpuset_t),
380a9643ea8Slogwang &cpuset);
381a9643ea8Slogwang
382a9643ea8Slogwang if (CPU_COUNT(&cpuset) != 1)
383a9643ea8Slogwang return POSIX_ERRNO(EINVAL);
384a9643ea8Slogwang
385a9643ea8Slogwang for (lcore = 0; lcore < LTHREAD_MAX_LCORES; lcore++) {
386a9643ea8Slogwang if (!CPU_ISSET(lcore, &cpuset))
387a9643ea8Slogwang continue;
388a9643ea8Slogwang break;
389a9643ea8Slogwang }
390a9643ea8Slogwang }
391a9643ea8Slogwang return lthread_create((struct lthread **)tid, lcore,
392579bf1e2Sjfb8856606 func, arg);
393a9643ea8Slogwang }
394a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_create(tid, attr, func, arg);
395a9643ea8Slogwang }
396a9643ea8Slogwang
pthread_detach(pthread_t tid)397a9643ea8Slogwang int pthread_detach(pthread_t tid)
398a9643ea8Slogwang {
399a9643ea8Slogwang if (override) {
400a9643ea8Slogwang struct lthread *lt = (struct lthread *)tid;
401a9643ea8Slogwang
402a9643ea8Slogwang if (lt == lthread_current()) {
403a9643ea8Slogwang lthread_detach();
404a9643ea8Slogwang return 0;
405a9643ea8Slogwang }
406a9643ea8Slogwang NOT_IMPLEMENTED;
407a9643ea8Slogwang }
408a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_detach(tid);
409a9643ea8Slogwang }
410a9643ea8Slogwang
pthread_equal(pthread_t a,pthread_t b)411a9643ea8Slogwang int pthread_equal(pthread_t a, pthread_t b)
412a9643ea8Slogwang {
413a9643ea8Slogwang NOT_IMPLEMENTED;
414a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_equal(a, b);
415a9643ea8Slogwang }
416a9643ea8Slogwang
pthread_exit_override(void * v)417a9643ea8Slogwang void pthread_exit_override(void *v)
418a9643ea8Slogwang {
419a9643ea8Slogwang if (override) {
420a9643ea8Slogwang lthread_exit(v);
421a9643ea8Slogwang return;
422a9643ea8Slogwang }
423a9643ea8Slogwang _sys_pthread_funcs.f_pthread_exit(v);
424a9643ea8Slogwang }
425a9643ea8Slogwang
426a9643ea8Slogwang void
pthread_getspecific(pthread_key_t key)427a9643ea8Slogwang *pthread_getspecific(pthread_key_t key)
428a9643ea8Slogwang {
429a9643ea8Slogwang if (override)
430a9643ea8Slogwang return lthread_getspecific((unsigned int) key);
431a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_getspecific(key);
432a9643ea8Slogwang }
433a9643ea8Slogwang
pthread_getcpuclockid(pthread_t a,clockid_t * b)434a9643ea8Slogwang int pthread_getcpuclockid(pthread_t a, clockid_t *b)
435a9643ea8Slogwang {
436a9643ea8Slogwang NOT_IMPLEMENTED;
437a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_getcpuclockid(a, b);
438a9643ea8Slogwang }
439a9643ea8Slogwang
pthread_join(pthread_t tid,void ** val)440a9643ea8Slogwang int pthread_join(pthread_t tid, void **val)
441a9643ea8Slogwang {
442a9643ea8Slogwang if (override)
443a9643ea8Slogwang return lthread_join((struct lthread *)tid, val);
444a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_join(tid, val);
445a9643ea8Slogwang }
446a9643ea8Slogwang
pthread_key_create(pthread_key_t * keyptr,void (* dtor)(void *))447a9643ea8Slogwang int pthread_key_create(pthread_key_t *keyptr, void (*dtor) (void *))
448a9643ea8Slogwang {
449a9643ea8Slogwang if (override)
450a9643ea8Slogwang return lthread_key_create((unsigned int *)keyptr, dtor);
451a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_key_create(keyptr, dtor);
452a9643ea8Slogwang }
453a9643ea8Slogwang
pthread_key_delete(pthread_key_t key)454a9643ea8Slogwang int pthread_key_delete(pthread_key_t key)
455a9643ea8Slogwang {
456a9643ea8Slogwang if (override) {
457a9643ea8Slogwang lthread_key_delete((unsigned int) key);
458a9643ea8Slogwang return 0;
459a9643ea8Slogwang }
460a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_key_delete(key);
461a9643ea8Slogwang }
462a9643ea8Slogwang
463a9643ea8Slogwang
464a9643ea8Slogwang int
pthread_mutex_init(pthread_mutex_t * mutex,const pthread_mutexattr_t * attr)465a9643ea8Slogwang pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr)
466a9643ea8Slogwang {
467a9643ea8Slogwang if (override)
468a9643ea8Slogwang return lthread_mutex_init(NULL,
469a9643ea8Slogwang (struct lthread_mutex **)mutex,
470a9643ea8Slogwang (const struct lthread_mutexattr *)attr);
471a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_mutex_init(mutex, attr);
472a9643ea8Slogwang }
473a9643ea8Slogwang
pthread_mutex_lock(pthread_mutex_t * mutex)474a9643ea8Slogwang int pthread_mutex_lock(pthread_mutex_t *mutex)
475a9643ea8Slogwang {
476a9643ea8Slogwang if (override)
477a9643ea8Slogwang return lthread_mutex_lock(*(struct lthread_mutex **)mutex);
478a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_mutex_lock(mutex);
479a9643ea8Slogwang }
480a9643ea8Slogwang
pthread_mutex_trylock(pthread_mutex_t * mutex)481a9643ea8Slogwang int pthread_mutex_trylock(pthread_mutex_t *mutex)
482a9643ea8Slogwang {
483a9643ea8Slogwang if (override)
484a9643ea8Slogwang return lthread_mutex_trylock(*(struct lthread_mutex **)mutex);
485a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_mutex_trylock(mutex);
486a9643ea8Slogwang }
487a9643ea8Slogwang
pthread_mutex_timedlock(pthread_mutex_t * mutex,const struct timespec * b)488a9643ea8Slogwang int pthread_mutex_timedlock(pthread_mutex_t *mutex, const struct timespec *b)
489a9643ea8Slogwang {
490a9643ea8Slogwang NOT_IMPLEMENTED;
491a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_mutex_timedlock(mutex, b);
492a9643ea8Slogwang }
493a9643ea8Slogwang
pthread_mutex_unlock(pthread_mutex_t * mutex)494a9643ea8Slogwang int pthread_mutex_unlock(pthread_mutex_t *mutex)
495a9643ea8Slogwang {
496a9643ea8Slogwang if (override)
497a9643ea8Slogwang return lthread_mutex_unlock(*(struct lthread_mutex **)mutex);
498a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_mutex_unlock(mutex);
499a9643ea8Slogwang }
500a9643ea8Slogwang
pthread_once(pthread_once_t * a,void (b)(void))501a9643ea8Slogwang int pthread_once(pthread_once_t *a, void (b) (void))
502a9643ea8Slogwang {
503a9643ea8Slogwang NOT_IMPLEMENTED;
504a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_once(a, b);
505a9643ea8Slogwang }
506a9643ea8Slogwang
pthread_rwlock_destroy(pthread_rwlock_t * a)507a9643ea8Slogwang int pthread_rwlock_destroy(pthread_rwlock_t *a)
508a9643ea8Slogwang {
509a9643ea8Slogwang NOT_IMPLEMENTED;
510a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_destroy(a);
511a9643ea8Slogwang }
512a9643ea8Slogwang
pthread_rwlock_init(pthread_rwlock_t * a,const pthread_rwlockattr_t * b)513a9643ea8Slogwang int pthread_rwlock_init(pthread_rwlock_t *a, const pthread_rwlockattr_t *b)
514a9643ea8Slogwang {
515a9643ea8Slogwang NOT_IMPLEMENTED;
516a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_init(a, b);
517a9643ea8Slogwang }
518a9643ea8Slogwang
pthread_rwlock_rdlock(pthread_rwlock_t * a)519a9643ea8Slogwang int pthread_rwlock_rdlock(pthread_rwlock_t *a)
520a9643ea8Slogwang {
521a9643ea8Slogwang NOT_IMPLEMENTED;
522a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_rdlock(a);
523a9643ea8Slogwang }
524a9643ea8Slogwang
pthread_rwlock_timedrdlock(pthread_rwlock_t * a,const struct timespec * b)525a9643ea8Slogwang int pthread_rwlock_timedrdlock(pthread_rwlock_t *a, const struct timespec *b)
526a9643ea8Slogwang {
527a9643ea8Slogwang NOT_IMPLEMENTED;
528a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_timedrdlock(a, b);
529a9643ea8Slogwang }
530a9643ea8Slogwang
pthread_rwlock_timedwrlock(pthread_rwlock_t * a,const struct timespec * b)531a9643ea8Slogwang int pthread_rwlock_timedwrlock(pthread_rwlock_t *a, const struct timespec *b)
532a9643ea8Slogwang {
533a9643ea8Slogwang NOT_IMPLEMENTED;
534a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_timedwrlock(a, b);
535a9643ea8Slogwang }
536a9643ea8Slogwang
pthread_rwlock_tryrdlock(pthread_rwlock_t * a)537a9643ea8Slogwang int pthread_rwlock_tryrdlock(pthread_rwlock_t *a)
538a9643ea8Slogwang {
539a9643ea8Slogwang NOT_IMPLEMENTED;
540a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_tryrdlock(a);
541a9643ea8Slogwang }
542a9643ea8Slogwang
pthread_rwlock_trywrlock(pthread_rwlock_t * a)543a9643ea8Slogwang int pthread_rwlock_trywrlock(pthread_rwlock_t *a)
544a9643ea8Slogwang {
545a9643ea8Slogwang NOT_IMPLEMENTED;
546a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_trywrlock(a);
547a9643ea8Slogwang }
548a9643ea8Slogwang
pthread_rwlock_unlock(pthread_rwlock_t * a)549a9643ea8Slogwang int pthread_rwlock_unlock(pthread_rwlock_t *a)
550a9643ea8Slogwang {
551a9643ea8Slogwang NOT_IMPLEMENTED;
552a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_unlock(a);
553a9643ea8Slogwang }
554a9643ea8Slogwang
pthread_rwlock_wrlock(pthread_rwlock_t * a)555a9643ea8Slogwang int pthread_rwlock_wrlock(pthread_rwlock_t *a)
556a9643ea8Slogwang {
557a9643ea8Slogwang NOT_IMPLEMENTED;
558a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_rwlock_wrlock(a);
559a9643ea8Slogwang }
560a9643ea8Slogwang
5614418919fSjohnjiang #ifdef RTE_EXEC_ENV_LINUX
5622bfe3f2eSlogwang int
pthread_yield(void)5632bfe3f2eSlogwang pthread_yield(void)
564a9643ea8Slogwang {
565a9643ea8Slogwang if (override) {
566a9643ea8Slogwang lthread_yield();
567a9643ea8Slogwang return 0;
568a9643ea8Slogwang }
569a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_yield();
570a9643ea8Slogwang }
5712bfe3f2eSlogwang #else
5722bfe3f2eSlogwang void
pthread_yield(void)5732bfe3f2eSlogwang pthread_yield(void)
5742bfe3f2eSlogwang {
5752bfe3f2eSlogwang if (override)
5762bfe3f2eSlogwang lthread_yield();
5772bfe3f2eSlogwang else
5782bfe3f2eSlogwang _sys_pthread_funcs.f_pthread_yield();
5792bfe3f2eSlogwang }
5802bfe3f2eSlogwang #endif
581a9643ea8Slogwang
pthread_self(void)582a9643ea8Slogwang pthread_t pthread_self(void)
583a9643ea8Slogwang {
584a9643ea8Slogwang if (override)
585a9643ea8Slogwang return (pthread_t) lthread_current();
586a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_self();
587a9643ea8Slogwang }
588a9643ea8Slogwang
pthread_setspecific(pthread_key_t key,const void * data)589a9643ea8Slogwang int pthread_setspecific(pthread_key_t key, const void *data)
590a9643ea8Slogwang {
591a9643ea8Slogwang if (override) {
592a9643ea8Slogwang int rv = lthread_setspecific((unsigned int)key, data);
593a9643ea8Slogwang return rv;
594a9643ea8Slogwang }
595a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_setspecific(key, data);
596a9643ea8Slogwang }
597a9643ea8Slogwang
pthread_spin_init(pthread_spinlock_t * a,int b)598a9643ea8Slogwang int pthread_spin_init(pthread_spinlock_t *a, int b)
599a9643ea8Slogwang {
600a9643ea8Slogwang NOT_IMPLEMENTED;
601a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_spin_init(a, b);
602a9643ea8Slogwang }
603a9643ea8Slogwang
pthread_spin_destroy(pthread_spinlock_t * a)604a9643ea8Slogwang int pthread_spin_destroy(pthread_spinlock_t *a)
605a9643ea8Slogwang {
606a9643ea8Slogwang NOT_IMPLEMENTED;
607a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_spin_destroy(a);
608a9643ea8Slogwang }
609a9643ea8Slogwang
pthread_spin_lock(pthread_spinlock_t * a)610a9643ea8Slogwang int pthread_spin_lock(pthread_spinlock_t *a)
611a9643ea8Slogwang {
612a9643ea8Slogwang NOT_IMPLEMENTED;
613a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_spin_lock(a);
614a9643ea8Slogwang }
615a9643ea8Slogwang
pthread_spin_trylock(pthread_spinlock_t * a)616a9643ea8Slogwang int pthread_spin_trylock(pthread_spinlock_t *a)
617a9643ea8Slogwang {
618a9643ea8Slogwang NOT_IMPLEMENTED;
619a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_spin_trylock(a);
620a9643ea8Slogwang }
621a9643ea8Slogwang
pthread_spin_unlock(pthread_spinlock_t * a)622a9643ea8Slogwang int pthread_spin_unlock(pthread_spinlock_t *a)
623a9643ea8Slogwang {
624a9643ea8Slogwang NOT_IMPLEMENTED;
625a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_spin_unlock(a);
626a9643ea8Slogwang }
627a9643ea8Slogwang
pthread_cancel(pthread_t tid)628a9643ea8Slogwang int pthread_cancel(pthread_t tid)
629a9643ea8Slogwang {
630a9643ea8Slogwang if (override) {
631a9643ea8Slogwang lthread_cancel(*(struct lthread **)tid);
632a9643ea8Slogwang return 0;
633a9643ea8Slogwang }
634a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_cancel(tid);
635a9643ea8Slogwang }
636a9643ea8Slogwang
pthread_setcancelstate(int a,int * b)637a9643ea8Slogwang int pthread_setcancelstate(int a, int *b)
638a9643ea8Slogwang {
639a9643ea8Slogwang NOT_IMPLEMENTED;
640a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_setcancelstate(a, b);
641a9643ea8Slogwang }
642a9643ea8Slogwang
pthread_setcanceltype(int a,int * b)643a9643ea8Slogwang int pthread_setcanceltype(int a, int *b)
644a9643ea8Slogwang {
645a9643ea8Slogwang NOT_IMPLEMENTED;
646a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_setcanceltype(a, b);
647a9643ea8Slogwang }
648a9643ea8Slogwang
pthread_testcancel(void)649a9643ea8Slogwang void pthread_testcancel(void)
650a9643ea8Slogwang {
651a9643ea8Slogwang NOT_IMPLEMENTED;
652a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_testcancel();
653a9643ea8Slogwang }
654a9643ea8Slogwang
655a9643ea8Slogwang
pthread_getschedparam(pthread_t tid,int * a,struct sched_param * b)656a9643ea8Slogwang int pthread_getschedparam(pthread_t tid, int *a, struct sched_param *b)
657a9643ea8Slogwang {
658a9643ea8Slogwang NOT_IMPLEMENTED;
659a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_getschedparam(tid, a, b);
660a9643ea8Slogwang }
661a9643ea8Slogwang
pthread_setschedparam(pthread_t a,int b,const struct sched_param * c)662a9643ea8Slogwang int pthread_setschedparam(pthread_t a, int b, const struct sched_param *c)
663a9643ea8Slogwang {
664a9643ea8Slogwang NOT_IMPLEMENTED;
665a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_setschedparam(a, b, c);
666a9643ea8Slogwang }
667a9643ea8Slogwang
668a9643ea8Slogwang
nanosleep(const struct timespec * req,struct timespec * rem)669a9643ea8Slogwang int nanosleep(const struct timespec *req, struct timespec *rem)
670a9643ea8Slogwang {
671a9643ea8Slogwang if (override) {
672a9643ea8Slogwang uint64_t ns = req->tv_sec * 1000000000 + req->tv_nsec;
673a9643ea8Slogwang
674a9643ea8Slogwang lthread_sleep(ns);
675a9643ea8Slogwang return 0;
676a9643ea8Slogwang }
677a9643ea8Slogwang return _sys_pthread_funcs.f_nanosleep(req, rem);
678a9643ea8Slogwang }
679a9643ea8Slogwang
680a9643ea8Slogwang int
pthread_setaffinity_np(pthread_t thread,size_t cpusetsize,const rte_cpuset_t * cpuset)681a9643ea8Slogwang pthread_setaffinity_np(pthread_t thread, size_t cpusetsize,
6822bfe3f2eSlogwang const rte_cpuset_t *cpuset)
683a9643ea8Slogwang {
684a9643ea8Slogwang if (override) {
685a9643ea8Slogwang /* we only allow affinity with a single CPU */
686a9643ea8Slogwang if (CPU_COUNT(cpuset) != 1)
687a9643ea8Slogwang return POSIX_ERRNO(EINVAL);
688a9643ea8Slogwang
689a9643ea8Slogwang /* we only allow the current thread to sets its own affinity */
690a9643ea8Slogwang struct lthread *lt = (struct lthread *)thread;
691a9643ea8Slogwang
692a9643ea8Slogwang if (lthread_current() != lt)
693a9643ea8Slogwang return POSIX_ERRNO(EINVAL);
694a9643ea8Slogwang
695a9643ea8Slogwang /* determine the CPU being requested */
696a9643ea8Slogwang int i;
697a9643ea8Slogwang
698a9643ea8Slogwang for (i = 0; i < LTHREAD_MAX_LCORES; i++) {
699a9643ea8Slogwang if (!CPU_ISSET(i, cpuset))
700a9643ea8Slogwang continue;
701a9643ea8Slogwang break;
702a9643ea8Slogwang }
703a9643ea8Slogwang /* check requested core is allowed */
704a9643ea8Slogwang if (i == LTHREAD_MAX_LCORES)
705a9643ea8Slogwang return POSIX_ERRNO(EINVAL);
706a9643ea8Slogwang
707a9643ea8Slogwang /* finally we can set affinity to the requested lcore */
708a9643ea8Slogwang lthread_set_affinity(i);
709a9643ea8Slogwang return 0;
710a9643ea8Slogwang }
711a9643ea8Slogwang return _sys_pthread_funcs.f_pthread_setaffinity_np(thread, cpusetsize,
712a9643ea8Slogwang cpuset);
713a9643ea8Slogwang }
714