-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathndn.c
346 lines (297 loc) · 10.7 KB
/
ndn.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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
/*
* Copyright (C) 2016 Wentao Shang
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup net_ndn
* @{
*
* @file
*
* @author Wentao Shang <wentaoshang@gmail.com>
*/
#include "ndn.h"
#include "face-table.h"
#include "app.h"
#include "netif.h"
#include "l2.h"
#include "pit.h"
#include "fib.h"
#include "cs.h"
#include "forwarding-strategy.h"
#include "encoding/ndn-constants.h"
#include "encoding/name.h"
#include "encoding/interest.h"
#include "msg-type.h"
#define ENABLE_DEBUG 1
#include <debug.h>
#include <net/gnrc/netapi.h>
#include <net/gnrc/netif.h>
#include <net/gnrc/netreg.h>
#include <thread.h>
#include <timex.h>
#include <xtimer.h>
#define GNRC_NDN_STACK_SIZE (THREAD_STACKSIZE_DEFAULT)
#define GNRC_NDN_PRIO (THREAD_PRIORITY_MAIN - 3)
#define GNRC_NDN_MSG_QUEUE_SIZE (8U)
#if ENABLE_DEBUG
static char _stack[GNRC_NDN_STACK_SIZE + THREAD_EXTRA_STACKSIZE_PRINTF];
#else
static char _stack[GNRC_NDN_STACK_SIZE];
#endif
kernel_pid_t ndn_pid = KERNEL_PID_UNDEF;
static void _process_interest(kernel_pid_t face_id, int face_type,
ndn_shared_block_t* si)
{
assert(si != NULL);
// check cache table
ndn_shared_block_t* sd = ndn_cs_match(&si->block);
if (sd != NULL) {
ndn_shared_block_release(si);
// return data to incoming face
switch (face_type) {
case NDN_FACE_NETDEV:
DEBUG("ndn: send cached data to netdev face %"
PRIkernel_pid "\n", face_id);
ndn_netif_send(face_id, &sd->block);
ndn_shared_block_release(sd);
break;
case NDN_FACE_APP:
DEBUG("ndn: send cached data to app face %"
PRIkernel_pid "\n", face_id);
ndn_app_send_msg_to_app(face_id, sd, NDN_APP_MSG_TYPE_DATA);
break;
default:
ndn_shared_block_release(sd);
break;
}
return;
}
// check forwarding strategy
ndn_block_t name;
if (ndn_interest_get_name(&si->block, &name) < 0) {
DEBUG("ndn: cannot get name from interest block, drop packet\n");
ndn_shared_block_release(si);
return;
}
ndn_forwarding_strategy_t* strategy = ndn_forwarding_strategy_lookup(&name);
if (strategy == NULL) {
DEBUG("ndn: no forwarding strategy for interest name, drop packet\n");
ndn_shared_block_release(si);
return;
}
assert(strategy->after_receive_interest != NULL);
// add to pit table
ndn_pit_entry_t* pit_entry = NULL;
if (ndn_pit_add(face_id, face_type, si, strategy, &pit_entry) != 0) {
ndn_shared_block_release(si);
return;
}
// invoke forwarding strategy trigger and transfer ownership of si
DEBUG("ndn: invoke forwarding strategy trigger: after_receive_interest\n");
strategy->after_receive_interest(si, face_id, pit_entry);
return;
}
static void _process_data(kernel_pid_t face_id, int face_type,
ndn_shared_block_t* sd)
{
assert(sd != NULL);
(void)face_type;
// match data against pit
if (ndn_pit_match_data(sd, face_id) == 0) {
// found match in pit
// try to add data to CS
ndn_cs_add(sd);
} else {
// otherwise drop unsolicited data
DEBUG("ndn: no matching pit entry found for data\n");
}
ndn_shared_block_release(sd);
}
static void _process_packet(kernel_pid_t face_id, int face_type,
gnrc_pktsnip_t *pkt)
{
assert(pkt != NULL);
assert(pkt->type == GNRC_NETTYPE_NDN);
ndn_shared_block_t* sb = NULL;
uint8_t* buf = (uint8_t*)pkt->data;
/* check if the packet starts with l2 fragmentation header */
if (buf[0] & NDN_L2_FRAG_HB_MASK) {
uint16_t frag_id = (buf[1] << 8) + buf[2];
DEBUG("ndn: l2 fragment received (MF=%x, SEQ=%u, ID=%02x, "
"packet size = %zu, iface=%" PRIkernel_pid ")\n",
(buf[0] & NDN_L2_FRAG_MF_MASK) >> 5,
buf[0] & NDN_L2_FRAG_SEQ_MASK,
frag_id, pkt->size, face_id);
sb = ndn_l2_frag_receive(face_id, pkt, frag_id);
}
else {
ndn_block_t block;
if (ndn_block_from_packet(pkt, &block) != 0) {
DEBUG("ndn: cannot get block from packet\n");
gnrc_pktbuf_release(pkt);
return;
}
sb = ndn_shared_block_create(&block);
gnrc_pktbuf_release(pkt);
}
if (sb != NULL) {
// Read type
uint32_t num;
if (ndn_block_get_var_number(
sb->block.buf, sb->block.len, &num) < 0) {
DEBUG("ndn: cannot read NDN packet type from shared block\n");
ndn_shared_block_release(sb);
return;
}
switch (num) {
case NDN_TLV_INTEREST:
_process_interest(face_id, face_type, sb);
break;
case NDN_TLV_DATA:
_process_data(face_id, face_type, sb);
break;
default:
DEBUG("ndn: unknown reassembled packet type\n");
ndn_shared_block_release(sb);
break;
}
}
return;
}
/* Main event loop for NDN */
static void *_event_loop(void *args)
{
msg_t msg, reply, msg_q[GNRC_NDN_MSG_QUEUE_SIZE];
gnrc_netreg_entry_t me_reg;
(void)args;
msg_init_queue(msg_q, GNRC_NDN_MSG_QUEUE_SIZE);
me_reg.demux_ctx = GNRC_NETREG_DEMUX_CTX_ALL;
me_reg.target.pid = thread_getpid();
/* register interest in all NDN packets */
gnrc_netreg_register(GNRC_NETTYPE_NDN, &me_reg);
/* preinitialize ACK to GET/SET commands*/
reply.type = GNRC_NETAPI_MSG_TYPE_ACK;
/* start event loop */
while (1) {
msg_receive(&msg);
switch (msg.type) {
case NDN_PIT_MSG_TYPE_TIMEOUT:
DEBUG("ndn: PIT TIMEOUT message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
ndn_pit_timeout((msg_t*)msg.content.ptr);
break;
case NDN_L2_FRAG_MSG_TYPE_TIMEOUT:
DEBUG("ndn: L2_FRAG TIMEOUT message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
ndn_l2_frag_timeout((msg_t*)msg.content.ptr);
break;
case NDN_APP_MSG_TYPE_ADD_FACE:
DEBUG("ndn: ADD_FACE message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
if (ndn_face_table_add(
(kernel_pid_t)msg.content.value, NDN_FACE_APP) != 0) {
DEBUG("ndn: failed to add face id %d\n",
(int)msg.content.value);
reply.content.value = 1;
} else {
reply.content.value = 0; // indicate success
}
msg_reply(&msg, &reply);
break;
case NDN_APP_MSG_TYPE_REMOVE_FACE:
DEBUG("ndn: REMOVE_FACE message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
if (ndn_face_table_remove(
(kernel_pid_t)msg.content.value) != 0) {
DEBUG("ndn: failed to remove face id %d\n",
(int)msg.content.value);
reply.content.value = 1;
} else {
reply.content.value = 0; // indicate success
}
msg_reply(&msg, &reply);
break;
case NDN_APP_MSG_TYPE_ADD_FIB:
DEBUG("ndn: ADD_FIB message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
if (ndn_fib_add((ndn_shared_block_t*)msg.content.ptr,
msg.sender_pid,
NDN_FACE_APP) != 0) {
DEBUG("ndn: failed to add fib entry\n");
ndn_shared_block_release(
(ndn_shared_block_t*)msg.content.ptr);
reply.content.value = 1;
} else {
reply.content.value = 0; // indicate success
}
msg_reply(&msg, &reply);
break;
case NDN_APP_MSG_TYPE_ADD_STRATEGY:
DEBUG("ndn: ADD_STRATEGY messages received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
struct _ndn_app_add_strategy_param* param =
(struct _ndn_app_add_strategy_param*)msg.content.ptr;
if (ndn_forwarding_strategy_add(param->prefix,
param->strategy) != 0) {
DEBUG("ndn: failed to add forwarding strategy\n");
ndn_shared_block_release(param->prefix);
reply.content.value = 1;
} else {
reply.content.value = 0;
}
msg_reply(&msg, &reply);
break;
case GNRC_NETAPI_MSG_TYPE_RCV:
DEBUG("ndn: RCV message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
_process_packet(msg.sender_pid, NDN_FACE_NETDEV,
(gnrc_pktsnip_t *)msg.content.ptr);
break;
case NDN_APP_MSG_TYPE_INTEREST:
DEBUG("ndn: INTEREST message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
_process_interest(msg.sender_pid, NDN_FACE_APP,
(ndn_shared_block_t*)msg.content.ptr);
break;
case NDN_APP_MSG_TYPE_DATA:
DEBUG("ndn: DATA message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
_process_data(msg.sender_pid, NDN_FACE_APP,
(ndn_shared_block_t*)msg.content.ptr);
break;
case GNRC_NETAPI_MSG_TYPE_GET:
case GNRC_NETAPI_MSG_TYPE_SET:
reply.content.value = -ENOTSUP;
msg_reply(&msg, &reply);
break;
case GNRC_NETAPI_MSG_TYPE_SND:
DEBUG("ndn: SND message received from pid %"
PRIkernel_pid "\n", msg.sender_pid);
default:
break;
}
}
return NULL;
}
kernel_pid_t ndn_init(void)
{
ndn_face_table_init();
ndn_fib_init();
ndn_netif_auto_add();
ndn_forwarding_strategy_init();
ndn_pit_init();
ndn_cs_init();
/* check if thread is already running */
if (ndn_pid == KERNEL_PID_UNDEF) {
/* start UDP thread */
ndn_pid = thread_create(
_stack, sizeof(_stack), GNRC_NDN_PRIO,
THREAD_CREATE_STACKTEST, _event_loop, NULL, "ndn");
}
return ndn_pid;
}
/** @} */