18#if defined(RIOT_VERSION)
21#include "net/gnrc/ipv6.h"
22#include "net/gnrc/netreg.h"
27#include "net/sock/async.h"
31#define COAP_SELECT_THREAD_FLAG (1U << 4)
43 ssize_t bytes_written = 0;
50 bytes_written = (ssize_t)datalen;
52 bytes_written = sock_udp_send(&sock->udp, data, datalen,
NULL);
54 bytes_written = sock_udp_send(&sock->udp, data, datalen, &session->
addr_info.
remote.riot);
57 if (bytes_written < 0) {
58 errno = -bytes_written;
83 sock->
flags &= ~COAP_SOCKET_CAN_READ;
91 if (errno == ECONNREFUSED || errno == EHOSTUNREACH || errno == ECONNRESET) {
99 if (errno != EAGAIN) {
106 }
else if (len > 0) {
107 packet->
length = (size_t)len;
110 sock_udp_aux_rx_t aux;
111 sock_udp_ep_t remote;
113 aux.flags = SOCK_AUX_GET_LOCAL;
119 if (errno == ECONNREFUSED || errno == EHOSTUNREACH || errno == ECONNRESET) {
127 if (errno != EAGAIN) {
134 }
else if (len > 0) {
135 packet->
length = (size_t)len;
147#if COAP_SERVER_SUPPORT
150udp_recv_endpoint_cb(sock_udp_t *sock, sock_async_flags_t flags,
void *arg) {
154 if (!(flags & (SOCK_ASYNC_MSG_RECV | SOCK_ASYNC_MSG_SENT)))
157 if (flags & SOCK_ASYNC_MSG_RECV)
159 if (endpoint->context->selecting_thread) {
160 thread_flags_set(endpoint->context->selecting_thread,
161 COAP_SELECT_THREAD_FLAG);
171 ret = sock_udp_create(&sock->udp, &listen_addr->riot,
NULL, SOCK_FLAGS_REUSE_EP);
175 coap_log_warn(
"coap_socket_bind_udp: sock_udp_create: %s (%d)\n",
179 ret = sock_udp_get_local(&sock->udp, &bound_addr->riot);
183 coap_log_warn(
"coap_socket_bind_udp: sock_udp_get_local: %s\n",
186 sock_udp_set_cb(&sock->udp, udp_recv_endpoint_cb, sock->endpoint);
196#if COAP_CLIENT_SUPPORT
199udp_recv_session_cb(sock_udp_t *sock, sock_async_flags_t flags,
void *arg) {
203 if (!(flags & (SOCK_ASYNC_MSG_RECV | SOCK_ASYNC_MSG_SENT)))
206 if (flags & SOCK_ASYNC_MSG_RECV)
208 if (session->
context->selecting_thread) {
209 thread_flags_set(session->
context->selecting_thread,
210 COAP_SELECT_THREAD_FLAG);
222 sock_udp_ep_t remote;
231 if (local_if && local_if->riot.family) {
232 if (local_if->riot.family != connect_addr.riot.family) {
233 coap_log_warn(
"coap_socket_connect_udp: local address family != "
234 "remote address family\n");
239 local.netif = SOCK_ADDR_ANY_NETIF;
240 remote.netif = SOCK_ADDR_ANY_NETIF;
241 switch (connect_addr.riot.family) {
244 local.family = AF_INET;
247 memcpy(local.addr.ipv4, &local_if->riot.
addr.ipv4,
sizeof(local.addr.ipv4));
248 local.port = local_if->riot.port;
250 memset(local.addr.ipv4, 0,
sizeof(local.addr.ipv4));
252 remote.family = AF_INET;
253 memcpy(remote.addr.ipv4, &server->riot.
addr.ipv4,
sizeof(remote.addr.ipv4));
254 if (connect_addr.riot.port == 0)
255 connect_addr.riot.port = default_port;
256 remote.port = connect_addr.riot.port;
261 local.family = AF_INET6;
264 memcpy(local.addr.ipv6, &local_if->riot.
addr.ipv6,
sizeof(local.addr.ipv6));
265 local.port = local_if->riot.port;
267 memset(local.addr.ipv6, 0,
sizeof(local.addr.ipv6));
269 remote.family = AF_INET6;
270 memcpy(remote.addr.ipv6, &server->riot.
addr.ipv6,
sizeof(remote.addr.ipv6));
271 if (connect_addr.riot.port == 0)
272 connect_addr.riot.port = htons(default_port);
273 remote.port = connect_addr.riot.port;
277 coap_log_alert(
"coap_socket_connect_udp: unsupported sa_family %d\n",
278 connect_addr.riot.family);
282 ret = sock_udp_create(&sock->udp, &local, &remote, is_mcast ? 0 : SOCK_FLAGS_CONNECT_REMOTE);
286 coap_log_warn(
"coap_socket_connect_udp: sock_udp_create: %s (%d)\n",
290 ret = sock_udp_get_local(&sock->udp, &local);
294 coap_log_warn(
"coap_socket_connect_udp: sock_udp_get_local: %s\n",
297 memcpy(&local_addr->riot, &local,
sizeof(local_addr->riot));
299 ret = sock_udp_get_remote(&sock->udp, &remote);
303 coap_log_warn(
"coap_socket_connect_udp: sock_udp_get_remote: %s\n",
306 memcpy(&remote_addr->riot, &remote,
sizeof(remote_addr->riot));
308 sock_udp_set_cb(&sock->udp, udp_recv_session_cb, sock->
session);
331 sock_udp_close(&sock->udp);
343#pragma GCC diagnostic ignored "-Wunused-function"
int coap_is_mcast(const coap_address_t *a)
Checks if given address a denotes a multicast address.
void coap_address_copy(coap_address_t *dst, const coap_address_t *src)
int coap_debug_send_packet(void)
Check to see whether a packet should be sent or not.
struct coap_endpoint_t coap_endpoint_t
const char * coap_socket_strerror(void)
#define COAP_RXBUFFER_SIZE
#define COAP_SOCKET_MULTICAST
socket is used for multicast communication
#define COAP_SOCKET_SLAVE
socket is a slave socket - do not close
#define COAP_SOCKET_CAN_READ
non blocking socket can now read without blocking
#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.
RIOT-specific definitions for libcoap.
ssize_t coap_socket_recv(coap_socket_t *sock, coap_packet_t *packet)
Function interface for reading data.
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.
void coap_socket_dgrm_close(coap_socket_t *sock)
Function interface to close off a datagram socket.
#define coap_log_alert(...)
const char * coap_session_str(const coap_session_t *session)
Get session description.
#define coap_log_warn(...)
#define coap_log_crit(...)
coap_address_t remote
remote address and port
coap_address_t local
local address and port
Multi-purpose address abstraction.
union coap_address_t::@0 addr
size_t length
length of payload
coap_addr_tuple_t addr_info
local and remote addresses
unsigned char * payload
payload
Abstraction of virtual session that can be attached to coap_context_t (client) or coap_endpoint_t (se...
coap_socket_t sock
socket object for the session, if any
coap_addr_tuple_t addr_info
remote/local address info
coap_context_t * context
session's context
coap_session_t * session
Used to determine session owner.
coap_socket_flags_t flags
1 or more of COAP_SOCKET* flag values