libcoap 4.3.5-develop-1ba3158
Loading...
Searching...
No Matches
coap_netif.c
Go to the documentation of this file.
1/*
2 * coap_netif.c -- Netif functions for libcoap
3 *
4 * Copyright (C) 2023-2025 Jon Shallow <supjps-libcoap@jpshallow.com>
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
19
20/*
21 * return 1 netif still in use.
22 * 0 netif no longer available.
23 */
24int
26 return session->sock.flags != COAP_SOCKET_EMPTY;
27}
28
29#if COAP_SERVER_SUPPORT
30/*
31 * return 1 netif still in use.
32 * 0 netif no longer available.
33 */
34int
36 return endpoint->sock.flags != COAP_SOCKET_EMPTY;
37}
38
39int
41 const coap_address_t *listen_addr) {
42 if (!coap_socket_bind_udp(&endpoint->sock, listen_addr,
43 &endpoint->bind_addr)) {
44 return 0;
45 }
47 return 1;
48}
49#endif /* COAP_SERVER_SUPPORT */
50
51#if COAP_CLIENT_SUPPORT
52int
54 const coap_address_t *server, int default_port) {
55 if (!coap_socket_connect_udp(&session->sock, local_if, server,
56 default_port,
57 &session->addr_info.local,
58 &session->addr_info.remote)) {
59 return 0;
60 }
61 return 1;
62}
63#endif /* COAP_CLIENT_SUPPORT */
64
65/*
66 * dgram
67 * return +ve Number of bytes written.
68 * -1 Error error in errno).
69 * -2 ICMP error response
70 */
71ssize_t
73 ssize_t bytes_read;
74 int keep_errno;
75
76 bytes_read = coap_socket_recv(&session->sock, packet);
77 keep_errno = errno;
78 if (bytes_read == -1) {
79 coap_log_debug("* %s: netif: failed to read %zd bytes (%s) state %d\n",
80 coap_session_str(session), packet->length,
81 coap_socket_strerror(), session->state);
82 errno = keep_errno;
83 } else if (bytes_read > 0) {
84 coap_ticks(&session->last_rx_tx);
85 memcpy(&session->addr_info, &packet->addr_info,
86 sizeof(session->addr_info));
87 coap_log_debug("* %s: netif: recv %4zd bytes\n",
88 coap_session_str(session), bytes_read);
89 }
90 return bytes_read;
91}
92
93#if COAP_SERVER_SUPPORT
94/*
95 * dgram
96 * return +ve Number of bytes written.
97 * -1 Error error in errno).
98 * -2 ICMP error response
99 */
100ssize_t
102 ssize_t bytes_read;
103 int keep_errno;
104
105 bytes_read = coap_socket_recv(&endpoint->sock, packet);
106 keep_errno = errno;
107 if (bytes_read == -1) {
108 if (errno != EAGAIN) {
109 coap_log_debug("* %s: netif: failed to read %zd bytes (%s)\n",
110 coap_endpoint_str(endpoint), packet->length,
112 errno = keep_errno;
113 }
114 } else if (bytes_read > 0) {
115 /* Let the caller do the logging as session available by then */
116 }
117 return bytes_read;
118}
119#endif /* COAP_SERVER_SUPPORT */
120
121/*
122 * dgram
123 * return +ve Number of bytes written.
124 * -1 Error error in errno).
125 */
126ssize_t
127coap_netif_dgrm_write(coap_session_t *session, const uint8_t *data,
128 size_t datalen) {
129 ssize_t bytes_written;
130 int keep_errno;
131
132 coap_socket_t *sock = &session->sock;
133#if COAP_SERVER_SUPPORT
134 if (sock->flags == COAP_SOCKET_EMPTY) {
135 assert(session->endpoint != NULL);
136 sock = &session->endpoint->sock;
137 }
138#endif /* COAP_SERVER_SUPPORT */
139
140 bytes_written = coap_socket_send(sock, session, data, datalen);
141 keep_errno = errno;
142 if (bytes_written <= 0) {
143 coap_log_debug("* %s: netif: failed to send %zd bytes (%s) state %d\n",
144 coap_session_str(session), datalen,
145 coap_socket_strerror(), session->state);
146 errno = keep_errno;
147 } else {
148 coap_ticks(&session->last_rx_tx);
149 if (bytes_written == (ssize_t)datalen)
150 coap_log_debug("* %s: netif: sent %4zd bytes\n",
151 coap_session_str(session), bytes_written);
152 else
153 coap_log_debug("* %s: netif: sent %4zd of %4zd bytes\n",
154 coap_session_str(session), bytes_written, datalen);
155 }
156 return bytes_written;
157}
158
159#if !COAP_DISABLE_TCP
160#if COAP_SERVER_SUPPORT
161int
163 const coap_address_t *listen_addr) {
164 if (!coap_socket_bind_tcp(&endpoint->sock, listen_addr,
165 &endpoint->bind_addr)) {
166 return 0;
167 }
170 return 1;
171}
172
173int
174coap_netif_strm_accept(coap_endpoint_t *endpoint, coap_session_t *session, void *extra) {
175 if (!coap_socket_accept_tcp(&endpoint->sock, &session->sock,
176 &session->addr_info.local,
177 &session->addr_info.remote, extra)) {
178 return 0;
179 }
182 return 1;
183}
184#endif /* COAP_SERVER_SUPPORT */
185
186#if COAP_CLIENT_SUPPORT
187int
189 const coap_address_t *local_if,
190 const coap_address_t *server, int default_port) {
191 if (!coap_socket_connect_tcp1(&session->sock, local_if, server,
192 default_port,
193 &session->addr_info.local,
194 &session->addr_info.remote)) {
195 return 0;
196 }
197 return 1;
198}
199
200int
202 if (!coap_socket_connect_tcp2(&session->sock,
203 &session->addr_info.local,
204 &session->addr_info.remote)) {
205 return 0;
206 }
207 return 1;
208}
209#endif /* COAP_CLIENT_SUPPORT */
210
211/*
212 * strm
213 * return >=0 Number of bytes read.
214 * -1 Error (error in errno).
215 */
216ssize_t
217coap_netif_strm_read(coap_session_t *session, uint8_t *data, size_t datalen) {
218 ssize_t bytes_read = coap_socket_read(&session->sock, data, datalen);
219 int keep_errno = errno;
220
221 if (bytes_read >= 0) {
222 coap_log_debug("* %s: netif: recv %4zd bytes\n",
223 coap_session_str(session), bytes_read);
224 } else if (bytes_read == -1 && errno != EAGAIN) {
225 coap_log_debug("* %s: netif: failed to receive any bytes (%s) state %d\n",
226 coap_session_str(session), coap_socket_strerror(), session->state);
227 errno = keep_errno;
228 }
229 return bytes_read;
230}
231
232/*
233 * strm
234 * return +ve Number of bytes written.
235 * -1 Error (error in errno).
236 */
237ssize_t
238coap_netif_strm_write(coap_session_t *session, const uint8_t *data,
239 size_t datalen) {
240 ssize_t bytes_written = coap_socket_write(&session->sock, data, datalen);
241 int keep_errno = errno;
242
243 if (bytes_written <= 0) {
244 coap_log_debug("* %s: netif: failed to send %zd bytes (%s) state %d\n",
245 coap_session_str(session), datalen,
246 coap_socket_strerror(), session->state);
247 errno = keep_errno;
248 } else {
249 coap_ticks(&session->last_rx_tx);
250 if (bytes_written == (ssize_t)datalen)
251 coap_log_debug("* %s: netif: sent %4zd bytes\n",
252 coap_session_str(session), bytes_written);
253 else
254 coap_log_debug("* %s: netif: sent %4zd of %4zd bytes\n",
255 coap_session_str(session), bytes_written, datalen);
256 }
257 return bytes_written;
258}
259#endif /* COAP_DISABLE_TCP */
260
261void
263 if (coap_netif_available(session))
264 coap_socket_close(&session->sock);
265}
266
267#if COAP_SERVER_SUPPORT
268void
270 coap_socket_close(&endpoint->sock);
271}
272#endif /* COAP_SERVER_SUPPORT */
ssize_t coap_socket_read(coap_socket_t *sock, uint8_t *data, size_t data_len)
Function interface for data stream receiving off a socket.
Definition coap_io.c:683
void coap_socket_close(coap_socket_t *sock)
Function interface to close off a socket.
Definition coap_io.c:377
const char * coap_socket_strerror(void)
Definition coap_io.c:2291
ssize_t coap_socket_recv(coap_socket_t *sock, coap_packet_t *packet)
Function interface for reading data.
Definition coap_io.c:1031
ssize_t coap_socket_send(coap_socket_t *sock, coap_session_t *session, const uint8_t *data, size_t datalen)
Function interface for data transmission.
Definition coap_io.c:830
ssize_t coap_socket_write(coap_socket_t *sock, const uint8_t *data, size_t data_len)
Function interface for data stream sending off a socket.
Definition coap_io.c:624
int coap_socket_connect_udp(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)
#define COAP_SOCKET_WANT_ACCEPT
non blocking server socket is waiting for accept
#define COAP_SOCKET_NOT_EMPTY
the socket is not empty
#define COAP_SOCKET_BOUND
the socket is bound
#define COAP_SOCKET_WANT_READ
non blocking socket is waiting for reading
int coap_socket_bind_udp(coap_socket_t *sock, const coap_address_t *listen_addr, coap_address_t *bound_addr)
#define COAP_SOCKET_CONNECTED
the socket is connected
#define COAP_SOCKET_EMPTY
coap_socket_flags_t values
Library specific build wrapper for coap_internal.h.
CoAP session internal information.
void coap_ticks(coap_tick_t *)
Returns the current value of an internal tick counter.
#define coap_log_debug(...)
Definition coap_debug.h:120
const char * coap_endpoint_str(const coap_endpoint_t *endpoint)
Get endpoint description.
const char * coap_session_str(const coap_session_t *session)
Get session description.
int coap_netif_strm_connect2(coap_session_t *session)
Layer function interface for Netif stream connect (tcp).
ssize_t coap_netif_dgrm_write(coap_session_t *session, const uint8_t *data, size_t datalen)
Function interface for netif datagram data transmission.
Definition coap_netif.c:127
void coap_netif_close(coap_session_t *session)
Layer function interface for Netif close for a session.
Definition coap_netif.c:262
int coap_netif_dgrm_listen(coap_endpoint_t *endpoint, const coap_address_t *listen_addr)
Layer function interface for Netif datagram listem (udp).
ssize_t coap_netif_dgrm_read(coap_session_t *session, coap_packet_t *packet)
Function interface for layer data datagram receiving for sessions.
Definition coap_netif.c:72
void coap_netif_close_ep(coap_endpoint_t *endpoint)
Layer function interface for Netif close for a endpoint.
int coap_netif_strm_connect1(coap_session_t *session, const coap_address_t *local_if, const coap_address_t *server, int default_port)
Layer function interface for Netif stream connect (tcp).
ssize_t coap_netif_dgrm_read_ep(coap_endpoint_t *endpoint, coap_packet_t *packet)
Function interface for layer data datagram receiving for endpoints.
int coap_netif_strm_accept(coap_endpoint_t *endpoint, coap_session_t *session, void *extra)
Layer function interface for Netif stream accept.
int coap_netif_strm_listen(coap_endpoint_t *endpoint, const coap_address_t *listen_addr)
Layer function interface for Netif stream listem (tcp).
ssize_t coap_netif_strm_write(coap_session_t *session, const uint8_t *data, size_t datalen)
Function interface for netif stream data transmission.
Definition coap_netif.c:238
int coap_netif_dgrm_connect(coap_session_t *session, const coap_address_t *local_if, const coap_address_t *server, int default_port)
Layer function interface for Netif datagram connect (udp).
int coap_netif_available(coap_session_t *session)
Function interface to check whether netif for session is still available.
Definition coap_netif.c:25
ssize_t coap_netif_strm_read(coap_session_t *session, uint8_t *data, size_t datalen)
Function interface for layer data stream receiving.
Definition coap_netif.c:217
int coap_netif_available_ep(coap_endpoint_t *endpoint)
Function interface to check whether netif for endpoint is still available.
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:200
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:43
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:291
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:161
coap_address_t remote
remote address and port
Definition coap_io.h:56
coap_address_t local
local address and port
Definition coap_io.h:57
Multi-purpose address abstraction.
Abstraction of virtual endpoint that can be attached to coap_context_t.
coap_address_t bind_addr
local interface address
coap_socket_t sock
socket object for the interface, if any
size_t length
length of payload
coap_addr_tuple_t addr_info
local and remote addresses
Abstraction of virtual session that can be attached to coap_context_t (client) or coap_endpoint_t (se...
coap_endpoint_t * endpoint
session's endpoint
coap_socket_t sock
socket object for the session, if any
coap_session_state_t state
current state of relationship with peer
coap_addr_tuple_t addr_info
remote/local address info
coap_socket_flags_t flags
1 or more of COAP_SOCKET* flag values