libcoap 4.3.5-develop-1ba3158
Loading...
Searching...
No Matches
coap_threadsafe.c
Go to the documentation of this file.
1/* coap_threadsafe.c -- Thread safe function locking wrappers
2 *
3 * Copyright (C) 2023-2025 Jon Shallow <supjps-libcoap@jpshallow.com>
4 *
5 * SPDX-License-Identifier: BSD-2-Clause
6 *
7 * This file is part of the CoAP library libcoap. Please see
8 * README for terms of use.
9 */
10
17
18#if COAP_THREAD_SAFE
19#if COAP_THREAD_RECURSIVE_CHECK
20void
21coap_lock_unlock_func(const char *file, int line) {
22 assert(coap_thread_pid == global_lock.pid);
23 if (global_lock.in_callback) {
24 assert(global_lock.lock_count > 0);
25 global_lock.lock_count--;
26 } else {
27 global_lock.pid = 0;
28 global_lock.unlock_file = file;
29 global_lock.unlock_line = line;
30 coap_mutex_unlock(&global_lock.mutex);
31 }
32}
33
34int
35coap_lock_lock_func(const char *file, int line) {
36 if (!coap_started) {
37 /* libcoap not initialized with coap_startup() */
38 return 0;
39 }
40 if (coap_mutex_trylock(&global_lock.mutex)) {
41 if (coap_thread_pid == global_lock.pid) {
42 /* This thread locked the mutex */
43 if (global_lock.in_callback) {
44 /* This is called from within an app callback */
45 global_lock.lock_count++;
46 assert(global_lock.in_callback == global_lock.lock_count);
47 return 1;
48 } else {
49 coap_log_alert("Thread Deadlock: Last %s: %u, this %s: %u\n",
50 global_lock.lock_file, global_lock.lock_line, file, line);
51 assert(0);
52 }
53 }
54 /* Wait for the other thread to unlock */
55 coap_mutex_lock(&global_lock.mutex);
56 }
57 /* Just got the lock, so should not be in a locked callback */
58 assert(!global_lock.in_callback);
59 global_lock.pid = coap_thread_pid;
60 global_lock.lock_file = file;
61 global_lock.lock_line = line;
62 return 1;
63}
64
65#else /* ! COAP_THREAD_RECURSIVE_CHECK */
66
67void
68coap_lock_unlock_func(void) {
69 assert(coap_thread_pid == global_lock.pid);
70 if (global_lock.in_callback) {
71 assert(global_lock.lock_count > 0);
72 global_lock.lock_count--;
73 } else {
74 global_lock.pid = 0;
75 coap_mutex_unlock(&global_lock.mutex);
76 }
77}
78
79int
80coap_lock_lock_func(void) {
81 if (!coap_started) {
82 /* libcoap not initialized with coap_startup() */
83 return 0;
84 }
85 /*
86 * Some OS do not have support for coap_mutex_trylock() so
87 * cannot use that here and have to rely on lock-pid being stable
88 */
89 if (global_lock.in_callback && coap_thread_pid == global_lock.pid) {
90 global_lock.lock_count++;
91 assert(global_lock.in_callback == global_lock.lock_count);
92 return 1;
93 }
94 coap_mutex_lock(&global_lock.mutex);
95 /* Just got the lock, so should not be in a locked callback */
96 assert(!global_lock.in_callback);
97 global_lock.pid = coap_thread_pid;
98 return 1;
99}
100#endif /* ! COAP_THREAD_RECURSIVE_CHECK */
101
102#if !WITH_LWIP
103extern volatile int coap_thread_quit;
104static pthread_t *thread_id = NULL;
105static uint32_t thread_id_count = 0;
106
107/* Visible to only this thread */
109/* Visible across all threads */
110uint32_t max_thread_no = 0;
111
112typedef struct {
113 coap_context_t *context;
114 uint32_t thread_no;
115} coap_thread_param_t;
116
117static void *
118coap_io_process_worker_thread(void *arg) {
119 coap_thread_param_t *thread_param = (coap_thread_param_t *)arg;
120 coap_context_t *context = thread_param->context;
121
122 thread_no = thread_param->thread_no;
123 coap_free_type(COAP_STRING, thread_param);
124
125 coap_log_debug("Thread %lx start\n", pthread_self());
126
127 while (!coap_thread_quit) {
128 int result;
129
130 coap_lock_lock(context, return 0);
131 result = coap_io_process_lkd(context, COAP_IO_WAIT);
132 coap_lock_unlock(context);
133 if (result < 0)
134 break;
135 }
136 coap_log_debug("Thread %lx exit\n", pthread_self());
137 return 0;
138}
139
140int
141coap_io_process_configure_threads(coap_context_t *context, uint32_t thread_count) {
142 uint32_t i;
143
144 coap_mutex_lock(&m_io_threads);
145
146 thread_no = 1;
147 max_thread_no = 1 + thread_count;
148 coap_free_type(COAP_STRING, thread_id);
149 thread_id = coap_malloc_type(COAP_STRING, thread_count * sizeof(pthread_t));
150 if (!thread_id) {
151 coap_log_err("thread start up memory allocate failure\n");
152 coap_mutex_unlock(&m_io_threads);
153 return 0;
154 }
155 for (i = 0; i < thread_count ; i++) {
156 coap_thread_param_t *thread_param = coap_malloc_type(COAP_STRING, sizeof(coap_thread_param_t));
157 int s;
158
159 thread_param->context = context;
160 thread_param->thread_no = i + 2;
161 s = pthread_create(&thread_id[i], NULL,
162 &coap_io_process_worker_thread, thread_param);
163 if (s != 0) {
164 coap_log_err("thread start up failure (%s)\n", coap_socket_strerror());
165 coap_mutex_unlock(&m_io_threads);
166 return 0;
167 }
168 thread_id_count++;
169 }
170 coap_mutex_unlock(&m_io_threads);
171 return 1;
172}
173
174#ifdef HAVE_SIGNAL_H
175#include <signal.h>
176#endif /* HAVE_SIGNAL_H */
177void
179 uint32_t i;
180
181 (void)context;
182
183 coap_lock_unlock(context);
184 coap_mutex_lock(&m_io_threads);
185
186 for (i = 0; i < thread_id_count ; i++) {
187 int s = pthread_kill(thread_id[i], SIGINT);
188 if (s != 0) {
189 coap_log_err("thread kill failure\n");
190 }
191 }
192 for (i = 0; i < thread_id_count ; i++) {
193 void *retval;
194 int s = pthread_join(thread_id[i], &retval);
195 if (s != 0) {
196 coap_log_err("thread join failure\n");
197 }
198 }
199 coap_free_type(COAP_STRING, thread_id);
200 thread_id = NULL;
201 thread_id_count = 0;
202
203 coap_mutex_unlock(&m_io_threads);
204 coap_lock_lock(context, return);
205}
206#endif /* !WITH_LWIP */
207
208#else /* ! COAP_THREAD_SAFE */
209
210#ifdef __clang__
211/* Make compilers happy that do not like empty modules. As this function is
212 * never used, we ignore -Wunused-function at the end of compiling this file
213 */
214#pragma GCC diagnostic ignored "-Wunused-function"
215#endif
216static inline void
217dummy(void) {
218}
219
220#endif /* ! COAP_THREAD_SAFE */
uint32_t max_thread_no
COAP_THREAD_LOCAL_VAR uint32_t thread_no
const char * coap_socket_strerror(void)
Definition coap_io.c:2291
volatile int coap_thread_quit
Definition coap_io.c:2111
Library specific build wrapper for coap_internal.h.
@ COAP_STRING
Definition coap_mem.h:39
void * coap_malloc_type(coap_memory_tag_t type, size_t size)
Allocates a chunk of size bytes and returns a pointer to the newly allocated memory.
void coap_free_type(coap_memory_tag_t type, void *p)
Releases the memory that was allocated by coap_malloc_type().
#define coap_thread_pid
#define coap_mutex_unlock(a)
#define coap_mutex_trylock(a)
#define coap_mutex_lock(a)
int coap_started
Definition coap_net.c:4824
static void dummy(void)
int coap_io_process_lkd(coap_context_t *ctx, uint32_t timeout_ms)
The main I/O processing function.
Definition coap_io.c:1769
int coap_io_process_configure_threads(coap_context_t *context, uint32_t thread_count)
Configure a defined number of threads to do the alternate coap_io_process() work with traffic load ba...
void coap_io_process_remove_threads(coap_context_t *context)
Release the coap_io_process() worker threads.
#define COAP_IO_WAIT
Definition coap_net.h:703
#define coap_lock_unlock(c)
Dummy for no thread-safe code.
#define coap_lock_lock(c, failed)
Dummy for no thread-safe code.
#define coap_log_debug(...)
Definition coap_debug.h:120
#define coap_log_alert(...)
Definition coap_debug.h:84
#define coap_log_err(...)
Definition coap_debug.h:96
#define COAP_THREAD_LOCAL_VAR
Definition libcoap.h:80
The CoAP stack's global state is stored in a coap_context_t object.