2000-04-17 07:53:29 +00:00
|
|
|
/*
|
|
|
|
* BIRD Internet Routing Daemon -- Attribute Operations
|
|
|
|
*
|
|
|
|
* (c) 2000 Martin Mares <mj@ucw.cz>
|
|
|
|
*
|
|
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _BIRD_ATTRS_H_
|
|
|
|
#define _BIRD_ATTRS_H_
|
|
|
|
|
2009-09-17 15:52:36 +00:00
|
|
|
#include <stdint.h>
|
2016-12-07 13:11:28 +00:00
|
|
|
#include "lib/unaligned.h"
|
|
|
|
#include "nest/route.h"
|
|
|
|
|
2009-09-17 15:52:36 +00:00
|
|
|
|
2000-04-17 07:53:29 +00:00
|
|
|
/* a-path.c */
|
|
|
|
|
2000-04-17 10:18:55 +00:00
|
|
|
#define AS_PATH_SET 1 /* Types of path segments */
|
|
|
|
#define AS_PATH_SEQUENCE 2
|
2009-03-18 19:30:21 +00:00
|
|
|
#define AS_PATH_CONFED_SEQUENCE 3
|
|
|
|
#define AS_PATH_CONFED_SET 4
|
2000-04-17 10:18:55 +00:00
|
|
|
|
2008-10-26 21:36:08 +00:00
|
|
|
#define AS_PATH_MAXLEN 10000
|
|
|
|
|
|
|
|
#define AS_TRANS 23456
|
|
|
|
/* AS_TRANS is used when we need to store 32bit ASN larger than 0xFFFF
|
|
|
|
* to 16bit slot (like in 16bit AS_PATH). See RFC 4893 for details
|
|
|
|
*/
|
|
|
|
|
2013-07-09 21:27:10 +00:00
|
|
|
struct f_tree;
|
|
|
|
|
2017-01-22 15:32:42 +00:00
|
|
|
int as_path_valid(byte *data, uint len, int bs, int confed, char *err, uint elen);
|
2016-12-07 13:11:28 +00:00
|
|
|
int as_path_16to32(byte *dst, byte *src, uint len);
|
|
|
|
int as_path_32to16(byte *dst, byte *src, uint len);
|
|
|
|
int as_path_contains_as4(const struct adata *path);
|
|
|
|
int as_path_contains_confed(const struct adata *path);
|
|
|
|
struct adata *as_path_strip_confed(struct linpool *pool, const struct adata *op);
|
2017-01-22 15:32:42 +00:00
|
|
|
struct adata *as_path_prepend2(struct linpool *pool, const struct adata *op, int seq, u32 as);
|
2016-12-07 13:11:28 +00:00
|
|
|
struct adata *as_path_to_old(struct linpool *pool, const struct adata *path);
|
|
|
|
void as_path_cut(struct adata *path, uint num);
|
|
|
|
struct adata *as_path_merge(struct linpool *pool, struct adata *p1, struct adata *p2);
|
|
|
|
void as_path_format(const struct adata *path, byte *buf, uint size);
|
|
|
|
int as_path_getlen(const struct adata *path);
|
|
|
|
int as_path_getlen_int(const struct adata *path, int bs);
|
|
|
|
int as_path_get_first(const struct adata *path, u32 *orig_as);
|
2017-01-22 15:32:42 +00:00
|
|
|
int as_path_get_first_regular(const struct adata *path, u32 *last_as);
|
2016-12-07 13:11:28 +00:00
|
|
|
int as_path_get_last(const struct adata *path, u32 *last_as);
|
|
|
|
u32 as_path_get_last_nonaggregated(const struct adata *path);
|
|
|
|
int as_path_contains(const struct adata *path, u32 as, int min);
|
|
|
|
int as_path_match_set(const struct adata *path, struct f_tree *set);
|
2013-08-14 23:06:47 +00:00
|
|
|
struct adata *as_path_filter(struct linpool *pool, struct adata *path, struct f_tree *set, u32 key, int pos);
|
|
|
|
|
2016-12-07 13:11:28 +00:00
|
|
|
static inline struct adata *as_path_prepend(struct linpool *pool, const struct adata *path, u32 as)
|
2017-01-22 15:32:42 +00:00
|
|
|
{ return as_path_prepend2(pool, path, AS_PATH_SEQUENCE, as); }
|
2016-12-07 13:11:28 +00:00
|
|
|
|
2008-10-26 21:36:08 +00:00
|
|
|
|
2009-04-16 23:48:36 +00:00
|
|
|
#define PM_ASN 0
|
|
|
|
#define PM_QUESTION 1
|
|
|
|
#define PM_ASTERISK 2
|
2009-06-01 17:32:41 +00:00
|
|
|
#define PM_ASN_EXPR 3
|
2016-06-08 14:22:44 +00:00
|
|
|
#define PM_ASN_RANGE 4
|
2000-04-17 11:06:39 +00:00
|
|
|
|
2000-04-17 11:11:33 +00:00
|
|
|
struct f_path_mask {
|
|
|
|
struct f_path_mask *next;
|
2009-04-16 23:48:36 +00:00
|
|
|
int kind;
|
2009-06-01 17:32:41 +00:00
|
|
|
uintptr_t val;
|
2016-06-08 14:22:44 +00:00
|
|
|
uintptr_t val2;
|
2000-04-17 11:11:33 +00:00
|
|
|
};
|
2008-10-26 21:36:08 +00:00
|
|
|
|
2016-12-07 13:11:28 +00:00
|
|
|
int as_path_match(const struct adata *path, struct f_path_mask *mask);
|
|
|
|
|
|
|
|
|
|
|
|
/* Counterparts to appropriate as_path_* functions */
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
aggregator_16to32(byte *dst, byte *src)
|
|
|
|
{
|
|
|
|
put_u32(dst, get_u16(src));
|
|
|
|
memcpy(dst+4, src+2, 4);
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
aggregator_32to16(byte *dst, byte *src)
|
|
|
|
{
|
|
|
|
put_u16(dst, get_u32(src));
|
|
|
|
memcpy(dst+2, src+4, 4);
|
|
|
|
return 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
aggregator_contains_as4(struct adata *a)
|
|
|
|
{
|
|
|
|
return get_u32(a->data) > 0xFFFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct adata *
|
|
|
|
aggregator_to_old(struct linpool *pool, struct adata *a)
|
|
|
|
{
|
|
|
|
struct adata *d = lp_alloc_adata(pool, 8);
|
|
|
|
put_u32(d->data, 0xFFFF);
|
|
|
|
memcpy(d->data + 4, a->data + 4, 4);
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
2000-04-17 11:11:33 +00:00
|
|
|
|
2000-04-17 10:18:55 +00:00
|
|
|
/* a-set.c */
|
|
|
|
|
2011-08-12 19:03:43 +00:00
|
|
|
|
|
|
|
/* Extended Community subtypes (kinds) */
|
|
|
|
#define EC_RT 0x0002
|
|
|
|
#define EC_RO 0x0003
|
|
|
|
|
|
|
|
#define EC_GENERIC 0xFFFF
|
|
|
|
|
|
|
|
/* Transitive bit (for first u32 half of EC) */
|
|
|
|
#define EC_TBIT 0x40000000
|
|
|
|
|
2016-10-01 10:50:29 +00:00
|
|
|
#define ECOMM_LENGTH 8
|
2011-08-12 19:03:43 +00:00
|
|
|
|
|
|
|
/* RFC 4360 3.1. Two-Octet AS Specific Extended Community */
|
|
|
|
static inline u64 ec_as2(u64 kind, u64 key, u64 val)
|
|
|
|
{ return ((kind | 0x0000) << 48) | (key << 32) | val; }
|
|
|
|
|
|
|
|
/* RFC 5668 4-Octet AS Specific BGP Extended Community */
|
|
|
|
static inline u64 ec_as4(u64 kind, u64 key, u64 val)
|
|
|
|
{ return ((kind | 0x0200) << 48) | (key << 16) | val; }
|
|
|
|
|
|
|
|
/* RFC 4360 3.2. IPv4 Address Specific Extended Community */
|
|
|
|
static inline u64 ec_ip4(u64 kind, u64 key, u64 val)
|
|
|
|
{ return ((kind | 0x0100) << 48) | (key << 16) | val; }
|
|
|
|
|
|
|
|
static inline u64 ec_generic(u64 key, u64 val)
|
|
|
|
{ return (key << 32) | val; }
|
|
|
|
|
2016-10-01 10:50:29 +00:00
|
|
|
/* Large community value */
|
|
|
|
typedef struct lcomm {
|
|
|
|
u32 asn;
|
|
|
|
u32 ldp1;
|
|
|
|
u32 ldp2;
|
|
|
|
} lcomm;
|
|
|
|
|
2018-06-07 13:02:05 +00:00
|
|
|
static inline int int_set_get_size(struct adata *set)
|
|
|
|
{ return set->length / 4; }
|
2016-10-01 10:50:29 +00:00
|
|
|
|
2015-05-19 06:53:34 +00:00
|
|
|
int int_set_format(struct adata *set, int way, int from, byte *buf, uint size);
|
2011-08-12 19:03:43 +00:00
|
|
|
int ec_format(byte *buf, u64 ec);
|
2015-05-19 06:53:34 +00:00
|
|
|
int ec_set_format(struct adata *set, int from, byte *buf, uint size);
|
2016-10-01 10:50:29 +00:00
|
|
|
int lc_format(byte *buf, lcomm lc);
|
|
|
|
int lc_set_format(struct adata *set, int from, byte *buf, uint size);
|
2018-06-07 09:49:55 +00:00
|
|
|
|
2018-06-07 13:02:05 +00:00
|
|
|
int set_position(struct adata *list, void *val, int size);
|
|
|
|
static inline int set_contains(struct adata *list, void *val, int size)
|
|
|
|
{ return set_position(list, val, size) != -1; }
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline int int_set_contains(struct adata *list, u32 val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_contains(list, &val, sizeof(val)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline int ec_set_contains(struct adata *list, u64 val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_contains(list, &val, sizeof(val)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline int lc_set_contains(struct adata *list, lcomm val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_contains(list, &val, sizeof(val)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
|
2016-11-15 15:24:39 +00:00
|
|
|
struct adata *int_set_prepend(struct linpool *pool, struct adata *list, u32 val);
|
2018-06-07 09:49:55 +00:00
|
|
|
|
2018-06-07 13:02:05 +00:00
|
|
|
struct adata *set_add(struct linpool *pool, struct adata *list, void *val, int size);
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *int_set_add(struct linpool *pool, struct adata *list, u32 val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_add(pool, list, &val, sizeof(val)); };
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *ec_set_add(struct linpool *pool, struct adata *list, u64 val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_add(pool, list, &val, sizeof(val)); };
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *lc_set_add(struct linpool *pool, struct adata *list, lcomm val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_add(pool, list, &val, sizeof(val)); };
|
2018-06-07 09:49:55 +00:00
|
|
|
|
2018-06-07 13:02:05 +00:00
|
|
|
struct adata *set_del(struct linpool *pool, struct adata *list, void *val, int size);
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *int_set_del(struct linpool *pool, struct adata *list, u32 val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_del(pool, list, &val, sizeof(val)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *ec_set_del(struct linpool *pool, struct adata *list, u64 val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_del(pool, list, &val, sizeof(val)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *lc_set_del(struct linpool *pool, struct adata *list, lcomm val)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_del(pool, list, &val, sizeof(val)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
|
|
|
|
struct adata *set_union(struct linpool *pool, struct adata *l1, struct adata *l2, int size);
|
|
|
|
static inline struct adata *int_set_union(struct linpool *pool, struct adata *l1, struct adata *l2)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_union(pool, l1, l2, sizeof(u32)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *ec_set_union(struct linpool *pool, struct adata *l1, struct adata *l2)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_union(pool, l1, l2, sizeof(u64)); }
|
2018-06-07 09:49:55 +00:00
|
|
|
static inline struct adata *lc_set_union(struct linpool *pool, struct adata *l1, struct adata *l2)
|
2018-06-07 13:02:05 +00:00
|
|
|
{ return set_union(pool, l1, l2, sizeof(lcomm)); }
|
|
|
|
|
2000-04-17 11:34:38 +00:00
|
|
|
|
2016-12-07 13:11:28 +00:00
|
|
|
struct adata *ec_set_del_nontrans(struct linpool *pool, struct adata *set);
|
|
|
|
struct adata *int_set_sort(struct linpool *pool, struct adata *src);
|
|
|
|
struct adata *ec_set_sort(struct linpool *pool, struct adata *src);
|
|
|
|
struct adata *lc_set_sort(struct linpool *pool, struct adata *src);
|
2008-10-26 21:45:09 +00:00
|
|
|
|
2017-12-13 14:57:44 +00:00
|
|
|
void ec_set_sort_x(struct adata *set); /* Sort in place */
|
|
|
|
|
2000-04-17 07:53:29 +00:00
|
|
|
#endif
|