0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2025-01-09 10:31:53 +00:00
bird/lib/mpls.h
Jan Moskyto Matejka e1b7c6177a L3VPN: new protocol
2016-04-07 10:08:43 +02:00

65 lines
1.9 KiB
C

/*
* BIRD Internet Routing Daemon -- MPLS manipulation
*
* (c) 2016 Jan Matejka <mq@ucw.cz>
* (c) 2016 CZ.NIC z.s.p.o.
*
* Can be freely distributed and used under the terms of the GNU GPL.
*/
#ifndef _BIRD_MPLS_H_
#define _BIRD_MPLS_H_
#define MPLS_STACK_LENGTH 8 /* Adjust this if you need deeper MPLS stack */
#define MPLS_PXLEN 20 /* Length of the label in bits. Constant. */
#define MPLS_LABEL_MAX ((1<<MPLS_PXLEN)-1) /* Maximal possible label value. */
/*
* RFC 3032 updated by RFC 5462:
*
* 0 1 2 3
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ Label
* | Label | TC |S| TTL | Stack
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ Entry
*
* Label: Label Value, 20 bits
* TC: Traffic Class, 3 bits
* S: Bottom of Stack, 1 bit
* TTL: Time to Live, 8 bits
*/
typedef struct mpls_stack {
u8 len;
u32 label[MPLS_STACK_LENGTH];
} mpls_stack;
static inline char * const mpls_hton(mpls_stack s) {
static char buf[MPLS_STACK_LENGTH*4];
int i;
for (i = 0; i < s.len; i++) {
buf[i*4 + 0] = s.label[i] >> 12;
buf[i*4 + 1] = s.label[i] >> 4;
buf[i*4 + 2] = (s.label[i] << 4) | (i == s.len - 1 ? 0x1 : 0);
buf[i*4 + 3] = 0;
}
return buf;
}
static inline int mpls_buflen(const char *buf) {
// Looking for the Bottom of Stack set to 4.
int i;
for (i = 0; !(buf[i++*4 + 2] & 0x1); );
return i*4;
}
static inline mpls_stack mpls_ntoh(const char *buf) {
mpls_stack s = { .len = mpls_buflen(buf) };
int i;
for (i = 0; i < s.len; i++)
s.label[i] = (buf[i*4 + 0] << 12) | (buf[i*4 + 1] << 4) | (buf[i*4 + 2] >> 4);
return s;
}
#endif