-
Notifications
You must be signed in to change notification settings - Fork 0
/
microcoap_conn.c
113 lines (92 loc) · 3.84 KB
/
microcoap_conn.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
#include "net/af.h"
#include "net/conn/udp.h"
#ifdef MICROCOAP_DEBUG
#define ENABLE_DEBUG (1)
#else
#define ENABLE_DEBUG (0)
#endif
#include "debug.h"
#include "coap.h"
static uint8_t _udp_buf[512]; /* udp read buffer (max udp payload size) */
uint8_t scratch_raw[1024]; /* microcoap scratch buffer */
coap_rw_buffer_t scratch_buf = { scratch_raw, sizeof(scratch_raw) };
#define COAP_SERVER_PORT (5683)
/*
* Starts a blocking and never-returning loop dispatching CoAP requests.
*
* When using gnrc, make sure the calling thread has an initialized msg queue.
*/
void microcoap_server_loop(void)
{
uint8_t laddr[16] = { 0 };
uint8_t raddr[16] = { 0 };
size_t raddr_len;
uint16_t rport;
conn_udp_t conn;
/* Cria conexão UDP.
* &conn é o objeto de conexão.
* laddr é o endereço local. Como é zero, o endereço na conexão será
* não especificado (conn->addr = 0);
/* Retorna 0 em caso de sucesso. */
int rc = conn_udp_create(&conn, laddr, sizeof(laddr), AF_INET6, COAP_SERVER_PORT);
while (1) {
DEBUG("Waiting for incoming UDP packet...\n");
/* Espera uma conexão UDP e joga o pacote recebido em _udp_buf.
* Retorna o número de bytes lidos ou zero (se tudo estiver OK).
* Qualquer número negativo é um erro. Aparentemente é compatível
* com os valores de errno setados pela função recvfrom() POSIX. */
rc = conn_udp_recvfrom(&conn, (char *)_udp_buf, sizeof(_udp_buf), raddr, &raddr_len, &rport);
if (rc < 0) {
DEBUG("Error in conn_udp_recvfrom(). rc=%u\n", rc);
continue;
}
/* Quantidade de bytes lidos */
size_t n = rc;
coap_packet_t pkt;
DEBUG("Received packet: ");
/* Printa o pacote em formato hexadecimal */
/* Função definida em RIOT/pkg/microcoap/microcoap/coap.c:25 */
coap_dump(_udp_buf, n, true);
DEBUG("\n");
/* Completa o pacote coap PKT com as informações lidas do buffer.
* Retorna 0 em caso de sucesso. Caso contrário, retorna um erro
* definido em RIOT/pkg/microcoap/microcoap/coap.h:112 */
/* Função definida em RIOT/pkg/microcoap/microcoap/coap.c:214 */
if (0 != (rc = coap_parse(&pkt, _udp_buf, n))) {
DEBUG("Bad packet rc=%d\n", rc);
}
else {
coap_packet_t rsppkt;
DEBUG("content:\n");
/* Função definida em RIOT/pkg/microcoap/microcoap/coap.c:204 */
coap_dumpPacket(&pkt);
/* Escaneia os endpoints criados em coap.c (local)
* e chama o handler cujo caminho bater. */
/* Função definida em RIOT/pkg/microcoap/microcoap/coap.c:395 */
coap_handle_req(&scratch_buf, &pkt, &rsppkt);
/* build reply */
size_t rsplen = sizeof(_udp_buf);
/* Constrói o pacote coap. Pega o rsppkt gerado pelo handler
* do endereço anteriormente e constrói no buffer. O formato
* do pacote é definido em:
* http://tools.ietf.org/html/draft-ietf-core-coap-18#section-3
*
* Função definida em RIOT/pkg/microcoap/microcoap/coap.c:265 */
if ((rc = coap_build(_udp_buf, &rsplen, &rsppkt)) != 0) {
DEBUG("coap_build failed rc=%d\n", rc);
}
else {
DEBUG("Sending packet: ");
coap_dump(_udp_buf, rsplen, true);
DEBUG("\n");
DEBUG("content:\n");
coap_dumpPacket(&rsppkt);
/* send reply via UDP */
rc = conn_udp_sendto(_udp_buf, rsplen, NULL, 0, raddr, raddr_len, AF_INET6, COAP_SERVER_PORT, rport);
if (rc < 0) {
DEBUG("Error sending CoAP reply via udp; %u\n", rc);
}
}
}
}
}