1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/net/ccn_lite/ccnl-core.c

1472 lines
42 KiB
C
Raw Normal View History

2013-10-28 10:25:17 +01:00
/*
* @f ccnl-core.c
* @b CCN lite, core CCNx protocol logic
*
* Copyright (C) 2011-13, Christian Tschudin, University of Basel
* Copyright (C) 2013, Christian Mehlis, Freie Universität Berlin
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*
* File history:
* 2011-04-09 created
*/
#define CCNL_VERSION "2013-07-27"
#include <string.h>
#include <stdio.h>
#include "ccnl.h"
#include "ccnl-core.h"
#include "ccnl-pdu.h"
#include "ccnx.h"
#include "ccnl-ext.h"
#include "ccnl-platform.h"
#include "ccnl-includes.h"
#include "ccnl-riot-compat.h"
#define USE_RIOT_MSG
2013-11-29 22:42:56 +01:00
#define CCNL_DYNAMIC_FIB (0)
2013-10-28 10:25:17 +01:00
static struct ccnl_interest_s *ccnl_interest_remove(struct ccnl_relay_s *ccnl,
struct ccnl_interest_s *i);
void ccnl_ll_TX(struct ccnl_relay_s *ccnl, struct ccnl_if_s *ifc,
sockunion *dest, struct ccnl_buf_s *buf);
void free_2ptr_list(void *a, void *b)
{
ccnl_free(a);
ccnl_free(b);
}
void free_3ptr_list(void *a, void *b, void *c)
{
ccnl_free(a);
ccnl_free(b);
ccnl_free(c);
}
void free_4ptr_list(void *a, void *b, void *c, void *d)
{
ccnl_free(a);
ccnl_free(b);
ccnl_free(c);
ccnl_free(d);
}
void free_prefix(struct ccnl_prefix_s *p)
{
if (p) {
free_4ptr_list(p->path, p->comp, p->complen, p);
}
}
void free_content(struct ccnl_content_s *c)
{
free_prefix(c->name);
free_2ptr_list(c->pkt, c);
}
2013-11-29 22:42:56 +01:00
void free_forward(struct ccnl_forward_s *fwd)
{
free_prefix(fwd->prefix);
ccnl_free(fwd);
}
2013-10-28 10:25:17 +01:00
// ----------------------------------------------------------------------
// datastructure support functions
struct ccnl_buf_s *
ccnl_buf_new(void *data, int len)
{
struct ccnl_buf_s *b = (struct ccnl_buf_s *) ccnl_malloc(sizeof(*b) + len);
if (!b) {
return NULL;
}
b->next = NULL;
b->datalen = len;
if (data) {
memcpy(b->data, data, len);
}
return b;
}
struct ccnl_buf_s *buf_dup(struct ccnl_buf_s *B)
{
return (B) ? ccnl_buf_new(B->data, B->datalen) : NULL;
}
int buf_equal(struct ccnl_buf_s *X, struct ccnl_buf_s *Y)
{
return ((X) && (Y) && (X->datalen == Y->datalen) && !memcmp(X->data, Y->data, X->datalen));
}
int ccnl_prefix_cmp(struct ccnl_prefix_s *name, unsigned char *md,
struct ccnl_prefix_s *p, int mode)
/* returns -1 if no match at all (all modes) or exact match failed
returns 0 if full match (CMP_EXACT)
returns n>0 for matched components (CMP_MATCH, CMP_LONGEST) */
{
int i, clen, nlen = name->compcnt + (md ? 1 : 0), rc = -1;
unsigned char *comp;
if (mode == CMP_EXACT && nlen != p->compcnt) {
goto done;
}
for (i = 0; i < nlen && i < p->compcnt; i++) {
comp = i < name->compcnt ? name->comp[i] : md;
clen = i < name->compcnt ? name->complen[i] : 32; // SHA256_DIGEST_LEN
if (clen != p->complen[i] || memcmp(comp, p->comp[i], p->complen[i])) {
rc = mode == CMP_EXACT ? -1 : i;
goto done;
}
}
rc = (mode == CMP_EXACT) ? 0 : i;
done:
DEBUGMSG(49,
"ccnl_prefix_cmp (mode=%d, nlen=%d, plen=%d, %d), name=%s: %d (%p)\n",
mode, nlen, p->compcnt, name->compcnt, ccnl_prefix_to_path(name),
rc, md);
return rc;
}
// ----------------------------------------------------------------------
// ccnb parsing support
static int hunt_for_end(unsigned char **buf, int *len, unsigned char **valptr,
int *vallen)
{
int typ, num;
while (dehead(buf, len, &num, &typ) == 0) {
if (num == 0 && typ == 0) {
return 0;
}
if (consume(typ, num, buf, len, valptr, vallen) < 0) {
return -1;
}
}
return -1;
}
int consume(int typ, int num, unsigned char **buf, int *len,
unsigned char **valptr, int *vallen)
{
if (typ == CCN_TT_BLOB || typ == CCN_TT_UDATA) {
if (valptr) {
*valptr = *buf;
}
if (vallen) {
*vallen = num;
}
*buf += num, *len -= num;
return 0;
}
if (typ == CCN_TT_DTAG || typ == CCN_TT_DATTR) {
return hunt_for_end(buf, len, valptr, vallen);
}
// case CCN_TT_TAG, CCN_TT_ATTR:
// case DTAG, DATTR:
return -1;
}
static int data2uint(unsigned char *cp, int len)
{
int i, val;
for (i = 0, val = 0; i < len; i++)
if (isdigit(cp[i])) {
val = 10 * val + cp[i] - '0';
}
else {
return -1;
}
return val;
}
struct ccnl_buf_s *
ccnl_extract_prefix_nonce_ppkd(unsigned char **data, int *datalen, int *scope,
int *aok, int *min, int *max, struct ccnl_prefix_s **prefix,
struct ccnl_buf_s **nonce, struct ccnl_buf_s **ppkd,
unsigned char **content, int *contlen)
{
unsigned char *start = *data - 2, *cp;
int num, typ, len;
struct ccnl_prefix_s *p;
struct ccnl_buf_s *buf, *n = 0, *pub = 0;
DEBUGMSG(99, "ccnl_extract_prefix\n");
p = (struct ccnl_prefix_s *) ccnl_calloc(1, sizeof(struct ccnl_prefix_s));
if (!p) {
puts("can't get more memory from malloc, dropping ccn msg...");
return NULL;
}
p->comp = (unsigned char **) ccnl_malloc(
CCNL_MAX_NAME_COMP * sizeof(unsigned char **));
p->complen = (int *) ccnl_malloc(CCNL_MAX_NAME_COMP * sizeof(int));
if (!p->comp || !p->complen) {
puts("can't get more memory from malloc, dropping ccn msg...");
goto Bail;
}
while (dehead(data, datalen, &num, &typ) == 0) {
if (num == 0 && typ == 0) {
break; // end
}
if (typ == CCN_TT_DTAG) {
if (num == CCN_DTAG_NAME) {
for (;;) {
if (dehead(data, datalen, &num, &typ) != 0) {
goto Bail;
}
if (num == 0 && typ == 0) {
break;
}
if (typ == CCN_TT_DTAG && num == CCN_DTAG_COMPONENT
&& p->compcnt < CCNL_MAX_NAME_COMP) {
if (hunt_for_end(data, datalen, p->comp + p->compcnt,
p->complen + p->compcnt) < 0) {
goto Bail;
}
p->compcnt++;
}
else {
if (consume(typ, num, data, datalen, 0, 0) < 0) {
goto Bail;
}
}
}
continue;
}
if (num == CCN_DTAG_SCOPE || num == CCN_DTAG_NONCE
|| num == CCN_DTAG_MINSUFFCOMP
|| num == CCN_DTAG_MAXSUFFCOMP
|| num == CCN_DTAG_PUBPUBKDIGEST) {
if (hunt_for_end(data, datalen, &cp, &len) < 0) {
goto Bail;
}
if (num == CCN_DTAG_SCOPE && len == 1 && scope) {
*scope = isdigit(*cp) && (*cp < '3') ? *cp - '0' : -1;
}
if (num == CCN_DTAG_ANSWERORIGKIND && aok) {
*aok = data2uint(cp, len);
}
if (num == CCN_DTAG_MINSUFFCOMP && min) {
*min = data2uint(cp, len);
}
if (num == CCN_DTAG_MAXSUFFCOMP && max) {
*max = data2uint(cp, len);
}
if (num == CCN_DTAG_NONCE && !n) {
n = ccnl_buf_new(cp, len);
}
if (num == CCN_DTAG_PUBPUBKDIGEST && !pub) {
pub = ccnl_buf_new(cp, len);
}
if (num == CCN_DTAG_EXCLUDE) {
DEBUGMSG(49, "warning: 'exclude' field ignored\n");
}
else {
continue;
}
}
if (num == CCN_DTAG_CONTENT || num == CCN_DTAG_CONTENTOBJ) {
if (consume(typ, num, data, datalen, content, contlen) < 0) {
goto Bail;
}
continue;
}
}
if (consume(typ, num, data, datalen, 0, 0) < 0) {
goto Bail;
}
}
if (prefix) {
p->comp[p->compcnt] = NULL;
*prefix = p;
}
else {
free_prefix(p);
}
if (nonce) {
*nonce = n;
}
else {
ccnl_free(n);
}
if (ppkd) {
*ppkd = pub;
}
else {
ccnl_free(pub);
}
buf = ccnl_buf_new(start, *data - start);
if (!buf) {
puts("can't get more memory from malloc, dropping ccn msg...");
}
// carefully rebase ptrs to new buf because of 64bit pointers:
if (content) {
*content = buf->data + (*content - start);
}
for (num = 0; num < p->compcnt; num++) {
p->comp[num] = buf->data + (p->comp[num] - start);
}
return buf;
Bail:
free_prefix(p);
free_2ptr_list(n, pub);
return NULL;
}
// ----------------------------------------------------------------------
// addresses, interfaces and faces
static void ccnl_face_CTS(struct ccnl_relay_s *ccnl, struct ccnl_face_s *f);
int ccnl_addr_cmp(sockunion *s1, sockunion *s2)
{
return (s1->id == s2->id);
}
struct ccnl_face_s *
ccnl_get_face_or_create(struct ccnl_relay_s *ccnl, int ifndx, uint16_t sender_id)
{
DEBUGMSG(1, "ifndx=%d sender_id=%d\n", ifndx, sender_id);
static int i;
struct ccnl_face_s *f;
for (f = ccnl->faces; f; f = f->next) {
DEBUGMSG(1, "f=%p\n", (void *) f);
if (ifndx == f->ifndx && (f->faceid == sender_id)) {
DEBUGMSG(1, "face found! ifidx=%d sender_id=%d faceid=%d\n", ifndx, sender_id, f->faceid);
ccnl_get_timeval(&f->last_used);
2013-10-28 10:25:17 +01:00
return f;
}
}
if (ifndx == -1) {
for (i = 0; i < ccnl->ifcount; i++) {
ifndx = i;
break;
}
if (ifndx == -1) { // no suitable interface found
return NULL;
}
}
f = (struct ccnl_face_s *) ccnl_calloc(1, sizeof(struct ccnl_face_s));
if (!f) {
return NULL;
}
2013-11-29 22:42:56 +01:00
f->faceid = sender_id;
2013-10-28 10:25:17 +01:00
DEBUGMSG(1, "faceid=%d\n", f->faceid);
f->ifndx = ifndx;
2013-11-29 22:42:56 +01:00
if (sender_id == RIOT_BROADCAST) {
f->flags |= CCNL_FACE_FLAGS_BROADCAST;
}
2013-10-28 10:25:17 +01:00
if (ifndx >= 0) {
if (ccnl->defaultFaceScheduler)
f->sched = ccnl->defaultFaceScheduler(ccnl,
(void ( *)(void *, void *)) ccnl_face_CTS);
if (ccnl->ifs[ifndx].reflect) {
f->flags |= CCNL_FACE_FLAGS_REFLECT;
}
if (ccnl->ifs[ifndx].fwdalli) {
f->flags |= CCNL_FACE_FLAGS_FWDALLI;
}
}
f->peer.id = sender_id;
#ifdef USE_FRAG
if (ifndx == RIOT_TRANS_IDX) {
// if newly created face, no fragment policy is defined yet
// turning on fragmentation for riot trans dev based faces
int flagval = CCNL_FACE_FLAGS_STATIC;
f->flags = flagval &
(CCNL_FACE_FLAGS_STATIC | CCNL_FACE_FLAGS_REFLECT);
if (f->frag) {
ccnl_frag_destroy(f->frag);
f->frag = NULL;
}
int mtu = ccnl->ifs[f->ifndx].mtu;
f->frag = ccnl_frag_new(CCNL_FRAG_CCNx2013, mtu); //TODO
}
#endif
ccnl_get_timeval(&f->last_used);
2013-10-28 10:25:17 +01:00
DBL_LINKED_LIST_ADD(ccnl->faces, f);
return f;
}
struct ccnl_face_s *
ccnl_face_remove(struct ccnl_relay_s *ccnl, struct ccnl_face_s *f)
{
struct ccnl_face_s *f2;
struct ccnl_interest_s *pit;
struct ccnl_forward_s **ppfwd;
DEBUGMSG(1, "ccnl_face_remove relay=%p face=%p\n", (void *) ccnl, (void *) f);
ccnl_sched_destroy(f->sched);
ccnl_frag_destroy(f->frag);
for (pit = ccnl->pit; pit;) {
struct ccnl_pendint_s **ppend, *pend;
if (pit->from == f) {
pit->from = NULL;
}
for (ppend = &pit->pending; *ppend;) {
if ((*ppend)->face == f) {
pend = *ppend;
*ppend = pend->next;
ccnl_free(pend);
}
else {
ppend = &(*ppend)->next;
}
}
if (pit->pending) {
pit = pit->next;
}
else {
pit = ccnl_interest_remove(ccnl, pit);
}
}
for (ppfwd = &ccnl->fib; *ppfwd;) {
if ((*ppfwd)->face == f) {
struct ccnl_forward_s *pfwd = *ppfwd;
*ppfwd = pfwd->next;
2013-11-29 22:42:56 +01:00
ccnl_forward_remove(ccnl, pfwd);
2013-10-28 10:25:17 +01:00
}
else {
ppfwd = &(*ppfwd)->next;
}
}
while (f->outq) {
struct ccnl_buf_s *tmp = f->outq->next;
ccnl_free(f->outq);
f->outq = tmp;
}
2013-12-04 01:12:19 +01:00
ccnl_face_print_stat(f);
2013-10-28 10:25:17 +01:00
f2 = f->next;
DBL_LINKED_LIST_REMOVE(ccnl->faces, f);
ccnl_free(f);
return f2;
}
2013-12-04 01:12:19 +01:00
void ccnl_face_print_stat(struct ccnl_face_s *f)
{
DEBUGMSG(1, "ccnl_face_print_stat: faceid=%d ifndx=%d\n", f->faceid, f->ifndx);
DEBUGMSG(1, " STAT interest send=%d:%d:%d:%d:%d\n", f->stat.send_interest[0],
f->stat.send_interest[1], f->stat.send_interest[2],
f->stat.send_interest[3], f->stat.send_interest[4]);
DEBUGMSG(1, " STAT content send=%d:%d:%d:%d:%d\n", f->stat.send_content[0],
f->stat.send_content[1], f->stat.send_content[2],
f->stat.send_content[3], f->stat.send_content[4]);
DEBUGMSG(1, " STAT interest received=%d\n", f->stat.received_interest);
DEBUGMSG(1, " STAT content received=%d\n", f->stat.received_content);
}
2013-10-28 10:25:17 +01:00
void ccnl_interface_cleanup(struct ccnl_if_s *i)
{
int j;
DEBUGMSG(99, "ccnl_interface_cleanup\n");
ccnl_sched_destroy(i->sched);
for (j = 0; j < i->qlen; j++) {
struct ccnl_txrequest_s *r = i->queue
+ (i->qfront + j) % CCNL_MAX_IF_QLEN;
ccnl_free(r->buf);
}
}
// ----------------------------------------------------------------------
// face and interface queues, scheduling
void ccnl_interface_CTS(void *aux1, void *aux2)
{
struct ccnl_relay_s *ccnl = (struct ccnl_relay_s *) aux1;
struct ccnl_if_s *ifc = (struct ccnl_if_s *) aux2;
struct ccnl_txrequest_s *r, req;
DEBUGMSG(25, "ccnl_interface_CTS interface=%p, qlen=%d, sched=%p\n",
(void *) ifc, ifc->qlen, (void *) ifc->sched);
if (ifc->qlen <= 0) {
return;
}
r = ifc->queue + ifc->qfront;
memcpy(&req, r, sizeof(req));
ifc->qfront = (ifc->qfront + 1) % CCNL_MAX_IF_QLEN;
ifc->qlen--;
ccnl_ll_TX(ccnl, ifc, &req.dst, req.buf);
ccnl_free(req.buf);
}
void ccnl_interface_enqueue(void (tx_done)(void *, int, int),
struct ccnl_face_s *f, struct ccnl_relay_s *ccnl, struct ccnl_if_s *ifc,
struct ccnl_buf_s *buf, sockunion *dest)
{
struct ccnl_txrequest_s *r;
DEBUGMSG(25, "ccnl_interface_enqueue interface=%p buf=%p (qlen=%d)\n",
(void *) ifc, (void *) buf, ifc->qlen);
if (ifc->qlen >= CCNL_MAX_IF_QLEN) {
DEBUGMSG(2, " DROPPING buf=%p\n", (void *) buf);
ccnl_free(buf);
return;
}
r = ifc->queue + ((ifc->qfront + ifc->qlen) % CCNL_MAX_IF_QLEN);
r->buf = buf;
memcpy(&r->dst, dest, sizeof(sockunion));
r->txdone = tx_done;
r->txdone_face = f;
ifc->qlen++;
ccnl_interface_CTS(ccnl, ifc);
}
struct ccnl_buf_s *
ccnl_face_dequeue(struct ccnl_relay_s *ccnl, struct ccnl_face_s *f)
{
(void) ccnl; /* unused */
struct ccnl_buf_s *pkt;
DEBUGMSG(20, "ccnl_face_dequeue face=%p (id=%d.%d)\n", (void *) f, ccnl->id,
f->faceid);
if (!f->outq) {
return NULL;
}
pkt = f->outq;
f->outq = pkt->next;
if (!pkt->next) {
f->outqend = NULL;
}
pkt->next = NULL;
return pkt;
}
void ccnl_face_CTS_done(void *ptr, int cnt, int len)
{
DEBUGMSG(99, "ccnl_face_CTS_done face=%p cnt=%d len=%d\n", ptr, cnt, len);
}
void ccnl_face_CTS(struct ccnl_relay_s *ccnl, struct ccnl_face_s *f)
{
struct ccnl_buf_s *buf;
DEBUGMSG(99, "ccnl_face_CTS face=%p sched=%p\n", (void *) f,
(void *) f->sched);
if (!f->frag || f->frag->protocol == CCNL_FRAG_NONE) {
buf = ccnl_face_dequeue(ccnl, f);
if (buf)
ccnl_interface_enqueue(ccnl_face_CTS_done, f, ccnl,
ccnl->ifs + f->ifndx, buf, &f->peer);
}
#ifdef USE_FRAG
else {
sockunion dst;
int ifndx = f->ifndx;
buf = ccnl_frag_getnext(f->frag, &ifndx, &dst);
if (!buf) {
buf = ccnl_face_dequeue(ccnl, f);
ccnl_frag_reset(f->frag, buf, f->ifndx, &f->peer);
buf = ccnl_frag_getnext(f->frag, &ifndx, &dst);
}
if (buf) {
ccnl_interface_enqueue(ccnl_face_CTS_done, f,
ccnl, ccnl->ifs + ifndx, buf, &dst);
}
}
#endif
}
int ccnl_face_enqueue(struct ccnl_relay_s *ccnl, struct ccnl_face_s *to,
struct ccnl_buf_s *buf)
{
struct ccnl_buf_s *msg;
DEBUGMSG(20, "ccnl_face_enqueue face=%p (id=%d.%d) buf=%p len=%d\n",
(void *) to, ccnl->id, to->faceid, (void *) buf, buf->datalen);
for (msg = to->outq; msg; msg = msg->next) // already in the queue?
if (buf_equal(msg, buf)) {
DEBUGMSG(31, " not enqueued because already there\n");
ccnl_free(buf);
return -1;
}
buf->next = NULL;
if (to->outqend) {
to->outqend->next = buf;
}
else {
to->outq = buf;
}
to->outqend = buf;
ccnl_face_CTS(ccnl, to);
return 0;
}
// ----------------------------------------------------------------------
// handling of interest messages
int ccnl_nonce_find_or_append(struct ccnl_relay_s *ccnl,
struct ccnl_buf_s *nonce)
{
struct ccnl_buf_s *n, *n2 = 0;
int i;
DEBUGMSG(99, "ccnl_nonce_find_or_append\n");
for (n = ccnl->nonces, i = 0; n; n = n->next, i++) {
if (buf_equal(n, nonce)) {
return -1;
}
if (n->next) {
n2 = n;
}
}
n = ccnl_buf_new(nonce->data, nonce->datalen);
if (n) {
n->next = ccnl->nonces;
ccnl->nonces = n;
if (i >= CCNL_MAX_NONCES && n2) {
ccnl_free(n2->next);
n2->next = 0;
}
}
return 0;
}
struct ccnl_interest_s *
ccnl_interest_new(struct ccnl_relay_s *ccnl, struct ccnl_face_s *from,
struct ccnl_buf_s **pkt, struct ccnl_prefix_s **prefix, int minsuffix,
int maxsuffix, struct ccnl_buf_s **ppkd)
{
struct ccnl_interest_s *i = (struct ccnl_interest_s *) ccnl_calloc(1,
sizeof(struct ccnl_interest_s));
DEBUGMSG(99, "ccnl_new_interest\n");
if (!i) {
puts("can't get more memory from malloc, dropping ccn msg...");
return NULL;
}
i->from = from;
i->prefix = *prefix;
*prefix = 0;
i->pkt = *pkt;
*pkt = 0;
i->ppkd = *ppkd;
*ppkd = 0;
i->minsuffix = minsuffix;
i->maxsuffix = maxsuffix;
DBL_LINKED_LIST_ADD(ccnl->pit, i);
return i;
}
int ccnl_interest_append_pending(struct ccnl_interest_s *i,
struct ccnl_face_s *from)
{
struct ccnl_pendint_s *pi, *last = NULL;
DEBUGMSG(99, "ccnl_append_pending\n");
for (pi = i->pending; pi; pi = pi->next) { // check whether already listed
if (pi->face == from) {
DEBUGMSG(40, " we found a matching interest, updating time\n");
ccnl_get_timeval(&pi->last_used);
2013-10-28 10:25:17 +01:00
return 0;
}
last = pi;
}
pi = (struct ccnl_pendint_s *) ccnl_calloc(1,
sizeof(struct ccnl_pendint_s));
DEBUGMSG(40, " appending a new pendint entry %p\n", (void *) pi);
if (!pi) {
return -1;
}
pi->face = from;
ccnl_get_timeval(&pi->last_used);
2013-10-28 10:25:17 +01:00
if (last) {
last->next = pi;
}
else {
i->pending = pi;
}
return 0;
}
void ccnl_interest_propagate(struct ccnl_relay_s *ccnl,
struct ccnl_interest_s *i)
{
struct ccnl_forward_s *fwd;
DEBUGMSG(99, "ccnl_interest_propagate\n");
// CONFORM: "A node MUST implement some strategy rule, even if it is only to
// transmit an Interest Message on all listed dest faces in sequence."
// CCNL strategy: we forward on all FWD entries with a prefix match
2013-11-29 22:42:56 +01:00
int forward_cnt = 0;
2013-10-28 10:25:17 +01:00
for (fwd = ccnl->fib; fwd; fwd = fwd->next) {
int rc = ccnl_prefix_cmp(fwd->prefix, NULL, i->prefix, CMP_LONGEST);
DEBUGMSG(40, " ccnl_interest_propagate, rc=%d/%d\n", rc,
fwd->prefix->compcnt);
if (rc < fwd->prefix->compcnt) {
continue;
}
DEBUGMSG(40, " ccnl_interest_propagate, fwd==%p\n", (void *) fwd);
// suppress forwarding to origin of interest, except wireless
if (!i->from || fwd->face != i->from
|| (i->from->flags & CCNL_FACE_FLAGS_REFLECT)) {
2013-11-29 22:42:56 +01:00
i->forwarded_over = fwd;
2013-12-04 01:12:19 +01:00
fwd->face->stat.send_interest[i->retries]++;
2013-10-28 10:25:17 +01:00
ccnl_face_enqueue(ccnl, fwd->face, buf_dup(i->pkt));
2013-11-29 22:42:56 +01:00
ccnl_get_timeval(&fwd->last_used);
forward_cnt++;
2013-10-28 10:25:17 +01:00
}
}
2013-11-29 22:42:56 +01:00
if (forward_cnt == 0) {
2013-12-03 23:30:23 +01:00
DEBUGMSG(40, " ccnl_interest_propagate: using broadcast face!\n");
2013-12-04 01:12:19 +01:00
ccnl->ifs[RIOT_TRANS_IDX].broadcast_face->stat.send_interest[i->retries]++;
2013-11-29 22:42:56 +01:00
ccnl_face_enqueue(ccnl, ccnl->ifs[RIOT_TRANS_IDX].broadcast_face, buf_dup(i->pkt));
2013-10-28 10:25:17 +01:00
}
return;
}
struct ccnl_interest_s *
ccnl_interest_remove(struct ccnl_relay_s *ccnl, struct ccnl_interest_s *i)
{
struct ccnl_interest_s *i2;
DEBUGMSG(40, "ccnl_interest_remove %p\n", (void *) i);
while (i->pending) {
struct ccnl_pendint_s *tmp = i->pending->next;
ccnl_free(i->pending);
i->pending = tmp;
}
i2 = i->next;
DBL_LINKED_LIST_REMOVE(ccnl->pit, i);
free_prefix(i->prefix);
free_3ptr_list(i->ppkd, i->pkt, i);
return i2;
}
// ----------------------------------------------------------------------
// handling of content messages
int ccnl_i_prefixof_c(struct ccnl_prefix_s *prefix, struct ccnl_buf_s *ppkd,
int minsuffix, int maxsuffix, struct ccnl_content_s *c)
{
unsigned char *md;
DEBUGMSG(99, "ccnl_i_prefixof_c prefix=%s min=%d max=%d\n",
ccnl_prefix_to_path(prefix), minsuffix, maxsuffix);
// CONFORM: we do prefix match, honour min. and maxsuffix,
// and check the PublisherPublicKeyDigest if present
// NON-CONFORM: "Note that to match a ContentObject must satisfy
// all of the specifications given in the Interest Message."
// >> CCNL does not honour the exclusion filtering
if ((ppkd && !buf_equal(ppkd, c->ppkd))
|| (prefix->compcnt + minsuffix) > (c->name->compcnt + 1)
|| (prefix->compcnt + maxsuffix) < (c->name->compcnt + 1)) {
return 0;
}
md = NULL;
if ((prefix->compcnt - c->name->compcnt) == 1) {
md = compute_ccnx_digest(c->pkt);
}
return ccnl_prefix_cmp(c->name, md, prefix, CMP_MATCH) == prefix->compcnt;
}
struct ccnl_content_s *
ccnl_content_new(struct ccnl_relay_s *ccnl, struct ccnl_buf_s **pkt,
struct ccnl_prefix_s **prefix, struct ccnl_buf_s **ppkd,
unsigned char *content, int contlen)
{
(void) ccnl; /* unused */
struct ccnl_content_s *c;
// DEBUGMSG(99, "ccnl_content_new <%s>\n",
// prefix == NULL ? NULL : ccnl_prefix_to_path(*prefix));
c = (struct ccnl_content_s *) ccnl_calloc(1, sizeof(struct ccnl_content_s));
if (!c) {
return NULL;
}
ccnl_get_timeval(&c->last_used);
2013-10-28 10:25:17 +01:00
c->content = content;
c->contentlen = contlen;
c->pkt = *pkt;
*pkt = NULL;
c->name = *prefix;
*prefix = NULL;
if (ppkd) {
c->ppkd = *ppkd;
*ppkd = NULL;
}
return c;
}
struct ccnl_content_s *
ccnl_content_remove(struct ccnl_relay_s *ccnl, struct ccnl_content_s *c)
{
struct ccnl_content_s *c2;
DEBUGMSG(99, "ccnl_content_remove\n");
c2 = c->next;
DBL_LINKED_LIST_REMOVE(ccnl->contents, c);
free_content(c);
ccnl->contentcnt--;
return c2;
}
struct ccnl_content_s *
ccnl_content_add2cache(struct ccnl_relay_s *ccnl, struct ccnl_content_s *c)
{
DEBUGMSG(99, "ccnl_content_add2cache (%d/%d)\n", ccnl->contentcnt,
ccnl->max_cache_entries);
if (ccnl->max_cache_entries <= 0) {
DEBUGMSG(1, " content store disabled...\n");
return NULL;
}
while (ccnl->max_cache_entries <= ccnl->contentcnt) {
DEBUGMSG(1, " remove oldest content...\n");
struct ccnl_content_s *c2, *oldest = NULL;
2013-10-28 10:25:17 +01:00
for (c2 = ccnl->contents; c2; c2 = c2->next) {
2013-11-29 22:42:56 +01:00
DEBUGMSG(1, " '%s' -> %ld:%ld\n", ccnl_prefix_to_path(c2->name), c2->last_used.tv_sec, c2->last_used.tv_usec);
2013-10-28 10:25:17 +01:00
if (!(c2->flags & CCNL_CONTENT_FLAGS_STATIC)
2013-11-29 22:42:56 +01:00
&& ((!oldest) || timevaldelta(&c2->last_used, &oldest->last_used) < 0)) {
oldest = c2;
2013-10-28 10:25:17 +01:00
}
}
2013-10-28 10:25:17 +01:00
if (oldest) {
2013-11-29 22:42:56 +01:00
DEBUGMSG(1, " replaced: '%s'\n", ccnl_prefix_to_path(oldest->name));
ccnl_content_remove(ccnl, oldest);
} else {
DEBUGMSG(1, " no dynamic content to remove...\n");
break;
2013-10-28 10:25:17 +01:00
}
}
DEBUGMSG(1, " add new content to store: '%s'\n", ccnl_prefix_to_path(c->name));
2013-10-28 10:25:17 +01:00
DBL_LINKED_LIST_ADD(ccnl->contents, c);
ccnl->contentcnt++;
return c;
}
// deliver new content c to all clients with (loosely) matching interest,
// but only one copy per face
// returns: number of forwards
int ccnl_content_serve_pending(struct ccnl_relay_s *ccnl,
struct ccnl_content_s *c)
{
struct ccnl_interest_s *i;
struct ccnl_face_s *f;
int cnt = 0;
DEBUGMSG(99, "ccnl_content_serve_pending\n");
for (f = ccnl->faces; f; f = f->next) {
f->flags &= ~CCNL_FACE_FLAGS_SERVED; // reply on a face only once
}
for (i = ccnl->pit; i;) {
struct ccnl_pendint_s *pi;
if (!ccnl_i_prefixof_c(i->prefix, i->ppkd, i->minsuffix, i->maxsuffix,
c)) {
i = i->next;
continue;
}
// CONFORM: "Data MUST only be transmitted in response to
// an Interest that matches the Data."
for (pi = i->pending; pi; pi = pi->next) {
if (pi->face->flags & CCNL_FACE_FLAGS_SERVED) {
continue;
}
pi->face->flags |= CCNL_FACE_FLAGS_SERVED;
if (pi->face->ifndx >= 0) {
DEBUGMSG(6, " forwarding content <%s>\n",
ccnl_prefix_to_path(c->name));
2013-12-04 01:12:19 +01:00
pi->face->stat.send_content[c->served_cnt % CCNL_MAX_CONTENT_SERVED_STAT]++;
2013-10-28 10:25:17 +01:00
ccnl_face_enqueue(ccnl, pi->face, buf_dup(c->pkt));
}
c->served_cnt++;
2013-11-29 22:42:56 +01:00
ccnl_get_timeval(&c->last_used);
2013-10-28 10:25:17 +01:00
cnt++;
}
i = ccnl_interest_remove(ccnl, i);
}
return cnt;
}
2013-11-29 22:42:56 +01:00
/**
* returns an entry from the fib which has a common prefix with p.
* only returns that entry if it fullfils the aggregate threshold!
*/
struct ccnl_forward_s *ccn_forward_find_common_prefix_to_aggregate(struct ccnl_relay_s *ccnl, struct ccnl_prefix_s *p, int *match_len)
{
if (!ccnl->fib) {
DEBUGMSG(999, "ccn_forward_find_common_prefix: fib was empty\n");
return NULL;
}
/* fib as at least one enty */
struct ccnl_forward_s *fwd2 = ccnl->fib;
while (fwd2) {
DEBUGMSG(1, "ccn_forward_find_common_prefix: '%s' vs. '%s'\n", ccnl_prefix_to_path(p), ccnl_prefix_to_path(fwd2->prefix));
*match_len = ccnl_prefix_cmp(fwd2->prefix, 0, p, CMP_LONGEST);
/* check for threshold and never date up a static enty */
if ((ccnl->fib_threshold_aggregate <= *match_len) && !(fwd2->flags & CCNL_FORWARD_FLAGS_STATIC)) {
return fwd2;
}
fwd2 = fwd2->next;
}
return NULL;
}
static struct ccnl_forward_s *ccnl_forward_new(struct ccnl_prefix_s *p, struct ccnl_face_s *f, int threshold_prefix, int flags)
{
struct ccnl_forward_s *fwd = ccnl_calloc(1, sizeof(struct ccnl_forward_s));
fwd->prefix = ccnl_prefix_clone_strip(p, threshold_prefix);
fwd->face = f;
fwd->flags = flags;
return fwd;
}
void ccnl_content_learn_name_route(struct ccnl_relay_s *ccnl, struct ccnl_prefix_s *p, struct ccnl_face_s *f, int threshold_prefix, int flags)
{
/*
* we want to insert a new dynamic route in the fib, let's try to aggregate!
* for aggregation we ask the fib for a prefix match
*/
int match_len;
struct ccnl_forward_s *fwd = ccn_forward_find_common_prefix_to_aggregate(ccnl, p, &match_len);
if (!fwd) {
/* there was no prefix match with the user defined creteria. */
/* create a new fib entry */
fwd = ccnl_forward_new(p, f, threshold_prefix, flags);
DBL_LINKED_LIST_ADD(ccnl->fib, fwd);
DEBUGMSG(999, "ccnl_content_learn_name_route: new route '%s' on face %d learned\n", ccnl_prefix_to_path(fwd->prefix), f->faceid);
} else {
/* there was a prefix match with the user defined creteria. */
/* if the new entry has shorter prefix */
if (p->compcnt < fwd->prefix->compcnt) {
/* we need to aggregate! */
DBL_LINKED_LIST_REMOVE(ccnl->fib, fwd);
/* create a new fib entry */
fwd = ccnl_forward_new(p, f, (p->compcnt - match_len), flags);
DBL_LINKED_LIST_ADD(ccnl->fib, fwd);
DEBUGMSG(999, "ccnl_content_learn_name_route: route '%s' on face %d replaced\n", ccnl_prefix_to_path(fwd->prefix), f->faceid);
} else {
/* we don't need to do an update, because we know a shorter prefix already */
}
}
/* refresh fwd entry */
DEBUGMSG(999, "ccnl_content_learn_name_route refresh route '%s' on face %d\n", ccnl_prefix_to_path(fwd->prefix), f->faceid);
ccnl_get_timeval(&fwd->last_used);
}
struct ccnl_forward_s *
ccnl_forward_remove(struct ccnl_relay_s *ccnl, struct ccnl_forward_s *fwd)
{
struct ccnl_forward_s *fwd2;
DEBUGMSG(40, "ccnl_forward_remove %p\n", (void *) fwd);
fwd2 = fwd->next;
DBL_LINKED_LIST_REMOVE(ccnl->fib, fwd);
for (struct ccnl_interest_s *p = ccnl->pit; p; p = p->next) {
if (p->forwarded_over == fwd) {
p->forwarded_over = NULL;
2013-11-29 22:42:56 +01:00
}
}
DEBUGMSG(40, " ccnl_forward_remove next=%p\n", (void *) fwd2);
2013-11-29 22:42:56 +01:00
free_forward(fwd);
return fwd2;
}
bool ccnl_is_timeouted(struct timeval *now, struct timeval *last_used, time_t timeout_s, time_t timeout_us)
{
struct timeval abs_timeout = { last_used->tv_sec + timeout_s, last_used->tv_usec + timeout_us };
return timevaldelta(now, &abs_timeout) > 0;
}
void ccnl_do_retransmit(void *ptr, void *dummy)
2013-10-28 10:25:17 +01:00
{
(void) dummy; /* unused */
struct ccnl_relay_s *relay = (struct ccnl_relay_s *) ptr;
struct ccnl_interest_s *i = relay->pit;
while (i) { // CONFORM: "Entries in the PIT MUST timeout rather
// than being held indefinitely."
if (i->retries > CCNL_MAX_INTEREST_RETRANSMIT) {
2013-10-28 10:25:17 +01:00
i = ccnl_interest_remove(relay, i);
}
else {
// CONFORM: "A node MUST retransmit Interest Messages
// periodically for pending PIT entries."
DEBUGMSG(7, " retransmit %d <%s>\n", i->retries,
ccnl_prefix_to_path(i->prefix));
2013-11-29 22:42:56 +01:00
if (i->forwarded_over
&& !(i->forwarded_over->flags & CCNL_FORWARD_FLAGS_STATIC)
&& (i->retries >= CCNL_MAX_INTEREST_OPTIMISTIC)) {
DEBUGMSG(1, " removed dynamic forward %p\n", (void *) i->forwarded_over);
2013-11-29 22:42:56 +01:00
ccnl_forward_remove(relay, i->forwarded_over);
}
2013-10-28 10:25:17 +01:00
i->retries++;
2013-11-29 22:42:56 +01:00
ccnl_interest_propagate(relay, i);
2013-10-28 10:25:17 +01:00
i = i->next;
}
}
}
void ccnl_do_ageing(void *ptr, void *dummy)
{
(void) dummy; /* unused */
struct ccnl_relay_s *relay = (struct ccnl_relay_s *) ptr;
struct ccnl_content_s *c = relay->contents;
struct ccnl_face_s *f = relay->faces;
struct timeval now;
ccnl_get_timeval(&now);
//DEBUGMSG(999, "ccnl_do_ageing %ld:%ld\n", now.tv_sec, now.tv_usec);
while (c) {
if (ccnl_is_timeouted(&now, &c->last_used, CCNL_CONTENT_TIMEOUT_SEC, CCNL_CONTENT_TIMEOUT_USEC)
&& !(c->flags & CCNL_CONTENT_FLAGS_STATIC)) {
c = ccnl_content_remove(relay, c);
}
else {
c = c->next;
}
}
2013-10-28 10:25:17 +01:00
while (f) {
if (!(f->flags & CCNL_FACE_FLAGS_STATIC)
&& ccnl_is_timeouted(&now, &f->last_used, CCNL_FACE_TIMEOUT_SEC, CCNL_FACE_TIMEOUT_USEC)) {
2013-10-28 10:25:17 +01:00
f = ccnl_face_remove(relay, f);
}
else {
f = f->next;
}
}
2013-11-29 22:42:56 +01:00
struct ccnl_forward_s *fwd = relay->fib;
while (fwd) {
if (!(fwd->flags & CCNL_FORWARD_FLAGS_STATIC)
&& ccnl_is_timeouted(&now, &fwd->last_used, CCNL_FWD_TIMEOUT_SEC, CCNL_FWD_TIMEOUT_USEC)) {
fwd = ccnl_forward_remove(relay, fwd);
} else {
fwd = fwd->next;
}
}
2013-10-28 10:25:17 +01:00
}
void ccnl_core_cleanup(struct ccnl_relay_s *ccnl)
{
int k;
DEBUGMSG(99, "ccnl_core_cleanup %p\n", (void *) ccnl);
while (ccnl->pit) {
ccnl_interest_remove(ccnl, ccnl->pit);
}
while (ccnl->faces) {
ccnl_face_remove(ccnl, ccnl->faces); // also removes all FWD entries
}
while (ccnl->contents) {
ccnl_content_remove(ccnl, ccnl->contents);
}
while (ccnl->nonces) {
struct ccnl_buf_s *tmp = ccnl->nonces->next;
ccnl_free(ccnl->nonces);
ccnl->nonces = tmp;
}
for (k = 0; k < ccnl->ifcount; k++) {
ccnl_interface_cleanup(ccnl->ifs + k);
}
}
// ----------------------------------------------------------------------
// the core logic of CCN:
int ccnl_core_RX_i_or_c(struct ccnl_relay_s *relay, struct ccnl_face_s *from,
unsigned char **data, int *datalen)
{
int rc = -1, scope = 3, aok = 3, minsfx = 0, maxsfx = CCNL_MAX_NAME_COMP,
contlen;
struct ccnl_buf_s *buf = 0, *nonce = 0, *ppkd = 0;
struct ccnl_interest_s *i = 0;
struct ccnl_content_s *c = 0;
struct ccnl_prefix_s *p = 0;
unsigned char *content = 0;
DEBUGMSG(1, "ccnl_core_RX_i_or_c: (%d bytes left)\n", *datalen);
buf = ccnl_extract_prefix_nonce_ppkd(data, datalen, &scope, &aok, &minsfx,
&maxsfx, &p, &nonce, &ppkd, &content, &contlen);
if (!buf) {
DEBUGMSG(6, " parsing error or no prefix\n");
goto Done;
}
if (nonce && ccnl_nonce_find_or_append(relay, nonce)) {
DEBUGMSG(6, " dropped because of duplicate nonce\n");
goto Skip;
}
if (buf->data[0] == 0x01 && buf->data[1] == 0xd2) { // interest
DEBUGMSG(1, "ccnl_core_RX_i_or_c: interest=<%s>\n", ccnl_prefix_to_path(p));
2013-12-04 01:12:19 +01:00
from->stat.received_interest++;
2013-10-28 10:25:17 +01:00
if (p->compcnt > 0 && p->comp[0][0] == (unsigned char) 0xc1) {
goto Skip;
}
if (p->compcnt == 4 && !memcmp(p->comp[0], "ccnx", 4)) {
DEBUGMSG(1, "it's a mgnt msg!\n");
rc = ccnl_mgmt(relay, buf, p, from);
DEBUGMSG(1, "mgnt processing done!\n");
goto Done;
}
// CONFORM: Step 1:
if (aok & 0x01) { // honor "answer-from-existing-content-store" flag
for (c = relay->contents; c; c = c->next) {
if (!ccnl_i_prefixof_c(p, ppkd, minsfx, maxsfx, c)) {
continue;
}
// FIXME: should check stale bit in aok here
DEBUGMSG(7, " matching content for interest, content %p\n",
(void *) c);
2013-12-04 01:12:19 +01:00
from->stat.send_content[c->served_cnt % CCNL_MAX_CONTENT_SERVED_STAT]++;
c->served_cnt++;
2013-10-28 10:25:17 +01:00
if (from->ifndx >= 0) {
ccnl_face_enqueue(relay, from, buf_dup(c->pkt));
}
goto Skip;
}
}
// CONFORM: Step 2: check whether interest is already known
for (i = relay->pit; i; i = i->next) {
if (!ccnl_prefix_cmp(i->prefix, NULL, p, CMP_EXACT)
&& i->minsuffix == minsfx && i->maxsuffix == maxsfx
&& ((!ppkd && !i->ppkd) || buf_equal(ppkd, i->ppkd))) {
break;
}
}
if (!i) { // this is a new/unknown I request: create and propagate
i = ccnl_interest_new(relay, from, &buf, &p, minsfx, maxsfx, &ppkd);
if (i) { // CONFORM: Step 3 (and 4)
DEBUGMSG(7, " created new interest entry %p\n", (void *) i);
if (scope > 2) {
ccnl_interest_propagate(relay, i);
}
}
}
else if (scope > 2 && (from->flags & CCNL_FACE_FLAGS_FWDALLI)) {
DEBUGMSG(7, " old interest, nevertheless propagated %p\n",
(void *) i);
ccnl_interest_propagate(relay, i);
}
if (i) { // store the I request, for the incoming face (Step 3)
DEBUGMSG(7, " appending interest entry %p\n", (void *) i);
ccnl_interest_append_pending(i, from);
}
}
else { // content
DEBUGMSG(6, " content=<%s>\n", ccnl_prefix_to_path(p));
2013-12-04 01:12:19 +01:00
from->stat.received_content++;
2013-10-28 10:25:17 +01:00
// CONFORM: Step 1:
for (c = relay->contents; c; c = c->next)
if (buf_equal(c->pkt, buf)) {
goto Skip; // content is dup
}
c = ccnl_content_new(relay, &buf, &p, &ppkd, content, contlen);
if (c) { // CONFORM: Step 2 (and 3)
if (!ccnl_content_serve_pending(relay, c)) { // unsolicited content
// CONFORM: "A node MUST NOT forward unsolicited data [...]"
DEBUGMSG(7, " removed because no matching interest\n");
free_content(c);
goto Skip;
2013-11-29 22:42:56 +01:00
} else {
#if CCNL_DYNAMIC_FIB
/* content has matched an interest, we considder this name as availeble on this face */
ccnl_content_learn_name_route(relay, c->name, from, relay->fib_threshold_prefix, 0);
#endif
2013-10-28 10:25:17 +01:00
}
if (relay->max_cache_entries != 0) { // it's set to -1 or a limit
DEBUGMSG(7, " adding content to cache\n");
ccnl_content_add2cache(relay, c);
}
else {
DEBUGMSG(7, " content not added to cache\n");
free_content(c);
}
}
}
Skip:
rc = 0;
Done:
free_prefix(p);
free_3ptr_list(buf, nonce, ppkd);
DEBUGMSG(1, "leaving\n");
return rc;
}
int ccnl_core_RX_datagram(struct ccnl_relay_s *relay, struct ccnl_face_s *from,
unsigned char **data, int *datalen)
{
int rc = 0, num = 0, typ = 0;
DEBUGMSG(1, "ccnl_core_RX_datagram: %d bytes from face=%p (id=%d.%d)\n",
*datalen, (void *) from, relay->id, from ? from->faceid : -1);
while (rc >= 0 && *datalen > 0) {
if (dehead(data, datalen, &num, &typ) || typ != CCN_TT_DTAG) {
return -1;
}
switch (num) {
case CCN_DTAG_INTEREST:
DEBUGMSG(1, "INTEREST\n");
case CCN_DTAG_CONTENTOBJ:
rc = ccnl_core_RX_i_or_c(relay, from, data, datalen);
continue;
#ifdef USE_FRAG
case CCNL_DTAG_FRAGMENT2012:
rc = ccnl_frag_RX_frag2012(ccnl_core_RX_datagram, relay, from, data, datalen);
continue;
case CCNL_DTAG_FRAGMENT:
rc = ccnl_frag_RX_CCNx2013(ccnl_core_RX_datagram, relay, from, data, datalen);
continue;
#endif
default:
DEBUGMSG(15, " unknown datagram type %d\n", num);
return -1;
}
}
return rc;
}
void
ccnl_core_RX(struct ccnl_relay_s *relay, int ifndx, unsigned char *data,
int datalen, uint16_t sender_id)
{
struct ccnl_face_s *from;
DEBUGMSG(14, "ccnl_core_RX ifndx=%d, %d bytes\n", ifndx, datalen);
from = ccnl_get_face_or_create(relay, ifndx, sender_id);
DEBUGMSG(1, "ccnl_core_RX: faceid=%d frag=%p\n", from->faceid, (void *) from->frag);
if (!from) {
return;
}
ccnl_core_RX_datagram(relay, from, &data, &datalen);
}
const char *
compile_string(void)
{
static const char *cp = ""
#ifdef USE_CCNxDIGEST
"USE_CCNxDIGEST "
#endif
#ifdef USE_DEBUG
"USE_DEBUG "
#endif
#ifdef USE_DEBUG_MALLOC
"USE_DEBUG_MALLOC "
#endif
#ifdef USE_FRAG
"USE_FRAG "
#endif
#ifdef USE_ETHERNET
"USE_ETHERNET "
#endif
#ifdef USE_UDP
"USE_UDP "
#endif
#ifdef USE_HTTP_STATUS
"USE_HTTP_STATUS "
#endif
#ifdef USE_MGMT
"USE_MGMT "
#endif
#ifdef USE_SCHEDULER
"USE_SCHEDULER "
#endif
#ifdef USE_UNIXSOCKET
"USE_UNIXSOCKET "
#endif
#ifdef USE_APPSERVER
"USE_APPSERVER "
#endif
;
return cp;
}
// eof