libcoap 4.3.5-develop-84778a7
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Modules Pages
coap_tcp.c
Go to the documentation of this file.
1/*
2 * coap_tcp.c -- TCP functions for libcoap
3 *
4 * Copyright (C) 2019-2025 Olaf Bergmann <bergmann@tzi.org> and others
5 *
6 * SPDX-License-Identifier: BSD-2-Clause
7 *
8 * This file is part of the CoAP library libcoap. Please see README for terms
9 * of use.
10 */
11
18
19#if COAP_AF_UNIX_SUPPORT
20#ifdef HAVE_UNISTD_H
21#include <unistd.h>
22#endif /* HAVE_UNISTD_H */
23#ifdef _WIN32
24#include <stdio.h>
25#endif /* _WIN32 */
26#endif /* COAP_AF_UNIX_SUPPORT */
27
28int
30 return !COAP_DISABLE_TCP;
31}
32
33#if !COAP_DISABLE_TCP && !defined(WITH_LWIP) && !defined(WITH_CONTIKI) && !defined(RIOT_VERSION)
34#include <sys/types.h>
35#ifdef HAVE_SYS_SOCKET_H
36# include <sys/socket.h>
37# define OPTVAL_T(t) (t)
38# define OPTVAL_GT(t) (t)
39#endif
40#ifdef HAVE_SYS_IOCTL_H
41#include <sys/ioctl.h>
42#endif
43#ifdef HAVE_WS2TCPIP_H
44#include <ws2tcpip.h>
45# define OPTVAL_T(t) (const char*)(t)
46# define OPTVAL_GT(t) (char*)(t)
47# undef CMSG_DATA
48# define CMSG_DATA WSA_CMSG_DATA
49#endif
50
51int
53 const coap_address_t *local_if,
54 const coap_address_t *server,
55 int default_port,
56 coap_address_t *local_addr,
57 coap_address_t *remote_addr) {
58 int on = 1;
59#if COAP_IPV6_SUPPORT
60 int off = 0;
61#endif /* COAP_IPV6_SUPPORT */
62#ifdef _WIN32
63 u_long u_on = 1;
64#endif
65 coap_address_t connect_addr;
66 coap_address_copy(&connect_addr, server);
67
68 sock->flags &= ~COAP_SOCKET_CONNECTED;
69 sock->fd = socket(server->addr.sa.sa_family, SOCK_STREAM, 0);
70
71 if (sock->fd == COAP_INVALID_SOCKET) {
72 coap_log_warn("coap_socket_connect_tcp1: socket: %s\n",
74 goto error;
75 }
76
77#ifdef _WIN32
78 if (ioctlsocket(sock->fd, FIONBIO, &u_on) == COAP_SOCKET_ERROR) {
79#else
80 if (ioctl(sock->fd, FIONBIO, &on) == COAP_SOCKET_ERROR) {
81#endif
82 coap_log_warn("coap_socket_connect_tcp1: ioctl FIONBIO: %s\n",
84 }
85
86 switch (server->addr.sa.sa_family) {
87#if COAP_IPV4_SUPPORT
88 case AF_INET:
89 if (connect_addr.addr.sin.sin_port == 0)
90 connect_addr.addr.sin.sin_port = htons(default_port);
91 break;
92#endif /* COAP_IPV4_SUPPORT */
93#if COAP_IPV6_SUPPORT
94 case AF_INET6:
95 if (connect_addr.addr.sin6.sin6_port == 0)
96 connect_addr.addr.sin6.sin6_port = htons(default_port);
97 /* Configure the socket as dual-stacked */
98 if (setsockopt(sock->fd, IPPROTO_IPV6, IPV6_V6ONLY, OPTVAL_T(&off),
99 sizeof(off)) == COAP_SOCKET_ERROR)
100 coap_log_warn("coap_socket_connect_tcp1: setsockopt IPV6_V6ONLY: %s\n",
102 break;
103#endif /* COAP_IPV6_SUPPORT */
104#if COAP_AF_UNIX_SUPPORT
105 case AF_UNIX:
106 break;
107#endif /* COAP_AF_UNIX_SUPPORT */
108 default:
109 coap_log_alert("coap_socket_connect_tcp1: unsupported sa_family\n");
110 break;
111 }
112
113 if (local_if && local_if->addr.sa.sa_family) {
114 coap_address_copy(local_addr, local_if);
115 if (setsockopt(sock->fd, SOL_SOCKET, SO_REUSEADDR, OPTVAL_T(&on), sizeof(on)) == COAP_SOCKET_ERROR)
116 coap_log_warn("coap_socket_connect_tcp1: setsockopt SO_REUSEADDR: %s\n",
118 if (bind(sock->fd, &local_if->addr.sa,
120 local_if->addr.sa.sa_family == AF_INET ?
121 (socklen_t)sizeof(struct sockaddr_in) :
122#endif /* COAP_IPV4_SUPPORT */
123 (socklen_t)local_if->size) == COAP_SOCKET_ERROR) {
124 coap_log_warn("coap_socket_connect_tcp1: bind: %s\n",
126 goto error;
127 }
128 } else {
129 local_addr->addr.sa.sa_family = server->addr.sa.sa_family;
130 }
131
132 if (connect(sock->fd, &connect_addr.addr.sa, connect_addr.size) == COAP_SOCKET_ERROR) {
133#ifdef _WIN32
134 if (WSAGetLastError() == WSAEWOULDBLOCK) {
135#else
136 if (errno == EINPROGRESS) {
137#endif
138 /*
139 * COAP_SOCKET_CONNECTED needs to be set here as there will be reads/writes
140 * by underlying TLS libraries during connect() and we do not want to
141 * assert() in coap_read_session() or coap_write_session() when called by coap_read()
142 */
144 return 1;
145 }
146 coap_log_warn("coap_socket_connect_tcp1: connect: %s\n",
148 goto error;
149 }
150
151 if (getsockname(sock->fd, &local_addr->addr.sa, &local_addr->size) == COAP_SOCKET_ERROR) {
152 coap_log_warn("coap_socket_connect_tcp1: getsockname: %s\n",
154 }
155
156 if (getpeername(sock->fd, &remote_addr->addr.sa, &remote_addr->size) == COAP_SOCKET_ERROR) {
157 coap_log_warn("coap_socket_connect_tcp1: getpeername: %s\n",
159 }
160
162 return 1;
163
164error:
165#if COAP_AF_UNIX_SUPPORT
166 if (local_if && local_if->addr.sa.sa_family == AF_UNIX) {
167#ifdef _WIN32
168 _unlink(local_if->addr.cun.sun_path);
169#else /* ! _WIN32 */
170 unlink(local_if->addr.cun.sun_path);
171#endif /* ! _WIN32 */
172 }
173#endif /* COAP_AF_UNIX_SUPPORT */
174 coap_socket_close(sock);
175 return 0;
176}
177
178int
180 coap_address_t *local_addr,
181 coap_address_t *remote_addr) {
182 int error = 0;
183#ifdef _WIN32
184 int optlen = (int)sizeof(error);
185#else
186 socklen_t optlen = (socklen_t)sizeof(error);
187#endif
188
190
191 if (getsockopt(sock->fd, SOL_SOCKET, SO_ERROR, OPTVAL_GT(&error),
192 &optlen) == COAP_SOCKET_ERROR) {
193 coap_log_warn("coap_socket_connect_tcp2: getsockopt: %s\n",
195 }
196
197 if (error) {
198 coap_log_warn("coap_socket_connect_tcp2: connect failed: %s\n",
200 coap_socket_close(sock);
201 return 0;
202 }
203
204 if (getsockname(sock->fd, &local_addr->addr.sa, &local_addr->size) == COAP_SOCKET_ERROR) {
205 coap_log_warn("coap_socket_connect_tcp: getsockname: %s\n",
207 }
208
209 if (getpeername(sock->fd, &remote_addr->addr.sa, &remote_addr->size) == COAP_SOCKET_ERROR) {
210 coap_log_warn("coap_socket_connect_tcp: getpeername: %s\n",
212 }
213
214 return 1;
215}
216
217int
219 const coap_address_t *listen_addr,
220 coap_address_t *bound_addr) {
221 int on = 1;
222#if COAP_IPV6_SUPPORT
223 int off = 0;
224#endif /* COAP_IPV6_SUPPORT */
225#ifdef _WIN32
226 u_long u_on = 1;
227#endif
228
229 sock->fd = socket(listen_addr->addr.sa.sa_family, SOCK_STREAM, 0);
230
231 if (sock->fd == COAP_INVALID_SOCKET) {
232 coap_log_warn("coap_socket_bind_tcp: socket: %s\n",
234 goto error;
235 }
236
237#ifdef _WIN32
238 if (ioctlsocket(sock->fd, FIONBIO, &u_on) == COAP_SOCKET_ERROR) {
239#else
240 if (ioctl(sock->fd, FIONBIO, &on) == COAP_SOCKET_ERROR) {
241#endif
242 coap_log_warn("coap_socket_bind_tcp: ioctl FIONBIO: %s\n",
244 }
245 if (setsockopt(sock->fd, SOL_SOCKET, SO_KEEPALIVE, OPTVAL_T(&on),
246 sizeof(on)) == COAP_SOCKET_ERROR)
247 coap_log_warn("coap_socket_bind_tcp: setsockopt SO_KEEPALIVE: %s\n",
249
250 if (setsockopt(sock->fd, SOL_SOCKET, SO_REUSEADDR, OPTVAL_T(&on),
251 sizeof(on)) == COAP_SOCKET_ERROR)
252 coap_log_warn("coap_socket_bind_tcp: setsockopt SO_REUSEADDR: %s\n",
254
255 switch (listen_addr->addr.sa.sa_family) {
256#if COAP_IPV4_SUPPORT
257 case AF_INET:
258 break;
259#endif /* COAP_IPV4_SUPPORT */
260#if COAP_IPV6_SUPPORT
261 case AF_INET6:
262 /* Configure the socket as dual-stacked */
263 if (setsockopt(sock->fd, IPPROTO_IPV6, IPV6_V6ONLY, OPTVAL_T(&off),
264 sizeof(off)) == COAP_SOCKET_ERROR)
265 coap_log_alert("coap_socket_bind_tcp: setsockopt IPV6_V6ONLY: %s\n",
267 break;
268#endif /* COAP_IPV6_SUPPORT */
269#if COAP_AF_UNIX_SUPPORT
270 case AF_UNIX:
271 break;
272#endif /* COAP_AF_UNIX_SUPPORT */
273 default:
274 coap_log_alert("coap_socket_bind_tcp: unsupported sa_family\n");
275 }
276
277 if (bind(sock->fd, &listen_addr->addr.sa,
279 listen_addr->addr.sa.sa_family == AF_INET ?
280 (socklen_t)sizeof(struct sockaddr_in) :
281#endif /* COAP_IPV4_SUPPORT */
282 (socklen_t)listen_addr->size) == COAP_SOCKET_ERROR) {
283 coap_log_alert("coap_socket_bind_tcp: bind: %s\n",
285 goto error;
286 }
287
288 bound_addr->size = (socklen_t)sizeof(*bound_addr);
289 if (getsockname(sock->fd, &bound_addr->addr.sa, &bound_addr->size) < 0) {
290 coap_log_warn("coap_socket_bind_tcp: getsockname: %s\n",
292 goto error;
293 }
294
295 if (listen(sock->fd, 5) == COAP_SOCKET_ERROR) {
296 coap_log_alert("coap_socket_bind_tcp: listen: %s\n",
298 goto error;
299 }
300
301 return 1;
302
303error:
304 coap_socket_close(sock);
305 return 0;
306}
307
308int
310 coap_socket_t *new_client,
311 coap_address_t *local_addr,
312 coap_address_t *remote_addr,
313 void *extra) {
314#ifdef _WIN32
315 u_long u_on = 1;
316#else
317 int on = 1;
318#endif
319 (void)extra;
320
321 new_client->fd = accept(server->fd, &remote_addr->addr.sa,
322 &remote_addr->size);
323 if (new_client->fd == COAP_INVALID_SOCKET) {
324 if (errno != EAGAIN) {
325 coap_log_warn("coap_socket_accept_tcp: accept: %s\n",
327 }
328 return 0;
329 }
330
331 server->flags &= ~COAP_SOCKET_CAN_ACCEPT;
332
333 if (getsockname(new_client->fd, &local_addr->addr.sa, &local_addr->size) < 0)
334 coap_log_warn("coap_socket_accept_tcp: getsockname: %s\n",
336
337#ifdef _WIN32
338 if (ioctlsocket(new_client->fd, FIONBIO, &u_on) == COAP_SOCKET_ERROR) {
339#else
340 if (ioctl(new_client->fd, FIONBIO, &on) == COAP_SOCKET_ERROR) {
341#endif
342 coap_log_warn("coap_socket_accept_tcp: ioctl FIONBIO: %s\n",
344 }
345 return 1;
346}
347#endif /* !COAP_DISABLE_TCP */
void coap_address_copy(coap_address_t *dst, const coap_address_t *src)
#define COAP_IPV4_SUPPORT
const char * coap_socket_format_errno(int error)
Definition coap_io.c:2308
void coap_socket_close(coap_socket_t *sock)
Function interface to close off a socket.
Definition coap_io.c:380
const char * coap_socket_strerror(void)
Definition coap_io.c:2319
#define COAP_SOCKET_ERROR
Definition coap_io.h:49
#define COAP_INVALID_SOCKET
Definition coap_io.h:50
#define COAP_SOCKET_CAN_CONNECT
non blocking client socket can now connect without blocking
#define COAP_SOCKET_WANT_CONNECT
non blocking client socket is waiting for connect
#define COAP_SOCKET_CONNECTED
the socket is connected
Library specific build wrapper for coap_internal.h.
#define coap_log_alert(...)
Definition coap_debug.h:84
#define coap_log_warn(...)
Definition coap_debug.h:102
int coap_tcp_is_supported(void)
Check whether TCP is available.
Definition coap_tcp.c:29
int coap_socket_bind_tcp(coap_socket_t *sock, const coap_address_t *listen_addr, coap_address_t *bound_addr)
Create a new TCP socket and then listen for new incoming TCP sessions.
Definition coap_tcp.c:218
int coap_socket_connect_tcp1(coap_socket_t *sock, const coap_address_t *local_if, const coap_address_t *server, int default_port, coap_address_t *local_addr, coap_address_t *remote_addr)
Create a new TCP socket and initiate the connection.
Definition coap_tcp.c:52
int coap_socket_accept_tcp(coap_socket_t *server, coap_socket_t *new_client, coap_address_t *local_addr, coap_address_t *remote_addr, void *extra)
Accept a new incoming TCP session.
Definition coap_tcp.c:309
int coap_socket_connect_tcp2(coap_socket_t *sock, coap_address_t *local_addr, coap_address_t *remote_addr)
Complete the TCP Connection.
Definition coap_tcp.c:179
Multi-purpose address abstraction.
socklen_t size
size of addr
struct sockaddr_in sin
struct coap_sockaddr_un cun
struct sockaddr_in6 sin6
struct sockaddr sa
union coap_address_t::@0 addr
char sun_path[COAP_UNIX_PATH_MAX]
coap_socket_flags_t flags
1 or more of COAP_SOCKET* flag values