mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-22 08:51:54 +00:00
4ac55615e4
closes #86 The uncork events are running from mainloop so these should just dispatch the right event to the right loop. Doing anything long there is bad for performance and latency as the uncork list may be huge.
3524 lines
86 KiB
C
3524 lines
86 KiB
C
/*
|
|
* BIRD -- BGP Packet Processing
|
|
*
|
|
* (c) 2000 Martin Mares <mj@ucw.cz>
|
|
* (c) 2008--2016 Ondrej Zajicek <santiago@crfreenet.org>
|
|
* (c) 2008--2016 CZ.NIC z.s.p.o.
|
|
*
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
*/
|
|
|
|
#undef LOCAL_DEBUG
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "nest/bird.h"
|
|
#include "nest/iface.h"
|
|
#include "nest/protocol.h"
|
|
#include "nest/route.h"
|
|
#include "lib/attrs.h"
|
|
#include "proto/mrt/mrt.h"
|
|
#include "conf/conf.h"
|
|
#include "lib/unaligned.h"
|
|
#include "lib/flowspec.h"
|
|
#include "lib/socket.h"
|
|
|
|
#include "nest/cli.h"
|
|
|
|
#include "bgp.h"
|
|
#ifdef CONFIG_BMP
|
|
#include "proto/bmp/bmp.h"
|
|
#endif
|
|
|
|
|
|
#define BGP_RR_REQUEST 0
|
|
#define BGP_RR_BEGIN 1
|
|
#define BGP_RR_END 2
|
|
|
|
#define BGP_NLRI_MAX (4 + 1 + 32)
|
|
|
|
#define BGP_MPLS_BOS 1 /* Bottom-of-stack bit */
|
|
#define BGP_MPLS_MAX 10 /* Max number of labels that 24*n <= 255 */
|
|
#define BGP_MPLS_NULL 3 /* Implicit NULL label */
|
|
#define BGP_MPLS_MAGIC 0x800000 /* Magic withdraw label value, RFC 3107 3 */
|
|
|
|
|
|
static struct tbf rl_rcv_update = TBF_DEFAULT_LOG_LIMITS;
|
|
static struct tbf rl_snd_update = TBF_DEFAULT_LOG_LIMITS;
|
|
|
|
/* Table for state -> RFC 6608 FSM error subcodes */
|
|
static byte fsm_err_subcode[BS_MAX] = {
|
|
[BS_OPENSENT] = 1,
|
|
[BS_OPENCONFIRM] = 2,
|
|
[BS_ESTABLISHED] = 3
|
|
};
|
|
|
|
|
|
static struct bgp_channel *
|
|
bgp_get_channel(struct bgp_proto *p, u32 afi)
|
|
{
|
|
uint i;
|
|
|
|
for (i = 0; i < p->channel_count; i++)
|
|
if (p->afi_map[i] == afi)
|
|
return p->channel_map[i];
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static inline void
|
|
put_af3(byte *buf, u32 id)
|
|
{
|
|
put_u16(buf, id >> 16);
|
|
buf[2] = id & 0xff;
|
|
}
|
|
|
|
static inline void
|
|
put_af4(byte *buf, u32 id)
|
|
{
|
|
put_u16(buf, id >> 16);
|
|
buf[2] = 0;
|
|
buf[3] = id & 0xff;
|
|
}
|
|
|
|
static inline u32
|
|
get_af3(byte *buf)
|
|
{
|
|
return (get_u16(buf) << 16) | buf[2];
|
|
}
|
|
|
|
static inline u32
|
|
get_af4(byte *buf)
|
|
{
|
|
return (get_u16(buf) << 16) | buf[3];
|
|
}
|
|
|
|
static void
|
|
init_mrt_bgp_data(struct bgp_conn *conn, struct mrt_bgp_data *d)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
int p_ok = conn->state >= BS_OPENCONFIRM;
|
|
|
|
memset(d, 0, sizeof(struct mrt_bgp_data));
|
|
d->peer_as = p->remote_as;
|
|
d->local_as = p->local_as;
|
|
d->index = (p->neigh && p->neigh->iface) ? p->neigh->iface->index : 0;
|
|
d->af = ipa_is_ip4(p->remote_ip) ? BGP_AFI_IPV4 : BGP_AFI_IPV6;
|
|
d->peer_ip = conn->sk ? conn->sk->daddr : IPA_NONE;
|
|
d->local_ip = conn->sk ? conn->sk->saddr : IPA_NONE;
|
|
d->as4 = p_ok ? p->as4_session : 0;
|
|
}
|
|
|
|
static uint bgp_find_update_afi(byte *pos, uint len);
|
|
|
|
static int
|
|
bgp_estimate_add_path(struct bgp_proto *p, byte *pkt, uint len)
|
|
{
|
|
/* No need to estimate it for other messages than UPDATE */
|
|
if (pkt[18] != PKT_UPDATE)
|
|
return 0;
|
|
|
|
/* 1 -> no channel, 2 -> all channels, 3 -> some channels */
|
|
if (p->summary_add_path_rx < 3)
|
|
return p->summary_add_path_rx == 2;
|
|
|
|
uint afi = bgp_find_update_afi(pkt, len);
|
|
struct bgp_channel *c = bgp_get_channel(p, afi);
|
|
if (!c)
|
|
{
|
|
/* Either frame error (if !afi) or unknown AFI/SAFI,
|
|
will be reported later in regular parsing */
|
|
BGP_TRACE(D_PACKETS, "MRT processing noticed invalid packet");
|
|
return 0;
|
|
}
|
|
|
|
return c->add_path_rx;
|
|
}
|
|
|
|
static void
|
|
bgp_dump_message(struct bgp_conn *conn, byte *pkt, uint len)
|
|
{
|
|
struct mrt_bgp_data d;
|
|
init_mrt_bgp_data(conn, &d);
|
|
|
|
d.message = pkt;
|
|
d.msg_len = len;
|
|
d.add_path = bgp_estimate_add_path(conn->bgp, pkt, len);
|
|
|
|
mrt_dump_bgp_message(&d);
|
|
}
|
|
|
|
void
|
|
bgp_dump_state_change(struct bgp_conn *conn, uint old, uint new)
|
|
{
|
|
struct mrt_bgp_data d;
|
|
init_mrt_bgp_data(conn, &d);
|
|
|
|
d.old_state = old;
|
|
d.new_state = new;
|
|
|
|
mrt_dump_bgp_state_change(&d);
|
|
}
|
|
|
|
static byte *
|
|
bgp_create_notification(struct bgp_conn *conn, byte *buf)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
|
|
BGP_TRACE(D_PACKETS, "Sending NOTIFICATION(code=%d.%d)", conn->notify_code, conn->notify_subcode);
|
|
buf[0] = conn->notify_code;
|
|
buf[1] = conn->notify_subcode;
|
|
memcpy(buf+2, conn->notify_data, conn->notify_size);
|
|
return buf + 2 + conn->notify_size;
|
|
}
|
|
|
|
|
|
/* Capability negotiation as per RFC 5492 */
|
|
|
|
static const struct bgp_af_caps dummy_af_caps = { };
|
|
static const struct bgp_af_caps basic_af_caps = { .ready = 1 };
|
|
|
|
static const struct bgp_af_caps *
|
|
bgp_find_af_caps_(struct bgp_caps *caps, u32 afi)
|
|
{
|
|
struct bgp_af_caps *ac;
|
|
|
|
WALK_AF_CAPS(caps, ac)
|
|
if (ac->afi == afi)
|
|
return ac;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
const struct bgp_af_caps *
|
|
bgp_find_af_caps(struct bgp_caps *caps, u32 afi)
|
|
{
|
|
const struct bgp_af_caps *ac = bgp_find_af_caps_(caps, afi);
|
|
|
|
/* Return proper capability if found */
|
|
if (ac)
|
|
return ac;
|
|
|
|
/* Use default if capabilities were not announced */
|
|
if (!caps->length && (afi == BGP_AF_IPV4))
|
|
return &basic_af_caps;
|
|
|
|
/* Ignore AFIs that were not announced in multiprotocol capability */
|
|
return &dummy_af_caps;
|
|
}
|
|
|
|
static struct bgp_af_caps *
|
|
bgp_get_af_caps(struct bgp_caps **pcaps, u32 afi)
|
|
{
|
|
struct bgp_caps *caps = *pcaps;
|
|
struct bgp_af_caps *ac;
|
|
|
|
WALK_AF_CAPS(caps, ac)
|
|
if (ac->afi == afi)
|
|
return ac;
|
|
|
|
uint n = caps->af_count;
|
|
if (uint_is_pow2(n))
|
|
*pcaps = caps = mb_realloc(caps, sizeof(struct bgp_caps) +
|
|
(2 * n) * sizeof(struct bgp_af_caps));
|
|
|
|
ac = &caps->af_data[caps->af_count++];
|
|
memset(ac, 0, sizeof(struct bgp_af_caps));
|
|
ac->afi = afi;
|
|
|
|
return ac;
|
|
}
|
|
|
|
static int
|
|
bgp_af_caps_cmp(const void *X, const void *Y)
|
|
{
|
|
const struct bgp_af_caps *x = X, *y = Y;
|
|
return (x->afi < y->afi) ? -1 : (x->afi > y->afi) ? 1 : 0;
|
|
}
|
|
|
|
struct bgp_caps *
|
|
bgp_alloc_capabilities(struct bgp_proto *p, int n)
|
|
{
|
|
struct bgp_caps *caps = mb_allocz(p->p.pool, sizeof(struct bgp_caps) + n * sizeof(struct bgp_af_caps));
|
|
caps->role = BGP_ROLE_UNDEFINED;
|
|
return caps;
|
|
}
|
|
|
|
void
|
|
bgp_prepare_capabilities(struct bgp_conn *conn)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
struct bgp_channel *c;
|
|
struct bgp_caps *caps;
|
|
struct bgp_af_caps *ac;
|
|
|
|
if (!p->cf->capabilities)
|
|
{
|
|
/* Just prepare empty local_caps */
|
|
conn->local_caps = bgp_alloc_capabilities(p, 0);
|
|
return;
|
|
}
|
|
|
|
/* Prepare bgp_caps structure */
|
|
int n = list_length(&p->p.channels);
|
|
caps = bgp_alloc_capabilities(p, n);
|
|
conn->local_caps = caps;
|
|
|
|
caps->as4_support = p->cf->enable_as4;
|
|
caps->ext_messages = p->cf->enable_extended_messages;
|
|
caps->route_refresh = p->cf->enable_refresh;
|
|
caps->enhanced_refresh = p->cf->enable_refresh && p->cf->enable_enhanced_refresh;
|
|
caps->role = p->cf->local_role;
|
|
|
|
if (caps->as4_support)
|
|
caps->as4_number = p->public_as;
|
|
|
|
if (p->cf->gr_mode)
|
|
{
|
|
caps->gr_aware = 1;
|
|
caps->gr_time = p->cf->gr_time;
|
|
caps->gr_flags = p->p.gr_recovery ? BGP_GRF_RESTART : 0;
|
|
}
|
|
|
|
if (p->cf->llgr_mode)
|
|
caps->llgr_aware = 1;
|
|
|
|
if (p->cf->enable_hostname && config->hostname)
|
|
{
|
|
size_t length = strlen(config->hostname);
|
|
char *hostname = mb_allocz(p->p.pool, length+1);
|
|
memcpy(hostname, config->hostname, length+1);
|
|
caps->hostname = hostname;
|
|
}
|
|
|
|
/* Allocate and fill per-AF fields */
|
|
BGP_WALK_CHANNELS(p, c)
|
|
{
|
|
ac = &caps->af_data[caps->af_count++];
|
|
ac->afi = c->afi;
|
|
ac->ready = 1;
|
|
|
|
ac->ext_next_hop = bgp_channel_is_ipv4(c) && c->cf->ext_next_hop;
|
|
caps->any_ext_next_hop |= ac->ext_next_hop;
|
|
|
|
ac->add_path = c->cf->add_path;
|
|
caps->any_add_path |= ac->add_path;
|
|
|
|
if (c->cf->gr_able)
|
|
{
|
|
ac->gr_able = 1;
|
|
|
|
if (p->p.gr_recovery)
|
|
ac->gr_af_flags |= BGP_GRF_FORWARDING;
|
|
}
|
|
|
|
if (c->cf->llgr_able)
|
|
{
|
|
ac->llgr_able = 1;
|
|
ac->llgr_time = c->cf->llgr_time;
|
|
|
|
if (p->p.gr_recovery)
|
|
ac->llgr_flags |= BGP_LLGRF_FORWARDING;
|
|
}
|
|
}
|
|
|
|
/* Sort capability fields by AFI/SAFI */
|
|
qsort(caps->af_data, caps->af_count, sizeof(struct bgp_af_caps), bgp_af_caps_cmp);
|
|
}
|
|
|
|
static byte *
|
|
bgp_write_capabilities(struct bgp_conn *conn, byte *buf)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
struct bgp_caps *caps = conn->local_caps;
|
|
struct bgp_af_caps *ac;
|
|
byte *buf_head = buf;
|
|
byte *data;
|
|
|
|
/* Create capability list in buffer */
|
|
|
|
/*
|
|
* Note that max length is ~ 22+21*af_count. With max 12 channels that is
|
|
* 274. We are limited just by buffer size (4096, minus header), as we support
|
|
* extended optional parameres. Therefore, we have enough space for expansion.
|
|
*/
|
|
|
|
WALK_AF_CAPS(caps, ac)
|
|
if (ac->ready)
|
|
{
|
|
*buf++ = 1; /* Capability 1: Multiprotocol extensions */
|
|
*buf++ = 4; /* Capability data length */
|
|
put_af4(buf, ac->afi);
|
|
buf += 4;
|
|
}
|
|
|
|
if (caps->route_refresh)
|
|
{
|
|
*buf++ = 2; /* Capability 2: Support for route refresh */
|
|
*buf++ = 0; /* Capability data length */
|
|
}
|
|
|
|
if (caps->any_ext_next_hop)
|
|
{
|
|
*buf++ = 5; /* Capability 5: Support for extended next hop */
|
|
*buf++ = 0; /* Capability data length, will be fixed later */
|
|
data = buf;
|
|
|
|
WALK_AF_CAPS(caps, ac)
|
|
if (ac->ext_next_hop)
|
|
{
|
|
put_af4(buf, ac->afi);
|
|
put_u16(buf+4, BGP_AFI_IPV6);
|
|
buf += 6;
|
|
}
|
|
|
|
data[-1] = buf - data;
|
|
}
|
|
|
|
if (caps->ext_messages)
|
|
{
|
|
*buf++ = 6; /* Capability 6: Support for extended messages */
|
|
*buf++ = 0; /* Capability data length */
|
|
}
|
|
|
|
if (caps->role != BGP_ROLE_UNDEFINED)
|
|
{
|
|
*buf++ = 9; /* Capability 9: Announce chosen BGP role */
|
|
*buf++ = 1; /* Capability data length */
|
|
*buf++ = caps->role;
|
|
}
|
|
|
|
if (caps->gr_aware)
|
|
{
|
|
*buf++ = 64; /* Capability 64: Support for graceful restart */
|
|
*buf++ = 0; /* Capability data length, will be fixed later */
|
|
data = buf;
|
|
|
|
put_u16(buf, caps->gr_time);
|
|
buf[0] |= caps->gr_flags;
|
|
buf += 2;
|
|
|
|
WALK_AF_CAPS(caps, ac)
|
|
if (ac->gr_able)
|
|
{
|
|
put_af3(buf, ac->afi);
|
|
buf[3] = ac->gr_af_flags;
|
|
buf += 4;
|
|
}
|
|
|
|
data[-1] = buf - data;
|
|
}
|
|
|
|
if (caps->as4_support)
|
|
{
|
|
*buf++ = 65; /* Capability 65: Support for 4-octet AS number */
|
|
*buf++ = 4; /* Capability data length */
|
|
put_u32(buf, p->public_as);
|
|
buf += 4;
|
|
}
|
|
|
|
if (caps->any_add_path)
|
|
{
|
|
*buf++ = 69; /* Capability 69: Support for ADD-PATH */
|
|
*buf++ = 0; /* Capability data length, will be fixed later */
|
|
data = buf;
|
|
|
|
WALK_AF_CAPS(caps, ac)
|
|
if (ac->add_path)
|
|
{
|
|
put_af3(buf, ac->afi);
|
|
buf[3] = ac->add_path;
|
|
buf += 4;
|
|
}
|
|
|
|
data[-1] = buf - data;
|
|
}
|
|
|
|
if (caps->enhanced_refresh)
|
|
{
|
|
*buf++ = 70; /* Capability 70: Support for enhanced route refresh */
|
|
*buf++ = 0; /* Capability data length */
|
|
}
|
|
|
|
if (caps->llgr_aware)
|
|
{
|
|
*buf++ = 71; /* Capability 71: Support for long-lived graceful restart */
|
|
*buf++ = 0; /* Capability data length, will be fixed later */
|
|
data = buf;
|
|
|
|
WALK_AF_CAPS(caps, ac)
|
|
if (ac->llgr_able)
|
|
{
|
|
put_af3(buf, ac->afi);
|
|
buf[3] = ac->llgr_flags;
|
|
put_u24(buf+4, ac->llgr_time);
|
|
buf += 7;
|
|
}
|
|
|
|
data[-1] = buf - data;
|
|
}
|
|
|
|
if (caps->hostname)
|
|
{
|
|
*buf++ = 73; /* Capability 73: Hostname */
|
|
*buf++ = 0; /* Capability data length */
|
|
data = buf;
|
|
|
|
/* Hostname */
|
|
size_t length = strlen(caps->hostname);
|
|
*buf++ = length;
|
|
memcpy(buf, caps->hostname, length);
|
|
buf += length;
|
|
|
|
/* Domain, not implemented */
|
|
*buf++ = 0;
|
|
|
|
data[-1] = buf - data;
|
|
}
|
|
|
|
caps->length = buf - buf_head;
|
|
|
|
return buf;
|
|
}
|
|
|
|
static int
|
|
bgp_read_capabilities(struct bgp_conn *conn, byte *pos, int len)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
struct bgp_caps *caps;
|
|
struct bgp_af_caps *ac;
|
|
uint err_subcode = 0;
|
|
int i, cl;
|
|
u32 af;
|
|
|
|
if (!conn->remote_caps)
|
|
caps = bgp_alloc_capabilities(p, 1);
|
|
else
|
|
{
|
|
caps = conn->remote_caps;
|
|
conn->remote_caps = NULL;
|
|
}
|
|
|
|
caps->length += len;
|
|
|
|
while (len > 0)
|
|
{
|
|
if (len < 2 || len < (2 + pos[1]))
|
|
goto err;
|
|
|
|
/* Capability length */
|
|
cl = pos[1];
|
|
|
|
/* Capability type */
|
|
switch (pos[0])
|
|
{
|
|
case 1: /* Multiprotocol capability, RFC 4760 */
|
|
if (cl != 4)
|
|
goto err;
|
|
|
|
af = get_af4(pos+2);
|
|
ac = bgp_get_af_caps(&caps, af);
|
|
ac->ready = 1;
|
|
break;
|
|
|
|
case 2: /* Route refresh capability, RFC 2918 */
|
|
if (cl != 0)
|
|
goto err;
|
|
|
|
caps->route_refresh = 1;
|
|
break;
|
|
|
|
case 5: /* Extended next hop encoding capability, RFC 8950 */
|
|
if (cl % 6)
|
|
goto err;
|
|
|
|
for (i = 0; i < cl; i += 6)
|
|
{
|
|
/* Specified only for IPv4 prefixes with IPv6 next hops */
|
|
if ((get_u16(pos+2+i+0) != BGP_AFI_IPV4) ||
|
|
(get_u16(pos+2+i+4) != BGP_AFI_IPV6))
|
|
continue;
|
|
|
|
af = get_af4(pos+2+i);
|
|
ac = bgp_get_af_caps(&caps, af);
|
|
ac->ext_next_hop = 1;
|
|
}
|
|
break;
|
|
|
|
case 6: /* Extended message length capability, RFC 8654 */
|
|
if (cl != 0)
|
|
goto err;
|
|
|
|
caps->ext_messages = 1;
|
|
break;
|
|
|
|
case 9: /* BGP role capability, RFC 9234 */
|
|
if (cl != 1)
|
|
goto err;
|
|
|
|
/* Reserved value */
|
|
if (pos[2] == BGP_ROLE_UNDEFINED)
|
|
{ err_subcode = 11; goto err; }
|
|
|
|
/* Multiple inconsistent values */
|
|
if ((caps->role != BGP_ROLE_UNDEFINED) && (caps->role != pos[2]))
|
|
{ err_subcode = 11; goto err; }
|
|
|
|
caps->role = pos[2];
|
|
break;
|
|
|
|
case 64: /* Graceful restart capability, RFC 4724 */
|
|
if (cl % 4 != 2)
|
|
goto err;
|
|
|
|
/* Only the last instance is valid */
|
|
WALK_AF_CAPS(caps, ac)
|
|
{
|
|
ac->gr_able = 0;
|
|
ac->gr_af_flags = 0;
|
|
}
|
|
|
|
caps->gr_aware = 1;
|
|
caps->gr_flags = pos[2] & 0xf0;
|
|
caps->gr_time = get_u16(pos + 2) & 0x0fff;
|
|
|
|
for (i = 2; i < cl; i += 4)
|
|
{
|
|
af = get_af3(pos+2+i);
|
|
ac = bgp_get_af_caps(&caps, af);
|
|
ac->gr_able = 1;
|
|
ac->gr_af_flags = pos[2+i+3];
|
|
}
|
|
break;
|
|
|
|
case 65: /* AS4 capability, RFC 6793 */
|
|
if (cl != 4)
|
|
goto err;
|
|
|
|
caps->as4_support = 1;
|
|
caps->as4_number = get_u32(pos + 2);
|
|
break;
|
|
|
|
case 69: /* ADD-PATH capability, RFC 7911 */
|
|
if (cl % 4)
|
|
goto err;
|
|
|
|
for (i = 0; i < cl; i += 4)
|
|
{
|
|
byte val = pos[2+i+3];
|
|
if (!val || (val > BGP_ADD_PATH_FULL))
|
|
{
|
|
log(L_WARN "%s: Got ADD-PATH capability with unknown value %u, ignoring",
|
|
p->p.name, val);
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < cl; i += 4)
|
|
{
|
|
af = get_af3(pos+2+i);
|
|
ac = bgp_get_af_caps(&caps, af);
|
|
ac->add_path = pos[2+i+3];
|
|
}
|
|
break;
|
|
|
|
case 70: /* Enhanced route refresh capability, RFC 7313 */
|
|
if (cl != 0)
|
|
goto err;
|
|
|
|
caps->enhanced_refresh = 1;
|
|
break;
|
|
|
|
case 71: /* Long lived graceful restart capability, RFC draft */
|
|
if (cl % 7)
|
|
goto err;
|
|
|
|
/* Presumably, only the last instance is valid */
|
|
WALK_AF_CAPS(caps, ac)
|
|
{
|
|
ac->llgr_able = 0;
|
|
ac->llgr_flags = 0;
|
|
ac->llgr_time = 0;
|
|
}
|
|
|
|
caps->llgr_aware = 1;
|
|
|
|
for (i = 0; i < cl; i += 7)
|
|
{
|
|
af = get_af3(pos+2+i);
|
|
ac = bgp_get_af_caps(&caps, af);
|
|
ac->llgr_able = 1;
|
|
ac->llgr_flags = pos[2+i+3];
|
|
ac->llgr_time = get_u24(pos + 2+i+4);
|
|
}
|
|
break;
|
|
|
|
case 73: /* Hostname, RFC draft */
|
|
if ((cl < 2) || (cl < 2 + pos[2]))
|
|
goto err;
|
|
|
|
int length = pos[2];
|
|
char *hostname = mb_allocz(p->p.pool, length+1);
|
|
memcpy(hostname, pos + 3, length);
|
|
hostname[length] = 0;
|
|
|
|
for (i = 0; i < length; i++)
|
|
if (hostname[i] < ' ')
|
|
hostname[i] = ' ';
|
|
|
|
caps->hostname = hostname;
|
|
|
|
/* We can safely ignore all other capabilities */
|
|
}
|
|
|
|
ADVANCE(pos, len, 2 + cl);
|
|
}
|
|
|
|
/* The LLGR capability must be advertised together with the GR capability,
|
|
otherwise it must be disregarded */
|
|
if (!caps->gr_aware && caps->llgr_aware)
|
|
{
|
|
caps->llgr_aware = 0;
|
|
WALK_AF_CAPS(caps, ac)
|
|
{
|
|
ac->llgr_able = 0;
|
|
ac->llgr_flags = 0;
|
|
ac->llgr_time = 0;
|
|
}
|
|
}
|
|
|
|
conn->remote_caps = caps;
|
|
return 0;
|
|
|
|
err:
|
|
mb_free(caps);
|
|
bgp_error(conn, 2, err_subcode, NULL, 0);
|
|
return -1;
|
|
}
|
|
|
|
int
|
|
bgp_check_capabilities(struct bgp_conn *conn)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
struct bgp_caps *local = conn->local_caps;
|
|
struct bgp_caps *remote = conn->remote_caps;
|
|
struct bgp_channel *c;
|
|
int count = 0;
|
|
|
|
/* This is partially overlapping with bgp_conn_enter_established_state(),
|
|
but we need to run this just after we receive OPEN message */
|
|
|
|
if (p->cf->require_refresh && !remote->route_refresh)
|
|
return 0;
|
|
|
|
if (p->cf->require_enhanced_refresh && !remote->enhanced_refresh)
|
|
return 0;
|
|
|
|
if (p->cf->require_as4 && !remote->as4_support)
|
|
return 0;
|
|
|
|
if (p->cf->require_extended_messages && !remote->ext_messages)
|
|
return 0;
|
|
|
|
if (p->cf->require_hostname && !remote->hostname)
|
|
return 0;
|
|
|
|
if (p->cf->require_gr && !remote->gr_aware)
|
|
return 0;
|
|
|
|
if (p->cf->require_llgr && !remote->llgr_aware)
|
|
return 0;
|
|
|
|
/* No check for require_roles, as it uses error code 2.11 instead of 2.7 */
|
|
|
|
BGP_WALK_CHANNELS(p, c)
|
|
{
|
|
const struct bgp_af_caps *loc = bgp_find_af_caps(local, c->afi);
|
|
const struct bgp_af_caps *rem = bgp_find_af_caps(remote, c->afi);
|
|
|
|
/* Find out whether this channel will be active */
|
|
int active = loc->ready && rem->ready;
|
|
|
|
/* Mandatory must be active */
|
|
if (c->cf->mandatory && !active)
|
|
return 0;
|
|
|
|
if (active)
|
|
{
|
|
if (c->cf->require_ext_next_hop && !rem->ext_next_hop)
|
|
return 0;
|
|
|
|
if (c->cf->require_add_path && (loc->add_path & BGP_ADD_PATH_RX) && !(rem->add_path & BGP_ADD_PATH_TX))
|
|
return 0;
|
|
|
|
if (c->cf->require_add_path && (loc->add_path & BGP_ADD_PATH_TX) && !(rem->add_path & BGP_ADD_PATH_RX))
|
|
return 0;
|
|
|
|
count++;
|
|
}
|
|
}
|
|
|
|
/* We need at least one channel active */
|
|
if (!count)
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
bgp_read_options(struct bgp_conn *conn, byte *pos, uint len, uint rest)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
int ext = 0;
|
|
|
|
/* Handle extended length, RFC 9072 */
|
|
if ((len > 0) && (rest > 0) && (pos[0] == 255))
|
|
{
|
|
if (rest < 3)
|
|
goto err;
|
|
|
|
/* Update pos/len to describe optional data */
|
|
len = get_u16(pos+1);
|
|
ext = 1;
|
|
pos += 3;
|
|
rest -= 3;
|
|
}
|
|
|
|
/* Verify that optional data fits into OPEN packet */
|
|
if (len > rest)
|
|
goto err;
|
|
|
|
/* Length of option parameter header */
|
|
uint hlen = ext ? 3 : 2;
|
|
|
|
while (len > 0)
|
|
{
|
|
if (len < hlen)
|
|
goto err;
|
|
|
|
uint otype = get_u8(pos);
|
|
uint olen = ext ? get_u16(pos+1) : get_u8(pos+1);
|
|
|
|
if (len < (hlen + olen))
|
|
goto err;
|
|
|
|
if (otype == 2)
|
|
{
|
|
/* BGP capabilities, RFC 5492 */
|
|
if (p->cf->capabilities)
|
|
if (bgp_read_capabilities(conn, pos + hlen, olen) < 0)
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
/* Unknown option */
|
|
bgp_error(conn, 2, 4, pos, hlen + olen);
|
|
return -1;
|
|
}
|
|
|
|
ADVANCE(pos, len, hlen + olen);
|
|
}
|
|
|
|
/* Prepare empty caps if no capability option was announced */
|
|
if (!conn->remote_caps)
|
|
conn->remote_caps = bgp_alloc_capabilities(p, 0);
|
|
|
|
return 0;
|
|
|
|
err:
|
|
bgp_error(conn, 2, 0, NULL, 0);
|
|
return -1;
|
|
}
|
|
|
|
static byte *
|
|
bgp_copy_open(struct bgp_proto *p, const byte *pkt, uint len)
|
|
{
|
|
char *buf = mb_alloc(p->p.pool, len - BGP_HEADER_LENGTH);
|
|
memcpy(buf, pkt + BGP_HEADER_LENGTH, len - BGP_HEADER_LENGTH);
|
|
return buf;
|
|
}
|
|
|
|
static byte *
|
|
bgp_create_open(struct bgp_conn *conn, byte *buf)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
|
|
BGP_TRACE(D_PACKETS, "Sending OPEN(ver=%d,as=%d,hold=%d,id=%08x)",
|
|
BGP_VERSION, p->public_as, p->cf->hold_time, p->local_id);
|
|
|
|
buf[0] = BGP_VERSION;
|
|
put_u16(buf+1, (p->public_as < 0xFFFF) ? p->public_as : AS_TRANS);
|
|
put_u16(buf+3, p->cf->hold_time);
|
|
put_u32(buf+5, p->local_id);
|
|
|
|
if (p->cf->capabilities)
|
|
{
|
|
/* Prepare local_caps and write capabilities to buffer */
|
|
byte *pos = buf+12;
|
|
byte *end = bgp_write_capabilities(conn, pos);
|
|
uint len = end - pos;
|
|
|
|
if (len < 254)
|
|
{
|
|
buf[9] = len + 2; /* Optional parameters length */
|
|
buf[10] = 2; /* Option 2: Capability list */
|
|
buf[11] = len; /* Option data length */
|
|
}
|
|
else /* Extended length, RFC 9072 */
|
|
{
|
|
/* Move capabilities 4 B forward */
|
|
memmove(buf + 16, pos, len);
|
|
pos = buf + 16;
|
|
end = pos + len;
|
|
|
|
buf[9] = 255; /* Non-ext OP length, fake */
|
|
buf[10] = 255; /* Non-ext OP type, signals extended length */
|
|
put_u16(buf+11, len + 3); /* Extended optional parameters length */
|
|
buf[13] = 2; /* Option 2: Capability list */
|
|
put_u16(buf+14, len); /* Option extended data length */
|
|
}
|
|
|
|
return end;
|
|
}
|
|
else
|
|
{
|
|
buf[9] = 0; /* No optional parameters */
|
|
return buf + 10;
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
static void
|
|
bgp_rx_open(struct bgp_conn *conn, byte *pkt, uint len)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
struct bgp_conn *other;
|
|
u32 asn, hold, id;
|
|
|
|
/* Check state */
|
|
if (conn->state != BS_OPENSENT)
|
|
{ bgp_error(conn, 5, fsm_err_subcode[conn->state], NULL, 0); return; }
|
|
|
|
/* Check message length */
|
|
if (len < 29)
|
|
{ bgp_error(conn, 1, 2, pkt+16, 2); return; }
|
|
|
|
if (pkt[19] != BGP_VERSION)
|
|
{ u16 val = BGP_VERSION; bgp_error(conn, 2, 1, (byte *) &val, 2); return; }
|
|
|
|
asn = get_u16(pkt+20);
|
|
hold = get_u16(pkt+22);
|
|
id = get_u32(pkt+24);
|
|
BGP_TRACE(D_PACKETS, "Got OPEN(as=%d,hold=%d,id=%R)", asn, hold, id);
|
|
|
|
conn->remote_open_msg = bgp_copy_open(p, pkt, len);
|
|
conn->remote_open_length = len - BGP_HEADER_LENGTH;
|
|
|
|
if (bgp_read_options(conn, pkt+29, pkt[28], len-29) < 0)
|
|
return;
|
|
|
|
/* RFC 4271 4.2 - hold time must be either 0 or at least 3 */
|
|
if (hold > 0 && hold < 3)
|
|
{ bgp_error(conn, 2, 6, pkt+22, 2); return; }
|
|
|
|
/* Compute effective hold and keepalive times */
|
|
uint hold_time = MIN(hold, p->cf->hold_time);
|
|
uint keepalive_time = p->cf->keepalive_time ?
|
|
(p->cf->keepalive_time * hold_time / p->cf->hold_time) :
|
|
hold_time / 3;
|
|
|
|
uint send_hold_time = (p->cf->send_hold_time >= 0) ?
|
|
(p->cf->send_hold_time * hold_time / p->cf->hold_time) :
|
|
2 * hold_time;
|
|
|
|
/* Keepalive time might be rounded down to zero */
|
|
if (hold_time && !keepalive_time)
|
|
keepalive_time = 1;
|
|
|
|
/* Check effective values against configured minimums */
|
|
if ((hold_time < p->cf->min_hold_time) ||
|
|
(keepalive_time < p->cf->min_keepalive_time))
|
|
{ bgp_error(conn, 2, 6, pkt+22, 2); return; }
|
|
|
|
/* RFC 6286 2.2 - router ID is nonzero and AS-wide unique */
|
|
if (!id || (p->is_internal && id == p->local_id))
|
|
{ bgp_error(conn, 2, 3, pkt+24, -4); return; }
|
|
|
|
/* RFC 5492 5 - check for required capabilities */
|
|
if (p->cf->capabilities && !bgp_check_capabilities(conn))
|
|
{ bgp_error(conn, 2, 7, NULL, 0); return; }
|
|
|
|
struct bgp_caps *caps = conn->remote_caps;
|
|
|
|
if (caps->as4_support)
|
|
{
|
|
u32 as4 = caps->as4_number;
|
|
|
|
if ((as4 != asn) && (asn != AS_TRANS))
|
|
log(L_WARN "%s: Peer advertised inconsistent AS numbers", p->p.name);
|
|
|
|
/* When remote ASN is unspecified, it must be external one */
|
|
if (p->remote_as ? (as4 != p->remote_as) : (as4 == p->local_as))
|
|
{ as4 = htonl(as4); bgp_error(conn, 2, 2, (byte *) &as4, 4); return; }
|
|
|
|
conn->received_as = as4;
|
|
}
|
|
else
|
|
{
|
|
if (p->remote_as ? (asn != p->remote_as) : (asn == p->local_as))
|
|
{ bgp_error(conn, 2, 2, pkt+20, 2); return; }
|
|
|
|
conn->received_as = asn;
|
|
}
|
|
|
|
/* RFC 9234 4.2 - check role agreement */
|
|
u8 local_role = p->cf->local_role;
|
|
u8 neigh_role = caps->role;
|
|
|
|
if ((local_role != BGP_ROLE_UNDEFINED) &&
|
|
(neigh_role != BGP_ROLE_UNDEFINED) &&
|
|
!((local_role == BGP_ROLE_PEER && neigh_role == BGP_ROLE_PEER) ||
|
|
(local_role == BGP_ROLE_CUSTOMER && neigh_role == BGP_ROLE_PROVIDER) ||
|
|
(local_role == BGP_ROLE_PROVIDER && neigh_role == BGP_ROLE_CUSTOMER) ||
|
|
(local_role == BGP_ROLE_RS_CLIENT && neigh_role == BGP_ROLE_RS_SERVER) ||
|
|
(local_role == BGP_ROLE_RS_SERVER && neigh_role == BGP_ROLE_RS_CLIENT)))
|
|
{ bgp_error(conn, 2, 11, &neigh_role, -1); return; }
|
|
|
|
if ((p->cf->require_roles) && (neigh_role == BGP_ROLE_UNDEFINED))
|
|
{ bgp_error(conn, 2, 11, &neigh_role, -1); return; }
|
|
|
|
/* Check the other connection */
|
|
other = (conn == &p->outgoing_conn) ? &p->incoming_conn : &p->outgoing_conn;
|
|
switch (other->state)
|
|
{
|
|
case BS_CONNECT:
|
|
case BS_ACTIVE:
|
|
/* Stop outgoing connection attempts */
|
|
bgp_conn_enter_idle_state(other);
|
|
break;
|
|
|
|
case BS_IDLE:
|
|
case BS_OPENSENT:
|
|
case BS_CLOSE:
|
|
break;
|
|
|
|
case BS_OPENCONFIRM:
|
|
/*
|
|
* Description of collision detection rules in RFC 4271 is confusing and
|
|
* contradictory, but it is essentially:
|
|
*
|
|
* 1. Router with higher ID is dominant
|
|
* 2. If both have the same ID, router with higher ASN is dominant [RFC6286]
|
|
* 3. When both connections are in OpenConfirm state, one initiated by
|
|
* the dominant router is kept.
|
|
*
|
|
* The first line in the expression below evaluates whether the neighbor
|
|
* is dominant, the second line whether the new connection was initiated
|
|
* by the neighbor. If both are true (or both are false), we keep the new
|
|
* connection, otherwise we keep the old one.
|
|
*/
|
|
if (((p->local_id < id) || ((p->local_id == id) && (p->public_as < p->remote_as)))
|
|
== (conn == &p->incoming_conn))
|
|
{
|
|
/* Should close the other connection */
|
|
BGP_TRACE(D_EVENTS, "Connection collision, giving up the other connection");
|
|
bgp_error(other, 6, 7, NULL, 0);
|
|
break;
|
|
}
|
|
/* Fall thru */
|
|
case BS_ESTABLISHED:
|
|
/* Should close this connection */
|
|
BGP_TRACE(D_EVENTS, "Connection collision, giving up this connection");
|
|
bgp_error(conn, 6, 7, NULL, 0);
|
|
return;
|
|
|
|
default:
|
|
bug("bgp_rx_open: Unknown state");
|
|
}
|
|
|
|
/* Update our local variables */
|
|
conn->hold_time = hold_time;
|
|
conn->keepalive_time = keepalive_time;
|
|
conn->send_hold_time = send_hold_time;
|
|
conn->as4_session = conn->local_caps->as4_support && caps->as4_support;
|
|
conn->ext_messages = conn->local_caps->ext_messages && caps->ext_messages;
|
|
p->remote_id = id;
|
|
|
|
DBG("BGP: Hold timer set to %d, keepalive to %d, AS to %d, ID to %x, AS4 session to %d\n",
|
|
conn->hold_time, conn->keepalive_time, p->remote_as, p->remote_id, conn->as4_session);
|
|
|
|
bgp_schedule_packet(conn, NULL, PKT_KEEPALIVE);
|
|
bgp_start_timer(p, conn->hold_timer, conn->hold_time);
|
|
bgp_start_timer(p, conn->send_hold_timer, conn->send_hold_time);
|
|
bgp_conn_enter_openconfirm_state(conn);
|
|
}
|
|
|
|
|
|
/*
|
|
* Next hop handling
|
|
*/
|
|
|
|
#define REPORT(msg, args...) \
|
|
({ log(L_REMOTE "%s: " msg, s->proto->p.name, ## args); })
|
|
|
|
#define DISCARD(msg, args...) \
|
|
({ REPORT(msg, ## args); return; })
|
|
|
|
#define WITHDRAW(msg, args...) \
|
|
({ REPORT(msg, ## args); s->err_withdraw = 1; return; })
|
|
|
|
#define REJECT(msg, args...) \
|
|
({ log(L_ERR "%s: " msg, s->proto->p.name, ## args); s->err_reject = 1; return; })
|
|
|
|
#define BAD_AFI "Unexpected AF <%u/%u> in UPDATE"
|
|
#define BAD_NEXT_HOP "Invalid NEXT_HOP attribute"
|
|
#define NO_NEXT_HOP "Missing NEXT_HOP attribute"
|
|
#define NO_LABEL_STACK "Missing MPLS stack"
|
|
|
|
#define MISMATCHED_AF " - mismatched address family (%I for %s)"
|
|
|
|
static void
|
|
bgp_apply_next_hop(struct bgp_parse_state *s, ea_list **to, ip_addr gw, ip_addr ll)
|
|
{
|
|
struct bgp_proto *p = s->proto;
|
|
struct bgp_channel *c = s->channel;
|
|
|
|
if (c->cf->gw_mode == GW_DIRECT)
|
|
{
|
|
neighbor *nbr = NULL;
|
|
|
|
/* GW_DIRECT -> single_hop -> p->neigh != NULL */
|
|
if (ipa_nonzero2(gw))
|
|
nbr = neigh_find(&p->p, gw, NULL, 0);
|
|
else if (ipa_nonzero(ll))
|
|
nbr = neigh_find(&p->p, ll, p->neigh->iface, 0);
|
|
else
|
|
WITHDRAW(BAD_NEXT_HOP " - zero address");
|
|
|
|
if (!nbr)
|
|
WITHDRAW(BAD_NEXT_HOP " - address %I not directly reachable", ipa_nonzero(gw) ? gw : ll);
|
|
|
|
if (nbr->scope == SCOPE_HOST)
|
|
WITHDRAW(BAD_NEXT_HOP " - address %I is local", nbr->addr);
|
|
|
|
ea_set_attr_u32(to, &ea_gen_igp_metric, 0, c->cf->cost);
|
|
|
|
struct nexthop_adata_mpls nam;
|
|
memset(&nam, 0, sizeof nam);
|
|
nam.nhad.nh.gw = nbr->addr;
|
|
nam.nhad.nh.iface = nbr->iface;
|
|
nam.nhad.ad.length = NEXTHOP_NEXT(&nam.nhad.nh) - (void *) nam.nhad.ad.data;
|
|
ea_set_attr_data(to, &ea_gen_nexthop, 0, nam.nhad.ad.data, nam.nhad.ad.length);
|
|
}
|
|
else /* GW_RECURSIVE */
|
|
{
|
|
if (ipa_zero2(gw))
|
|
WITHDRAW(BAD_NEXT_HOP " - zero address");
|
|
|
|
rtable *tab = ipa_is_ip4(gw) ? c->igp_table_ip4 : c->igp_table_ip6;
|
|
ip_addr lla = (c->cf->next_hop_prefer == NHP_LOCAL) ? ll : IPA_NONE;
|
|
|
|
if (s->mpls)
|
|
{
|
|
u32 labels[BGP_MPLS_MAX];
|
|
ea_set_hostentry(to, c->c.table, tab, gw, lla, BGP_MPLS_MAX, labels);
|
|
}
|
|
else
|
|
ea_set_hostentry(to, c->c.table, tab, gw, lla, 0, NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
bgp_apply_mpls_labels(struct bgp_parse_state *s, ea_list **to, u32 lnum, u32 labels[lnum])
|
|
{
|
|
bgp_set_attr_data(to, BA_MPLS_LABEL_STACK, 0, labels, lnum * sizeof labels[0]);
|
|
|
|
if (lnum > MPLS_MAX_LABEL_STACK)
|
|
{
|
|
REPORT("Too many MPLS labels ($u)", lnum);
|
|
|
|
ea_set_dest(to, 0, RTD_UNREACHABLE);
|
|
return;
|
|
}
|
|
|
|
/* Handle implicit NULL as empty MPLS stack */
|
|
if ((lnum == 1) && (labels[0] == BGP_MPLS_NULL))
|
|
lnum = 0;
|
|
|
|
if (s->channel->cf->gw_mode == GW_DIRECT)
|
|
{
|
|
eattr *e = ea_find(*to, &ea_gen_nexthop);
|
|
SKIP_BACK_DECLARE(struct nexthop_adata_mpls, namp, nhad.ad, e->u.ptr);
|
|
|
|
namp->nhad.nh.labels = lnum;
|
|
memcpy(namp->nhad.nh.label, labels, lnum * sizeof(u32));
|
|
namp->nhad.ad.length = NEXTHOP_NEXT(&namp->nhad.nh) - (void *) namp->nhad.ad.data;
|
|
}
|
|
else /* GW_RECURSIVE */
|
|
{
|
|
eattr *e = ea_find(*to, &ea_gen_hostentry);
|
|
ASSERT_DIE(e);
|
|
struct hostentry_adata *head = (void *) e->u.ptr;
|
|
memcpy(head->labels, labels, lnum * sizeof(u32));
|
|
head->ad.length = (void *)(&head->labels[lnum]) - (void *) head->ad.data;
|
|
debug("he %p apply mpls labels (%u): %u\n", head->he, lnum, labels[0]);
|
|
}
|
|
}
|
|
|
|
static int
|
|
bgp_match_src(struct bgp_export_state *s, int mode)
|
|
{
|
|
switch (mode)
|
|
{
|
|
case NH_NO: return 0;
|
|
case NH_ALL: return 1;
|
|
case NH_IBGP: return s->src && s->src->is_internal;
|
|
case NH_EBGP: return s->src && !s->src->is_internal;
|
|
default: return 0;
|
|
}
|
|
}
|
|
|
|
static inline int
|
|
bgp_use_next_hop(struct bgp_export_state *s, eattr *a)
|
|
{
|
|
struct bgp_proto *p = s->proto;
|
|
struct bgp_channel *c = s->channel;
|
|
ip_addr *nh = (void *) a->u.ptr->data;
|
|
|
|
/* Handle next hop self option */
|
|
if (c->cf->next_hop_self && bgp_match_src(s, c->cf->next_hop_self))
|
|
return 0;
|
|
|
|
/* Handle next hop keep option */
|
|
if (c->cf->next_hop_keep && bgp_match_src(s, c->cf->next_hop_keep))
|
|
return 1;
|
|
|
|
/* Keep it when explicitly set in export filter */
|
|
if (a->fresh)
|
|
return 1;
|
|
|
|
/* Check for non-matching AF */
|
|
if ((ipa_is_ip4(*nh) != bgp_channel_is_ipv4(c)) && !c->ext_next_hop)
|
|
return 0;
|
|
|
|
/* Do not pass NEXT_HOP between different VRFs */
|
|
if (p->p.vrf && s->src && s->src->p.vrf && (p->p.vrf != s->src->p.vrf))
|
|
return 0;
|
|
|
|
/* Keep it when exported to internal peers */
|
|
if (p->is_interior && ipa_nonzero(*nh))
|
|
return 1;
|
|
|
|
/* Keep it when forwarded between single-hop BGPs on the same iface */
|
|
struct iface *ifa = (s->src && s->src->neigh && (s->src->p.proto_state != PS_DOWN)) ?
|
|
s->src->neigh->iface : NULL;
|
|
return p->neigh && (p->neigh->iface == ifa);
|
|
}
|
|
|
|
static inline struct nexthop *
|
|
bgp_use_gateway(struct bgp_export_state *s)
|
|
{
|
|
struct bgp_proto *p = s->proto;
|
|
struct bgp_channel *c = s->channel;
|
|
ea_list *ra = s->route->attrs;
|
|
|
|
/* Handle next hop self option - also applies to gateway */
|
|
if (c->cf->next_hop_self && bgp_match_src(s, c->cf->next_hop_self))
|
|
return NULL;
|
|
|
|
eattr *nhea = ea_find(ra, &ea_gen_nexthop);
|
|
if (!nhea)
|
|
return NULL;
|
|
|
|
/* We need one valid global gateway */
|
|
struct nexthop_adata *nhad = (struct nexthop_adata *) nhea->u.ptr;
|
|
if (!NEXTHOP_IS_REACHABLE(nhad) ||
|
|
!NEXTHOP_ONE(nhad) || ipa_zero(nhad->nh.gw) ||
|
|
ipa_is_link_local(nhad->nh.gw))
|
|
return NULL;
|
|
|
|
/* Check for non-matching AF */
|
|
if ((ipa_is_ip4(nhad->nh.gw) != bgp_channel_is_ipv4(c)) && !c->ext_next_hop)
|
|
return NULL;
|
|
|
|
/* Do not use gateway from different VRF */
|
|
if (p->p.vrf && nhad->nh.iface && !if_in_vrf(nhad->nh.iface, p->p.vrf))
|
|
return 0;
|
|
|
|
/* Use it when exported to internal peers */
|
|
if (p->is_interior)
|
|
return &nhad->nh;
|
|
|
|
/* Use it when forwarded to single-hop BGP peer on on the same iface */
|
|
if (p->neigh && (p->neigh->iface == nhad->nh.iface))
|
|
return &nhad->nh;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
bgp_update_next_hop_ip(struct bgp_export_state *s, eattr *a, ea_list **to)
|
|
{
|
|
if (!a || !bgp_use_next_hop(s, a))
|
|
{
|
|
struct nexthop *nhloc;
|
|
if (nhloc = bgp_use_gateway(s))
|
|
{
|
|
ip_addr nh[1] = { nhloc->gw };
|
|
bgp_set_attr_data(to, BA_NEXT_HOP, 0, nh, 16);
|
|
|
|
if (s->mpls)
|
|
{
|
|
u32 implicit_null = BGP_MPLS_NULL;
|
|
u32 *labels = nhloc->labels ? nhloc->label : &implicit_null;
|
|
uint lnum = nhloc->labels ? nhloc->labels : 1;
|
|
bgp_set_attr_data(to, BA_MPLS_LABEL_STACK, 0, labels, lnum * 4);
|
|
}
|
|
else
|
|
bgp_unset_attr(to, BA_MPLS_LABEL_STACK);
|
|
}
|
|
else
|
|
{
|
|
ip_addr nh[2] = { s->channel->next_hop_addr, s->channel->link_addr };
|
|
bgp_set_attr_data(to, BA_NEXT_HOP, 0, nh, ipa_nonzero(nh[1]) ? 32 : 16);
|
|
s->local_next_hop = 1;
|
|
|
|
if (s->mpls)
|
|
{
|
|
u32 label = ea_get_int(s->route->attrs, &ea_gen_mpls_label, BGP_MPLS_NULL);
|
|
bgp_set_attr_data(to, BA_MPLS_LABEL_STACK, 0, &label, 4);
|
|
}
|
|
else
|
|
bgp_unset_attr(to, BA_MPLS_LABEL_STACK);
|
|
}
|
|
}
|
|
|
|
/* Check if next hop is valid */
|
|
a = bgp_find_attr(*to, BA_NEXT_HOP);
|
|
if (!a)
|
|
REJECT(NO_NEXT_HOP);
|
|
|
|
ip_addr *nh = (void *) a->u.ptr->data;
|
|
ip_addr peer = s->proto->remote_ip;
|
|
uint len = a->u.ptr->length;
|
|
|
|
/* Forbid zero next hop */
|
|
if (ipa_zero2(nh[0]) && ((len != 32) || ipa_zero(nh[1])))
|
|
REJECT(BAD_NEXT_HOP " - zero address");
|
|
|
|
/* Forbid next hop equal to neighbor IP */
|
|
if (ipa_equal(peer, nh[0]) || ((len == 32) && ipa_equal(peer, nh[1])))
|
|
REJECT(BAD_NEXT_HOP " - neighbor address %I", peer);
|
|
|
|
/* Forbid next hop with non-matching AF */
|
|
if ((ipa_is_ip4(nh[0]) != bgp_channel_is_ipv4(s->channel)) &&
|
|
!s->channel->ext_next_hop)
|
|
REJECT(BAD_NEXT_HOP MISMATCHED_AF, nh[0], s->channel->desc->name);
|
|
|
|
/* Just check if MPLS stack */
|
|
if (s->mpls && !bgp_find_attr(*to, BA_MPLS_LABEL_STACK))
|
|
REJECT(NO_LABEL_STACK);
|
|
}
|
|
|
|
static uint
|
|
bgp_encode_next_hop_ip(struct bgp_write_state *s, eattr *a, byte *buf, uint size UNUSED)
|
|
{
|
|
/* This function is used only for MP-BGP, see bgp_encode_next_hop() for IPv4 BGP */
|
|
ip_addr *nh = (void *) a->u.ptr->data;
|
|
uint len = a->u.ptr->length;
|
|
|
|
ASSERT((len == 16) || (len == 32));
|
|
|
|
/*
|
|
* Both IPv4 and IPv6 next hops can be used (with ext_next_hop enabled). This
|
|
* is specified in RFC 8950 for IPv4 and in RFC 4798 for IPv6. The difference
|
|
* is that IPv4 address is directly encoded with IPv4 NLRI, but as IPv4-mapped
|
|
* IPv6 address with IPv6 NLRI.
|
|
*/
|
|
|
|
if (bgp_channel_is_ipv4(s->ptx->c) && ipa_is_ip4(nh[0]))
|
|
{
|
|
put_ip4(buf, ipa_to_ip4(nh[0]));
|
|
return 4;
|
|
}
|
|
|
|
put_ip6(buf, ipa_to_ip6(nh[0]));
|
|
|
|
if (len == 32)
|
|
put_ip6(buf+16, ipa_to_ip6(nh[1]));
|
|
|
|
return len;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_next_hop_ip(struct bgp_parse_state *s, byte *data, uint len, ea_list **to)
|
|
{
|
|
struct bgp_channel *c = s->channel;
|
|
struct adata *ad = lp_alloc_adata(s->pool, 32);
|
|
ip_addr *nh = (void *) ad->data;
|
|
|
|
if (len == 4)
|
|
{
|
|
nh[0] = ipa_from_ip4(get_ip4(data));
|
|
nh[1] = IPA_NONE;
|
|
}
|
|
else if (len == 16)
|
|
{
|
|
nh[0] = ipa_from_ip6(get_ip6(data));
|
|
nh[1] = IPA_NONE;
|
|
|
|
if (ipa_is_link_local(nh[0]))
|
|
{ nh[1] = nh[0]; nh[0] = IPA_NONE; }
|
|
}
|
|
else if (len == 32)
|
|
{
|
|
nh[0] = ipa_from_ip6(get_ip6(data));
|
|
nh[1] = ipa_from_ip6(get_ip6(data+16));
|
|
|
|
if (ipa_is_link_local(nh[0]))
|
|
{ nh[1] = nh[0]; nh[0] = IPA_NONE; }
|
|
|
|
if (ipa_is_ip4(nh[0]) || !ipa_is_link_local(nh[1]))
|
|
nh[1] = IPA_NONE;
|
|
}
|
|
else
|
|
bgp_parse_error(s, 9);
|
|
|
|
if (ipa_zero(nh[1]))
|
|
ad->length = 16;
|
|
|
|
if ((bgp_channel_is_ipv4(c) != ipa_is_ip4(nh[0])) && !c->ext_next_hop)
|
|
WITHDRAW(BAD_NEXT_HOP MISMATCHED_AF, nh[0], c->desc->name);
|
|
|
|
// XXXX validate next hop
|
|
|
|
bgp_set_attr_ptr(to, BA_NEXT_HOP, 0, ad);
|
|
bgp_apply_next_hop(s, to, nh[0], nh[1]);
|
|
}
|
|
|
|
static uint
|
|
bgp_encode_next_hop_vpn(struct bgp_write_state *s, eattr *a, byte *buf, uint size UNUSED)
|
|
{
|
|
ip_addr *nh = (void *) a->u.ptr->data;
|
|
uint len = a->u.ptr->length;
|
|
|
|
ASSERT((len == 16) || (len == 32));
|
|
|
|
/*
|
|
* Both IPv4 and IPv6 next hops can be used (with ext_next_hop enabled). This
|
|
* is specified in RFC 8950 for VPNv4 and in RFC 4659 for VPNv6. The difference
|
|
* is that IPv4 address is directly encoded with VPNv4 NLRI, but as IPv4-mapped
|
|
* IPv6 address with VPNv6 NLRI.
|
|
*/
|
|
|
|
if (bgp_channel_is_ipv4(s->ptx->c) && ipa_is_ip4(nh[0]))
|
|
{
|
|
put_u64(buf, 0); /* VPN RD is 0 */
|
|
put_ip4(buf+8, ipa_to_ip4(nh[0]));
|
|
return 12;
|
|
}
|
|
|
|
put_u64(buf, 0); /* VPN RD is 0 */
|
|
put_ip6(buf+8, ipa_to_ip6(nh[0]));
|
|
|
|
if (len == 16)
|
|
return 24;
|
|
|
|
put_u64(buf+24, 0); /* VPN RD is 0 */
|
|
put_ip6(buf+32, ipa_to_ip6(nh[1]));
|
|
|
|
return 48;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_next_hop_vpn(struct bgp_parse_state *s, byte *data, uint len, ea_list **to)
|
|
{
|
|
struct bgp_channel *c = s->channel;
|
|
struct adata *ad = lp_alloc_adata(s->pool, 32);
|
|
ip_addr *nh = (void *) ad->data;
|
|
|
|
if (len == 12)
|
|
{
|
|
nh[0] = ipa_from_ip4(get_ip4(data+8));
|
|
nh[1] = IPA_NONE;
|
|
}
|
|
else if (len == 24)
|
|
{
|
|
nh[0] = ipa_from_ip6(get_ip6(data+8));
|
|
nh[1] = IPA_NONE;
|
|
|
|
if (ipa_is_link_local(nh[0]))
|
|
{ nh[1] = nh[0]; nh[0] = IPA_NONE; }
|
|
}
|
|
else if (len == 48)
|
|
{
|
|
nh[0] = ipa_from_ip6(get_ip6(data+8));
|
|
nh[1] = ipa_from_ip6(get_ip6(data+32));
|
|
|
|
if (ipa_is_ip4(nh[0]) || !ip6_is_link_local(nh[1]))
|
|
nh[1] = IPA_NONE;
|
|
}
|
|
else
|
|
bgp_parse_error(s, 9);
|
|
|
|
if (ipa_zero(nh[1]))
|
|
ad->length = 16;
|
|
|
|
/* XXXX which error */
|
|
if ((get_u64(data) != 0) || ((len == 48) && (get_u64(data+24) != 0)))
|
|
bgp_parse_error(s, 9);
|
|
|
|
if ((bgp_channel_is_ipv4(c) != ipa_is_ip4(nh[0])) && !c->ext_next_hop)
|
|
WITHDRAW(BAD_NEXT_HOP MISMATCHED_AF, nh[0], c->desc->name);
|
|
|
|
// XXXX validate next hop
|
|
|
|
bgp_set_attr_ptr(to, BA_NEXT_HOP, 0, ad);
|
|
bgp_apply_next_hop(s, to, nh[0], nh[1]);
|
|
}
|
|
|
|
|
|
|
|
static uint
|
|
bgp_encode_next_hop_none(struct bgp_write_state *s UNUSED, eattr *a UNUSED, byte *buf UNUSED, uint size UNUSED)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_next_hop_none(struct bgp_parse_state *s UNUSED, byte *data UNUSED, uint len UNUSED, ea_list **to UNUSED)
|
|
{
|
|
/*
|
|
* Although we expect no next hop and RFC 7606 7.11 states that attribute
|
|
* MP_REACH_NLRI with unexpected next hop length is considered malformed,
|
|
* FlowSpec RFC 5575 4 states that next hop shall be ignored on receipt.
|
|
*/
|
|
|
|
return;
|
|
}
|
|
|
|
static void
|
|
bgp_update_next_hop_none(struct bgp_export_state *s UNUSED, eattr *a, ea_list **to)
|
|
{
|
|
/* NEXT_HOP shall not pass */
|
|
if (a)
|
|
bgp_unset_attr(to, BA_NEXT_HOP);
|
|
}
|
|
|
|
|
|
/*
|
|
* UPDATE
|
|
*/
|
|
|
|
static void
|
|
bgp_rte_update(struct bgp_parse_state *s, const net_addr *n, u32 path_id, ea_list *a0)
|
|
{
|
|
if (path_id != s->last_id)
|
|
{
|
|
rt_unlock_source(s->last_src);
|
|
|
|
s->last_src = rt_get_source(&s->proto->p, path_id);
|
|
s->last_id = path_id;
|
|
}
|
|
|
|
if (!a0)
|
|
{
|
|
/* Route update was changed to withdraw */
|
|
if (s->err_withdraw && s->reach_nlri_step)
|
|
REPORT("Invalid route %N withdrawn", n);
|
|
|
|
/* Route withdraw */
|
|
rte_update(&s->channel->c, n, NULL, s->last_src);
|
|
return;
|
|
}
|
|
|
|
rte e0 = {
|
|
.attrs = a0,
|
|
.src = s->last_src,
|
|
};
|
|
|
|
rte_update(&s->channel->c, n, &e0, s->last_src);
|
|
}
|
|
|
|
static void
|
|
bgp_encode_mpls_labels(struct bgp_write_state *s UNUSED, const adata *mpls, byte **pos, uint *size, byte *pxlen)
|
|
{
|
|
const u32 dummy = 0;
|
|
const u32 *labels = mpls ? (const u32 *) mpls->data : &dummy;
|
|
uint lnum = mpls ? (mpls->length / 4) : 1;
|
|
|
|
for (uint i = 0; i < lnum; i++)
|
|
{
|
|
put_u24(*pos, labels[i] << 4);
|
|
ADVANCE(*pos, *size, 3);
|
|
}
|
|
|
|
/* Add bottom-of-stack flag */
|
|
(*pos)[-1] |= BGP_MPLS_BOS;
|
|
|
|
*pxlen += 24 * lnum;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_mpls_labels(struct bgp_parse_state *s, byte **pos, uint *len, uint *pxlen, ea_list **to)
|
|
{
|
|
u32 labels[BGP_MPLS_MAX];
|
|
u32 label;
|
|
uint lnum = 0;
|
|
|
|
do {
|
|
if (*pxlen < 24)
|
|
bgp_parse_error(s, 1);
|
|
|
|
label = get_u24(*pos);
|
|
labels[lnum++] = label >> 4;
|
|
ADVANCE(*pos, *len, 3);
|
|
*pxlen -= 24;
|
|
|
|
/* RFC 8277 2.4 - withdraw does not have variable-size MPLS stack but
|
|
fixed-size 24-bit Compatibility field, which MUST be ignored */
|
|
if (!s->reach_nlri_step)
|
|
return;
|
|
}
|
|
while (!(label & BGP_MPLS_BOS));
|
|
|
|
if (!*to)
|
|
return;
|
|
|
|
/* Update next hop entry in rta */
|
|
bgp_apply_mpls_labels(s, to, lnum, labels);
|
|
|
|
return;
|
|
}
|
|
|
|
static uint
|
|
bgp_encode_nlri_ip4(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, uint size)
|
|
{
|
|
byte *pos = buf;
|
|
|
|
while (!EMPTY_LIST(buck->prefixes) && (size >= BGP_NLRI_MAX))
|
|
{
|
|
struct bgp_prefix *px = HEAD(buck->prefixes);
|
|
struct net_addr_ip4 *net = NET_PTR_IP4(&px->ni->addr[0]);
|
|
|
|
/* Encode path ID */
|
|
if (s->add_path)
|
|
{
|
|
put_u32(pos, px->src->global_id);
|
|
ADVANCE(pos, size, 4);
|
|
}
|
|
|
|
/* Encode prefix length */
|
|
*pos = net->pxlen;
|
|
ADVANCE(pos, size, 1);
|
|
|
|
/* Encode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_encode_mpls_labels(s, s->mpls_labels, &pos, &size, pos - 1);
|
|
|
|
/* Encode prefix body */
|
|
ip4_addr a = ip4_hton(net->prefix);
|
|
uint b = (net->pxlen + 7) / 8;
|
|
memcpy(pos, &a, b);
|
|
ADVANCE(pos, size, b);
|
|
|
|
bgp_done_prefix(s->ptx, px, buck);
|
|
}
|
|
|
|
return pos - buf;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_nlri_ip4(struct bgp_parse_state *s, byte *pos, uint len, ea_list *a)
|
|
{
|
|
while (len)
|
|
{
|
|
ea_list *ea = a;
|
|
net_addr_ip4 net;
|
|
u32 path_id = 0;
|
|
|
|
/* Decode path ID */
|
|
if (s->add_path)
|
|
{
|
|
if (len < 5)
|
|
bgp_parse_error(s, 1);
|
|
|
|
path_id = get_u32(pos);
|
|
ADVANCE(pos, len, 4);
|
|
}
|
|
|
|
/* Decode prefix length */
|
|
uint l = *pos;
|
|
ADVANCE(pos, len, 1);
|
|
|
|
if (len < ((l + 7) / 8))
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Decode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_decode_mpls_labels(s, &pos, &len, &l, &ea);
|
|
|
|
if (l > IP4_MAX_PREFIX_LENGTH)
|
|
bgp_parse_error(s, 10);
|
|
|
|
/* Decode prefix body */
|
|
ip4_addr addr = IP4_NONE;
|
|
uint b = (l + 7) / 8;
|
|
memcpy(&addr, pos, b);
|
|
ADVANCE(pos, len, b);
|
|
|
|
net = NET_ADDR_IP4(ip4_ntoh(addr), l);
|
|
net_normalize_ip4(&net);
|
|
|
|
// XXXX validate prefix
|
|
|
|
bgp_rte_update(s, (net_addr *) &net, path_id, ea);
|
|
}
|
|
}
|
|
|
|
|
|
static uint
|
|
bgp_encode_nlri_ip6(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, uint size)
|
|
{
|
|
byte *pos = buf;
|
|
|
|
while (!EMPTY_LIST(buck->prefixes) && (size >= BGP_NLRI_MAX))
|
|
{
|
|
struct bgp_prefix *px = HEAD(buck->prefixes);
|
|
struct net_addr_ip6 *net = NET_PTR_IP6(&px->ni->addr[0]);
|
|
|
|
/* Encode path ID */
|
|
if (s->add_path)
|
|
{
|
|
put_u32(pos, px->src->global_id);
|
|
ADVANCE(pos, size, 4);
|
|
}
|
|
|
|
/* Encode prefix length */
|
|
*pos = net->pxlen;
|
|
ADVANCE(pos, size, 1);
|
|
|
|
/* Encode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_encode_mpls_labels(s, s->mpls_labels, &pos, &size, pos - 1);
|
|
|
|
/* Encode prefix body */
|
|
ip6_addr a = ip6_hton(net->prefix);
|
|
uint b = (net->pxlen + 7) / 8;
|
|
memcpy(pos, &a, b);
|
|
ADVANCE(pos, size, b);
|
|
|
|
bgp_done_prefix(s->ptx, px, buck);
|
|
}
|
|
|
|
return pos - buf;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_nlri_ip6(struct bgp_parse_state *s, byte *pos, uint len, ea_list *a)
|
|
{
|
|
while (len)
|
|
{
|
|
ea_list *ea = a;
|
|
net_addr_ip6 net;
|
|
u32 path_id = 0;
|
|
|
|
/* Decode path ID */
|
|
if (s->add_path)
|
|
{
|
|
if (len < 5)
|
|
bgp_parse_error(s, 1);
|
|
|
|
path_id = get_u32(pos);
|
|
ADVANCE(pos, len, 4);
|
|
}
|
|
|
|
/* Decode prefix length */
|
|
uint l = *pos;
|
|
ADVANCE(pos, len, 1);
|
|
|
|
if (len < ((l + 7) / 8))
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Decode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_decode_mpls_labels(s, &pos, &len, &l, &ea);
|
|
|
|
if (l > IP6_MAX_PREFIX_LENGTH)
|
|
bgp_parse_error(s, 10);
|
|
|
|
/* Decode prefix body */
|
|
ip6_addr addr = IP6_NONE;
|
|
uint b = (l + 7) / 8;
|
|
memcpy(&addr, pos, b);
|
|
ADVANCE(pos, len, b);
|
|
|
|
net = NET_ADDR_IP6(ip6_ntoh(addr), l);
|
|
net_normalize_ip6(&net);
|
|
|
|
// XXXX validate prefix
|
|
|
|
bgp_rte_update(s, (net_addr *) &net, path_id, ea);
|
|
}
|
|
}
|
|
|
|
static uint
|
|
bgp_encode_nlri_vpn4(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, uint size)
|
|
{
|
|
byte *pos = buf;
|
|
|
|
while (!EMPTY_LIST(buck->prefixes) && (size >= BGP_NLRI_MAX))
|
|
{
|
|
struct bgp_prefix *px = HEAD(buck->prefixes);
|
|
struct net_addr_vpn4 *net = NET_PTR_VPN4(&px->ni->addr[0]);
|
|
|
|
/* Encode path ID */
|
|
if (s->add_path)
|
|
{
|
|
put_u32(pos, px->src->global_id);
|
|
ADVANCE(pos, size, 4);
|
|
}
|
|
|
|
/* Encode prefix length */
|
|
*pos = 64 + net->pxlen;
|
|
ADVANCE(pos, size, 1);
|
|
|
|
/* Encode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_encode_mpls_labels(s, s->mpls_labels, &pos, &size, pos - 1);
|
|
|
|
/* Encode route distinguisher */
|
|
put_u64(pos, net->rd);
|
|
ADVANCE(pos, size, 8);
|
|
|
|
/* Encode prefix body */
|
|
ip4_addr a = ip4_hton(net->prefix);
|
|
uint b = (net->pxlen + 7) / 8;
|
|
memcpy(pos, &a, b);
|
|
ADVANCE(pos, size, b);
|
|
|
|
bgp_done_prefix(s->ptx, px, buck);
|
|
}
|
|
|
|
return pos - buf;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_nlri_vpn4(struct bgp_parse_state *s, byte *pos, uint len, ea_list *a)
|
|
{
|
|
while (len)
|
|
{
|
|
ea_list *ea = a;
|
|
net_addr_vpn4 net;
|
|
u32 path_id = 0;
|
|
|
|
/* Decode path ID */
|
|
if (s->add_path)
|
|
{
|
|
if (len < 5)
|
|
bgp_parse_error(s, 1);
|
|
|
|
path_id = get_u32(pos);
|
|
ADVANCE(pos, len, 4);
|
|
}
|
|
|
|
/* Decode prefix length */
|
|
uint l = *pos;
|
|
ADVANCE(pos, len, 1);
|
|
|
|
if (len < ((l + 7) / 8))
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Decode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_decode_mpls_labels(s, &pos, &len, &l, &ea);
|
|
|
|
/* Decode route distinguisher */
|
|
if (l < 64)
|
|
bgp_parse_error(s, 1);
|
|
|
|
u64 rd = get_u64(pos);
|
|
ADVANCE(pos, len, 8);
|
|
l -= 64;
|
|
|
|
if (l > IP4_MAX_PREFIX_LENGTH)
|
|
bgp_parse_error(s, 10);
|
|
|
|
/* Decode prefix body */
|
|
ip4_addr addr = IP4_NONE;
|
|
uint b = (l + 7) / 8;
|
|
memcpy(&addr, pos, b);
|
|
ADVANCE(pos, len, b);
|
|
|
|
net = NET_ADDR_VPN4(ip4_ntoh(addr), l, rd);
|
|
net_normalize_vpn4(&net);
|
|
|
|
// XXXX validate prefix
|
|
|
|
bgp_rte_update(s, (net_addr *) &net, path_id, ea);
|
|
}
|
|
}
|
|
|
|
|
|
static uint
|
|
bgp_encode_nlri_vpn6(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, uint size)
|
|
{
|
|
byte *pos = buf;
|
|
|
|
while (!EMPTY_LIST(buck->prefixes) && (size >= BGP_NLRI_MAX))
|
|
{
|
|
struct bgp_prefix *px = HEAD(buck->prefixes);
|
|
struct net_addr_vpn6 *net = NET_PTR_VPN6(&px->ni->addr[0]);
|
|
|
|
/* Encode path ID */
|
|
if (s->add_path)
|
|
{
|
|
put_u32(pos, px->src->global_id);
|
|
ADVANCE(pos, size, 4);
|
|
}
|
|
|
|
/* Encode prefix length */
|
|
*pos = 64 + net->pxlen;
|
|
ADVANCE(pos, size, 1);
|
|
|
|
/* Encode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_encode_mpls_labels(s, s->mpls_labels, &pos, &size, pos - 1);
|
|
|
|
/* Encode route distinguisher */
|
|
put_u64(pos, net->rd);
|
|
ADVANCE(pos, size, 8);
|
|
|
|
/* Encode prefix body */
|
|
ip6_addr a = ip6_hton(net->prefix);
|
|
uint b = (net->pxlen + 7) / 8;
|
|
memcpy(pos, &a, b);
|
|
ADVANCE(pos, size, b);
|
|
|
|
bgp_done_prefix(s->ptx, px, buck);
|
|
}
|
|
|
|
return pos - buf;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_nlri_vpn6(struct bgp_parse_state *s, byte *pos, uint len, ea_list *a)
|
|
{
|
|
while (len)
|
|
{
|
|
ea_list *ea = a;
|
|
net_addr_vpn6 net;
|
|
u32 path_id = 0;
|
|
|
|
/* Decode path ID */
|
|
if (s->add_path)
|
|
{
|
|
if (len < 5)
|
|
bgp_parse_error(s, 1);
|
|
|
|
path_id = get_u32(pos);
|
|
ADVANCE(pos, len, 4);
|
|
}
|
|
|
|
/* Decode prefix length */
|
|
uint l = *pos;
|
|
ADVANCE(pos, len, 1);
|
|
|
|
if (len < ((l + 7) / 8))
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Decode MPLS labels */
|
|
if (s->mpls)
|
|
bgp_decode_mpls_labels(s, &pos, &len, &l, &ea);
|
|
|
|
/* Decode route distinguisher */
|
|
if (l < 64)
|
|
bgp_parse_error(s, 1);
|
|
|
|
u64 rd = get_u64(pos);
|
|
ADVANCE(pos, len, 8);
|
|
l -= 64;
|
|
|
|
if (l > IP6_MAX_PREFIX_LENGTH)
|
|
bgp_parse_error(s, 10);
|
|
|
|
/* Decode prefix body */
|
|
ip6_addr addr = IP6_NONE;
|
|
uint b = (l + 7) / 8;
|
|
memcpy(&addr, pos, b);
|
|
ADVANCE(pos, len, b);
|
|
|
|
net = NET_ADDR_VPN6(ip6_ntoh(addr), l, rd);
|
|
net_normalize_vpn6(&net);
|
|
|
|
// XXXX validate prefix
|
|
|
|
bgp_rte_update(s, (net_addr *) &net, path_id, ea);
|
|
}
|
|
}
|
|
|
|
|
|
static uint
|
|
bgp_encode_nlri_flow4(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, uint size)
|
|
{
|
|
byte *pos = buf;
|
|
|
|
while (!EMPTY_LIST(buck->prefixes) && (size >= 4))
|
|
{
|
|
struct bgp_prefix *px = HEAD(buck->prefixes);
|
|
struct net_addr_flow4 *net = NET_PTR_FLOW4(&px->ni->addr[0]);
|
|
uint flen = net->length - sizeof(net_addr_flow4);
|
|
|
|
/* Encode path ID */
|
|
if (s->add_path)
|
|
{
|
|
put_u32(pos, px->src->global_id);
|
|
ADVANCE(pos, size, 4);
|
|
}
|
|
|
|
if (flen > size)
|
|
break;
|
|
|
|
/* Copy whole flow data including length */
|
|
memcpy(pos, net->data, flen);
|
|
ADVANCE(pos, size, flen);
|
|
|
|
bgp_done_prefix(s->ptx, px, buck);
|
|
}
|
|
|
|
return pos - buf;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_nlri_flow4(struct bgp_parse_state *s, byte *pos, uint len, ea_list *a)
|
|
{
|
|
while (len)
|
|
{
|
|
u32 path_id = 0;
|
|
|
|
/* Decode path ID */
|
|
if (s->add_path)
|
|
{
|
|
if (len < 4)
|
|
bgp_parse_error(s, 1);
|
|
|
|
path_id = get_u32(pos);
|
|
ADVANCE(pos, len, 4);
|
|
}
|
|
|
|
if (len < 2)
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Decode flow length */
|
|
uint hlen = flow_hdr_length(pos);
|
|
uint dlen = flow_read_length(pos);
|
|
uint flen = hlen + dlen;
|
|
byte *data = pos + hlen;
|
|
|
|
if (len < flen)
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Validate flow data */
|
|
enum flow_validated_state r = flow4_validate(data, dlen);
|
|
if (r != FLOW_ST_VALID)
|
|
{
|
|
log(L_REMOTE "%s: Invalid flow route: %s", s->proto->p.name, flow_validated_state_str(r));
|
|
bgp_parse_error(s, 1);
|
|
}
|
|
|
|
ip4_addr px = IP4_NONE;
|
|
uint pxlen = 0;
|
|
|
|
/* Decode dst prefix */
|
|
if (data[0] == FLOW_TYPE_DST_PREFIX)
|
|
{
|
|
px = flow_read_ip4_part(data);
|
|
pxlen = flow_read_pxlen(data);
|
|
}
|
|
|
|
/* Prepare the flow */
|
|
net_addr *n = alloca(sizeof(struct net_addr_flow4) + flen);
|
|
net_fill_flow4(n, px, pxlen, pos, flen);
|
|
ADVANCE(pos, len, flen);
|
|
|
|
bgp_rte_update(s, n, path_id, a);
|
|
}
|
|
}
|
|
|
|
|
|
static uint
|
|
bgp_encode_nlri_flow6(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, uint size)
|
|
{
|
|
byte *pos = buf;
|
|
|
|
while (!EMPTY_LIST(buck->prefixes) && (size >= 4))
|
|
{
|
|
struct bgp_prefix *px = HEAD(buck->prefixes);
|
|
struct net_addr_flow6 *net = NET_PTR_FLOW6(&px->ni->addr[0]);
|
|
uint flen = net->length - sizeof(net_addr_flow6);
|
|
|
|
/* Encode path ID */
|
|
if (s->add_path)
|
|
{
|
|
put_u32(pos, px->src->global_id);
|
|
ADVANCE(pos, size, 4);
|
|
}
|
|
|
|
if (flen > size)
|
|
break;
|
|
|
|
/* Copy whole flow data including length */
|
|
memcpy(pos, net->data, flen);
|
|
ADVANCE(pos, size, flen);
|
|
|
|
bgp_done_prefix(s->ptx, px, buck);
|
|
}
|
|
|
|
return pos - buf;
|
|
}
|
|
|
|
static void
|
|
bgp_decode_nlri_flow6(struct bgp_parse_state *s, byte *pos, uint len, ea_list *a)
|
|
{
|
|
while (len)
|
|
{
|
|
u32 path_id = 0;
|
|
|
|
/* Decode path ID */
|
|
if (s->add_path)
|
|
{
|
|
if (len < 4)
|
|
bgp_parse_error(s, 1);
|
|
|
|
path_id = get_u32(pos);
|
|
ADVANCE(pos, len, 4);
|
|
}
|
|
|
|
if (len < 2)
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Decode flow length */
|
|
uint hlen = flow_hdr_length(pos);
|
|
uint dlen = flow_read_length(pos);
|
|
uint flen = hlen + dlen;
|
|
byte *data = pos + hlen;
|
|
|
|
if (len < flen)
|
|
bgp_parse_error(s, 1);
|
|
|
|
/* Validate flow data */
|
|
enum flow_validated_state r = flow6_validate(data, dlen);
|
|
if (r != FLOW_ST_VALID)
|
|
{
|
|
log(L_REMOTE "%s: Invalid flow route: %s", s->proto->p.name, flow_validated_state_str(r));
|
|
bgp_parse_error(s, 1);
|
|
}
|
|
|
|
ip6_addr px = IP6_NONE;
|
|
uint pxlen = 0;
|
|
|
|
/* Decode dst prefix */
|
|
if (data[0] == FLOW_TYPE_DST_PREFIX)
|
|
{
|
|
px = flow_read_ip6_part(data);
|
|
pxlen = flow_read_pxlen(data);
|
|
}
|
|
|
|
/* Prepare the flow */
|
|
net_addr *n = alloca(sizeof(struct net_addr_flow6) + flen);
|
|
net_fill_flow6(n, px, pxlen, pos, flen);
|
|
ADVANCE(pos, len, flen);
|
|
|
|
bgp_rte_update(s, n, path_id, a);
|
|
}
|
|
}
|
|
|
|
|
|
static const struct bgp_af_desc bgp_af_table[] = {
|
|
{
|
|
.afi = BGP_AF_IPV4,
|
|
.net = NET_IP4,
|
|
.name = "ipv4",
|
|
.encode_nlri = bgp_encode_nlri_ip4,
|
|
.decode_nlri = bgp_decode_nlri_ip4,
|
|
.encode_next_hop = bgp_encode_next_hop_ip,
|
|
.decode_next_hop = bgp_decode_next_hop_ip,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_IPV4_MC,
|
|
.net = NET_IP4,
|
|
.name = "ipv4-mc",
|
|
.encode_nlri = bgp_encode_nlri_ip4,
|
|
.decode_nlri = bgp_decode_nlri_ip4,
|
|
.encode_next_hop = bgp_encode_next_hop_ip,
|
|
.decode_next_hop = bgp_decode_next_hop_ip,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_IPV4_MPLS,
|
|
.net = NET_IP4,
|
|
.mpls = 1,
|
|
.name = "ipv4-mpls",
|
|
.encode_nlri = bgp_encode_nlri_ip4,
|
|
.decode_nlri = bgp_decode_nlri_ip4,
|
|
.encode_next_hop = bgp_encode_next_hop_ip,
|
|
.decode_next_hop = bgp_decode_next_hop_ip,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_IPV6,
|
|
.net = NET_IP6,
|
|
.name = "ipv6",
|
|
.encode_nlri = bgp_encode_nlri_ip6,
|
|
.decode_nlri = bgp_decode_nlri_ip6,
|
|
.encode_next_hop = bgp_encode_next_hop_ip,
|
|
.decode_next_hop = bgp_decode_next_hop_ip,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_IPV6_MC,
|
|
.net = NET_IP6,
|
|
.name = "ipv6-mc",
|
|
.encode_nlri = bgp_encode_nlri_ip6,
|
|
.decode_nlri = bgp_decode_nlri_ip6,
|
|
.encode_next_hop = bgp_encode_next_hop_ip,
|
|
.decode_next_hop = bgp_decode_next_hop_ip,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_IPV6_MPLS,
|
|
.net = NET_IP6,
|
|
.mpls = 1,
|
|
.name = "ipv6-mpls",
|
|
.encode_nlri = bgp_encode_nlri_ip6,
|
|
.decode_nlri = bgp_decode_nlri_ip6,
|
|
.encode_next_hop = bgp_encode_next_hop_ip,
|
|
.decode_next_hop = bgp_decode_next_hop_ip,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_VPN4_MPLS,
|
|
.net = NET_VPN4,
|
|
.mpls = 1,
|
|
.name = "vpn4-mpls",
|
|
.encode_nlri = bgp_encode_nlri_vpn4,
|
|
.decode_nlri = bgp_decode_nlri_vpn4,
|
|
.encode_next_hop = bgp_encode_next_hop_vpn,
|
|
.decode_next_hop = bgp_decode_next_hop_vpn,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_VPN6_MPLS,
|
|
.net = NET_VPN6,
|
|
.mpls = 1,
|
|
.name = "vpn6-mpls",
|
|
.encode_nlri = bgp_encode_nlri_vpn6,
|
|
.decode_nlri = bgp_decode_nlri_vpn6,
|
|
.encode_next_hop = bgp_encode_next_hop_vpn,
|
|
.decode_next_hop = bgp_decode_next_hop_vpn,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_VPN4_MC,
|
|
.net = NET_VPN4,
|
|
.name = "vpn4-mc",
|
|
.encode_nlri = bgp_encode_nlri_vpn4,
|
|
.decode_nlri = bgp_decode_nlri_vpn4,
|
|
.encode_next_hop = bgp_encode_next_hop_vpn,
|
|
.decode_next_hop = bgp_decode_next_hop_vpn,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_VPN6_MC,
|
|
.net = NET_VPN6,
|
|
.name = "vpn6-mc",
|
|
.encode_nlri = bgp_encode_nlri_vpn6,
|
|
.decode_nlri = bgp_decode_nlri_vpn6,
|
|
.encode_next_hop = bgp_encode_next_hop_vpn,
|
|
.decode_next_hop = bgp_decode_next_hop_vpn,
|
|
.update_next_hop = bgp_update_next_hop_ip,
|
|
},
|
|
{
|
|
.afi = BGP_AF_FLOW4,
|
|
.net = NET_FLOW4,
|
|
.no_igp = 1,
|
|
.name = "flow4",
|
|
.encode_nlri = bgp_encode_nlri_flow4,
|
|
.decode_nlri = bgp_decode_nlri_flow4,
|
|
.encode_next_hop = bgp_encode_next_hop_none,
|
|
.decode_next_hop = bgp_decode_next_hop_none,
|
|
.update_next_hop = bgp_update_next_hop_none,
|
|
},
|
|
{
|
|
.afi = BGP_AF_FLOW6,
|
|
.net = NET_FLOW6,
|
|
.no_igp = 1,
|
|
.name = "flow6",
|
|
.encode_nlri = bgp_encode_nlri_flow6,
|
|
.decode_nlri = bgp_decode_nlri_flow6,
|
|
.encode_next_hop = bgp_encode_next_hop_none,
|
|
.decode_next_hop = bgp_decode_next_hop_none,
|
|
.update_next_hop = bgp_update_next_hop_none,
|
|
},
|
|
};
|
|
|
|
const struct bgp_af_desc *
|
|
bgp_get_af_desc(u32 afi)
|
|
{
|
|
uint i;
|
|
for (i = 0; i < ARRAY_SIZE(bgp_af_table); i++)
|
|
if (bgp_af_table[i].afi == afi)
|
|
return &bgp_af_table[i];
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static inline uint
|
|
bgp_encode_nlri(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, byte *end)
|
|
{
|
|
return s->ptx->c->desc->encode_nlri(s, buck, buf, end - buf);
|
|
}
|
|
|
|
static inline uint
|
|
bgp_encode_next_hop(struct bgp_write_state *s, eattr *nh, byte *buf)
|
|
{
|
|
return s->ptx->c->desc->encode_next_hop(s, nh, buf, 255);
|
|
}
|
|
|
|
void
|
|
bgp_update_next_hop(struct bgp_export_state *s, eattr *a, ea_list **to)
|
|
{
|
|
s->channel->desc->update_next_hop(s, a, to);
|
|
}
|
|
|
|
#define MAX_ATTRS_LENGTH (end-buf+BGP_HEADER_LENGTH - 1024)
|
|
|
|
static byte *
|
|
bgp_create_ip_reach(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, byte *end)
|
|
{
|
|
/*
|
|
* 2 B Withdrawn Routes Length (zero)
|
|
* --- IPv4 Withdrawn Routes NLRI (unused)
|
|
* 2 B Total Path Attribute Length
|
|
* var Path Attributes
|
|
* var IPv4 Network Layer Reachability Information
|
|
*/
|
|
|
|
ASSERT_DIE(s->ptx->withdraw_bucket != buck);
|
|
|
|
int lr, la;
|
|
|
|
la = bgp_encode_attrs(s, buck->attrs, buf+4, buf + MAX_ATTRS_LENGTH);
|
|
if (la < 0)
|
|
{
|
|
/* Attribute list too long */
|
|
bgp_withdraw_bucket(s->ptx, buck);
|
|
return NULL;
|
|
}
|
|
|
|
put_u16(buf+0, 0);
|
|
put_u16(buf+2, la);
|
|
|
|
lr = bgp_encode_nlri(s, buck, buf+4+la, end);
|
|
|
|
return buf+4+la+lr;
|
|
}
|
|
|
|
static byte *
|
|
bgp_create_mp_reach(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, byte *end)
|
|
{
|
|
ASSERT_DIE(s->ptx->withdraw_bucket != buck);
|
|
|
|
/*
|
|
* 2 B IPv4 Withdrawn Routes Length (zero)
|
|
* --- IPv4 Withdrawn Routes NLRI (unused)
|
|
* 2 B Total Path Attribute Length
|
|
* 1 B MP_REACH_NLRI hdr - Attribute Flags
|
|
* 1 B MP_REACH_NLRI hdr - Attribute Type Code
|
|
* 2 B MP_REACH_NLRI hdr - Length of Attribute Data
|
|
* 2 B MP_REACH_NLRI data - Address Family Identifier
|
|
* 1 B MP_REACH_NLRI data - Subsequent Address Family Identifier
|
|
* 1 B MP_REACH_NLRI data - Length of Next Hop Network Address
|
|
* var MP_REACH_NLRI data - Network Address of Next Hop
|
|
* 1 B MP_REACH_NLRI data - Reserved (zero)
|
|
* var MP_REACH_NLRI data - Network Layer Reachability Information
|
|
* var Rest of Path Attributes
|
|
* --- IPv4 Network Layer Reachability Information (unused)
|
|
*/
|
|
|
|
int lh, lr, la; /* Lengths of next hop, NLRI and attributes */
|
|
|
|
/* Begin of MP_REACH_NLRI atribute */
|
|
buf[4] = BAF_OPTIONAL | BAF_EXT_LEN;
|
|
buf[5] = BA_MP_REACH_NLRI;
|
|
put_u16(buf+6, 0); /* Will be fixed later */
|
|
put_af3(buf+8, s->ptx->c->afi);
|
|
byte *pos = buf+11;
|
|
|
|
/* Encode attributes to temporary buffer */
|
|
byte *abuf = alloca(MAX_ATTRS_LENGTH);
|
|
la = bgp_encode_attrs(s, buck->attrs, abuf, abuf + MAX_ATTRS_LENGTH);
|
|
if (la < 0)
|
|
{
|
|
/* Attribute list too long */
|
|
bgp_withdraw_bucket(s->ptx, buck);
|
|
return NULL;
|
|
}
|
|
|
|
/* Encode the next hop */
|
|
lh = bgp_encode_next_hop(s, s->mp_next_hop, pos+1);
|
|
*pos = lh;
|
|
pos += 1+lh;
|
|
|
|
/* Reserved field */
|
|
*pos++ = 0;
|
|
|
|
/* Encode the NLRI */
|
|
lr = bgp_encode_nlri(s, buck, pos, end - la);
|
|
pos += lr;
|
|
|
|
/* End of MP_REACH_NLRI atribute, update data length */
|
|
put_u16(buf+6, pos-buf-8);
|
|
|
|
/* Copy remaining attributes */
|
|
memcpy(pos, abuf, la);
|
|
pos += la;
|
|
|
|
/* Initial UPDATE fields */
|
|
put_u16(buf+0, 0);
|
|
put_u16(buf+2, pos-buf-4);
|
|
|
|
return pos;
|
|
}
|
|
|
|
#undef MAX_ATTRS_LENGTH
|
|
|
|
static byte *
|
|
bgp_create_ip_unreach(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, byte *end)
|
|
{
|
|
/*
|
|
* 2 B Withdrawn Routes Length
|
|
* var IPv4 Withdrawn Routes NLRI
|
|
* 2 B Total Path Attribute Length (zero)
|
|
* --- Path Attributes (unused)
|
|
* --- IPv4 Network Layer Reachability Information (unused)
|
|
*/
|
|
|
|
uint len = bgp_encode_nlri(s, buck, buf+2, end);
|
|
|
|
put_u16(buf+0, len);
|
|
put_u16(buf+2+len, 0);
|
|
|
|
return buf+4+len;
|
|
}
|
|
|
|
static byte *
|
|
bgp_create_mp_unreach(struct bgp_write_state *s, struct bgp_bucket *buck, byte *buf, byte *end)
|
|
{
|
|
/*
|
|
* 2 B Withdrawn Routes Length (zero)
|
|
* --- IPv4 Withdrawn Routes NLRI (unused)
|
|
* 2 B Total Path Attribute Length
|
|
* 1 B MP_UNREACH_NLRI hdr - Attribute Flags
|
|
* 1 B MP_UNREACH_NLRI hdr - Attribute Type Code
|
|
* 2 B MP_UNREACH_NLRI hdr - Length of Attribute Data
|
|
* 2 B MP_UNREACH_NLRI data - Address Family Identifier
|
|
* 1 B MP_UNREACH_NLRI data - Subsequent Address Family Identifier
|
|
* var MP_UNREACH_NLRI data - Network Layer Reachability Information
|
|
* --- IPv4 Network Layer Reachability Information (unused)
|
|
*/
|
|
|
|
uint len = bgp_encode_nlri(s, buck, buf+11, end);
|
|
|
|
put_u16(buf+0, 0);
|
|
put_u16(buf+2, 7+len);
|
|
|
|
/* Begin of MP_UNREACH_NLRI atribute */
|
|
buf[4] = BAF_OPTIONAL | BAF_EXT_LEN;
|
|
buf[5] = BA_MP_UNREACH_NLRI;
|
|
put_u16(buf+6, 3+len);
|
|
put_af3(buf+8, s->ptx->c->afi);
|
|
|
|
return buf+11+len;
|
|
}
|
|
|
|
|
|
#ifdef CONFIG_BMP
|
|
|
|
static byte *
|
|
bgp_create_update_bmp(struct bgp_channel *c, byte *buf, struct bgp_bucket *buck, bool update)
|
|
{
|
|
struct bgp_proto *p = (void *) c->c.proto;
|
|
byte *end = buf + (BGP_MAX_EXT_MSG_LENGTH - BGP_HEADER_LENGTH);
|
|
byte *res = NULL;
|
|
/* FIXME: must be a bit shorter */
|
|
|
|
struct bgp_caps *peer = p->conn->remote_caps;
|
|
const struct bgp_af_caps *rem = bgp_find_af_caps(peer, c->afi);
|
|
|
|
struct bgp_write_state s = {
|
|
.proto = p,
|
|
.channel = c,
|
|
.pool = tmp_linpool,
|
|
.mp_reach = (c->afi != BGP_AF_IPV4) || rem->ext_next_hop,
|
|
.as4_session = 1,
|
|
.add_path = c->add_path_rx,
|
|
.mpls = c->desc->mpls,
|
|
};
|
|
|
|
if (!update)
|
|
{
|
|
res = !s.mp_reach ?
|
|
bgp_create_ip_unreach(&s, buck, buf, end):
|
|
bgp_create_mp_unreach(&s, buck, buf, end);
|
|
}
|
|
else
|
|
{
|
|
res = !s.mp_reach ?
|
|
bgp_create_ip_reach(&s, buck, buf, end):
|
|
bgp_create_mp_reach(&s, buck, buf, end);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static byte *
|
|
bgp_bmp_prepare_bgp_hdr(byte *buf, const u16 msg_size, const u8 msg_type)
|
|
{
|
|
memset(buf + BGP_MSG_HDR_MARKER_POS, 0xff, BGP_MSG_HDR_MARKER_SIZE);
|
|
put_u16(buf + BGP_MSG_HDR_LENGTH_POS, msg_size);
|
|
put_u8(buf + BGP_MSG_HDR_TYPE_POS, msg_type);
|
|
|
|
return buf + BGP_MSG_HDR_TYPE_POS + BGP_MSG_HDR_TYPE_SIZE;
|
|
}
|
|
|
|
byte *
|
|
bgp_bmp_encode_rte(struct bgp_channel *c, byte *buf, const net_addr *n,
|
|
const struct rte *new, const struct rte_src *src)
|
|
{
|
|
// struct bgp_proto *p = (void *) c->c.proto;
|
|
byte *pkt = buf + BGP_HEADER_LENGTH;
|
|
|
|
ea_list *attrs = new ? new->attrs : NULL;
|
|
uint ea_size = new ? (sizeof(ea_list) + attrs->count * sizeof(eattr)) : 0;
|
|
uint bucket_size = sizeof(struct bgp_bucket) + ea_size;
|
|
uint prefix_size = sizeof(struct bgp_prefix) + n->length;
|
|
|
|
struct lp_state *tmpp = lp_save(tmp_linpool);
|
|
|
|
/* Temporary bucket */
|
|
struct bgp_bucket *b = tmp_allocz(bucket_size);
|
|
b->bmp = 1;
|
|
init_list(&b->prefixes);
|
|
|
|
if (attrs)
|
|
memcpy(b->eattrs, attrs, ea_size);
|
|
|
|
/* Temporary prefix */
|
|
struct bgp_prefix *px = tmp_allocz(prefix_size);
|
|
px->path_id = (u32) src->private_id;
|
|
net_copy(px->net, n);
|
|
add_tail(&b->prefixes, &px->buck_node_xx);
|
|
|
|
byte *end = bgp_create_update_bmp(c, pkt, b, !!new);
|
|
|
|
if (end)
|
|
bgp_bmp_prepare_bgp_hdr(buf, end - buf, PKT_UPDATE);
|
|
|
|
lp_restore(tmp_linpool, tmpp);
|
|
|
|
return end;
|
|
}
|
|
|
|
#endif /* CONFIG_BMP */
|
|
|
|
|
|
static byte *
|
|
bgp_create_update(struct bgp_channel *c, byte *buf)
|
|
{
|
|
struct bgp_proto *p = (void *) c->c.proto;
|
|
struct bgp_bucket *buck;
|
|
byte *end = buf + (bgp_max_packet_length(p->conn) - BGP_HEADER_LENGTH);
|
|
byte *res = NULL;
|
|
struct lp_state *tmpp = NULL;
|
|
|
|
BGP_PTX_LOCK(c->tx, ptx);
|
|
|
|
again:
|
|
if (tmpp)
|
|
lp_restore(tmp_linpool, tmpp);
|
|
|
|
tmpp = lp_save(tmp_linpool);
|
|
|
|
/* Initialize write state */
|
|
struct bgp_write_state s = {
|
|
.proto = p,
|
|
.ptx = ptx,
|
|
.pool = tmp_linpool,
|
|
.mp_reach = (c->afi != BGP_AF_IPV4) || c->ext_next_hop,
|
|
.as4_session = p->as4_session,
|
|
.add_path = c->add_path_tx,
|
|
.mpls = c->desc->mpls,
|
|
};
|
|
|
|
/* Try unreachable bucket */
|
|
if ((buck = ptx->withdraw_bucket) && !EMPTY_LIST(buck->prefixes))
|
|
{
|
|
res = (c->afi == BGP_AF_IPV4) && !c->ext_next_hop ?
|
|
bgp_create_ip_unreach(&s, buck, buf, end):
|
|
bgp_create_mp_unreach(&s, buck, buf, end);
|
|
|
|
goto done;
|
|
}
|
|
|
|
/* Try reachable buckets */
|
|
if (!EMPTY_LIST(ptx->bucket_queue))
|
|
{
|
|
buck = HEAD(ptx->bucket_queue);
|
|
|
|
/* Cleanup empty buckets */
|
|
if (bgp_done_bucket(ptx, buck))
|
|
goto again;
|
|
|
|
res = !s.mp_reach ?
|
|
bgp_create_ip_reach(&s, buck, buf, end):
|
|
bgp_create_mp_reach(&s, buck, buf, end);
|
|
|
|
bgp_done_bucket(ptx, buck);
|
|
|
|
if (!res)
|
|
goto again;
|
|
|
|
goto done;
|
|
}
|
|
|
|
/* No more prefixes to send */
|
|
lp_restore(tmp_linpool, tmpp);
|
|
return NULL;
|
|
|
|
done:
|
|
BGP_TRACE_RL(&rl_snd_update, D_PACKETS, "Sending UPDATE");
|
|
p->stats.tx_updates++;
|
|
lp_restore(tmp_linpool, tmpp);
|
|
|
|
return res;
|
|
}
|
|
|
|
static byte *
|
|
bgp_create_ip_end_mark(struct bgp_channel *c UNUSED, byte *buf)
|
|
{
|
|
/* Empty update packet */
|
|
put_u32(buf, 0);
|
|
|
|
return buf+4;
|
|
}
|
|
|
|
static byte *
|
|
bgp_create_mp_end_mark(struct bgp_channel *c, byte *buf)
|
|
{
|
|
put_u16(buf+0, 0);
|
|
put_u16(buf+2, 6); /* length 4--9 */
|
|
|
|
/* Empty MP_UNREACH_NLRI atribute */
|
|
buf[4] = BAF_OPTIONAL;
|
|
buf[5] = BA_MP_UNREACH_NLRI;
|
|
buf[6] = 3; /* Length 7--9 */
|
|
put_af3(buf+7, c->afi);
|
|
|
|
return buf+10;
|
|
}
|
|
|
|
byte *
|
|
bgp_create_end_mark_(struct bgp_channel *c, byte *buf)
|
|
{
|
|
return (c->afi == BGP_AF_IPV4) ?
|
|
bgp_create_ip_end_mark(c, buf):
|
|
bgp_create_mp_end_mark(c, buf);
|
|
}
|
|
|
|
static byte *
|
|
bgp_create_end_mark(struct bgp_channel *c, byte *buf)
|
|
{
|
|
struct bgp_proto *p = (void *) c->c.proto;
|
|
|
|
BGP_TRACE(D_PACKETS, "Sending END-OF-RIB");
|
|
p->stats.tx_updates++;
|
|
|
|
return bgp_create_end_mark_(c, buf);
|
|
}
|
|
|
|
static inline void
|
|
bgp_rx_end_mark(struct bgp_parse_state *s, u32 afi)
|
|
{
|
|
struct bgp_proto *p = s->proto;
|
|
struct bgp_channel *c = bgp_get_channel(p, afi);
|
|
|
|
BGP_TRACE(D_PACKETS, "Got END-OF-RIB");
|
|
|
|
if (!c)
|
|
DISCARD(BAD_AFI, BGP_AFI(afi), BGP_SAFI(afi));
|
|
|
|
if (c->load_state == BFS_LOADING)
|
|
c->load_state = BFS_NONE;
|
|
|
|
if (p->p.gr_recovery)
|
|
channel_graceful_restart_unlock(&c->c);
|
|
|
|
if (c->gr_active)
|
|
bgp_graceful_restart_done(c);
|
|
}
|
|
|
|
static inline void
|
|
bgp_decode_nlri(struct bgp_parse_state *s, u32 afi, byte *nlri, uint len, ea_list *ea, byte *nh, uint nh_len)
|
|
{
|
|
struct bgp_channel *c = bgp_get_channel(s->proto, afi);
|
|
|
|
if (!c)
|
|
DISCARD(BAD_AFI, BGP_AFI(afi), BGP_SAFI(afi));
|
|
|
|
s->channel = c;
|
|
s->add_path = c->add_path_rx;
|
|
s->mpls = c->desc->mpls;
|
|
|
|
s->last_id = 0;
|
|
s->last_src = s->proto->p.main_source;
|
|
rt_lock_source(s->last_src);
|
|
|
|
/*
|
|
* IPv4 BGP and MP-BGP may be used together in one update, therefore we do not
|
|
* add BA_NEXT_HOP in bgp_decode_attrs(), but we add it here independently for
|
|
* IPv4 BGP and MP-BGP. We undo the attribute (and possibly others attached by
|
|
* decode_next_hop hooks) by restoring a->eattrs afterwards.
|
|
*/
|
|
|
|
if (ea)
|
|
{
|
|
ea_set_attr_data(&ea, &ea_gen_from, 0, &s->proto->remote_ip, sizeof(ip_addr));
|
|
ea_set_attr_u32(&ea, &ea_gen_preference, 0, c->c.preference);
|
|
ea_set_attr_u32(&ea, &ea_gen_source, 0, RTS_BGP);
|
|
|
|
c->desc->decode_next_hop(s, nh, nh_len, &ea);
|
|
bgp_finish_attrs(s, &ea);
|
|
|
|
/* Handle withdraw during next hop decoding */
|
|
if (s->err_withdraw)
|
|
ea = NULL;
|
|
|
|
if (ea)
|
|
ea = ea_lookup_tmp(ea, 0, EALS_CUSTOM);
|
|
}
|
|
|
|
c->desc->decode_nlri(s, nlri, len, ea);
|
|
|
|
rt_unlock_source(s->last_src);
|
|
}
|
|
|
|
static void
|
|
bgp_rx_update(struct bgp_conn *conn, byte *pkt, uint len)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
ea_list *ea = NULL;
|
|
|
|
BGP_TRACE_RL(&rl_rcv_update, D_PACKETS, "Got UPDATE");
|
|
p->last_rx_update = current_time();
|
|
p->stats.rx_updates++;
|
|
|
|
/* Workaround for some BGP implementations that skip initial KEEPALIVE */
|
|
if (conn->state == BS_OPENCONFIRM)
|
|
bgp_conn_enter_established_state(conn);
|
|
|
|
if (conn->state != BS_ESTABLISHED)
|
|
{ bgp_error(conn, 5, fsm_err_subcode[conn->state], NULL, 0); return; }
|
|
|
|
bgp_start_timer(p, conn->hold_timer, conn->hold_time);
|
|
|
|
struct lp_state *tmpp = lp_save(tmp_linpool);
|
|
|
|
/* Initialize parse state */
|
|
struct bgp_parse_state s = {
|
|
.proto = p,
|
|
.pool = tmp_linpool,
|
|
.as4_session = p->as4_session,
|
|
};
|
|
|
|
/* Parse error handler */
|
|
if (setjmp(s.err_jmpbuf))
|
|
{
|
|
bgp_error(conn, 3, s.err_subcode, NULL, 0);
|
|
goto done;
|
|
}
|
|
|
|
/* Check minimal length */
|
|
if (len < 23)
|
|
{
|
|
bgp_error(conn, 1, 2, pkt+16, 2);
|
|
goto done;
|
|
}
|
|
|
|
/* Skip fixed header */
|
|
uint pos = 19;
|
|
|
|
/*
|
|
* UPDATE message format
|
|
*
|
|
* 2 B IPv4 Withdrawn Routes Length
|
|
* var IPv4 Withdrawn Routes NLRI
|
|
* 2 B Total Path Attribute Length
|
|
* var Path Attributes
|
|
* var IPv4 Reachable Routes NLRI
|
|
*/
|
|
|
|
s.ip_unreach_len = get_u16(pkt + pos);
|
|
s.ip_unreach_nlri = pkt + pos + 2;
|
|
pos += 2 + s.ip_unreach_len;
|
|
|
|
if (pos + 2 > len)
|
|
bgp_parse_error(&s, 1);
|
|
|
|
s.attr_len = get_u16(pkt + pos);
|
|
s.attrs = pkt + pos + 2;
|
|
pos += 2 + s.attr_len;
|
|
|
|
if (pos > len)
|
|
bgp_parse_error(&s, 1);
|
|
|
|
s.ip_reach_len = len - pos;
|
|
s.ip_reach_nlri = pkt + pos;
|
|
|
|
if (s.attr_len)
|
|
ea = bgp_decode_attrs(&s, s.attrs, s.attr_len);
|
|
else
|
|
ea = NULL;
|
|
|
|
/* Check for End-of-RIB marker */
|
|
if (!s.attr_len && !s.ip_unreach_len && !s.ip_reach_len)
|
|
{ bgp_rx_end_mark(&s, BGP_AF_IPV4); goto done; }
|
|
|
|
/* Check for MP End-of-RIB marker */
|
|
if ((s.attr_len < 8) && !s.ip_unreach_len && !s.ip_reach_len &&
|
|
!s.mp_reach_len && !s.mp_unreach_len && s.mp_unreach_af)
|
|
{ bgp_rx_end_mark(&s, s.mp_unreach_af); goto done; }
|
|
|
|
if (s.ip_unreach_len)
|
|
bgp_decode_nlri(&s, BGP_AF_IPV4, s.ip_unreach_nlri, s.ip_unreach_len, NULL, NULL, 0);
|
|
|
|
if (s.mp_unreach_len)
|
|
bgp_decode_nlri(&s, s.mp_unreach_af, s.mp_unreach_nlri, s.mp_unreach_len, NULL, NULL, 0);
|
|
|
|
s.reach_nlri_step = 1;
|
|
|
|
if (s.ip_reach_len)
|
|
bgp_decode_nlri(&s, BGP_AF_IPV4, s.ip_reach_nlri, s.ip_reach_len,
|
|
ea, s.ip_next_hop_data, s.ip_next_hop_len);
|
|
|
|
if (s.mp_reach_len)
|
|
bgp_decode_nlri(&s, s.mp_reach_af, s.mp_reach_nlri, s.mp_reach_len,
|
|
ea, s.mp_next_hop_data, s.mp_next_hop_len);
|
|
|
|
done:
|
|
lp_restore(tmp_linpool, tmpp);
|
|
return;
|
|
}
|
|
|
|
static uint
|
|
bgp_find_update_afi(byte *pos, uint len)
|
|
{
|
|
/*
|
|
* This is stripped-down version of bgp_rx_update(), bgp_decode_attrs() and
|
|
* bgp_decode_mp_[un]reach_nlri() used by MRT code in order to find out which
|
|
* AFI/SAFI is associated with incoming UPDATE. Returns 0 for framing errors.
|
|
*/
|
|
if (len < 23)
|
|
return 0;
|
|
|
|
/* Assume there is no withrawn NLRI, read lengths and move to attribute list */
|
|
uint wlen = get_u16(pos + 19);
|
|
uint alen = get_u16(pos + 21);
|
|
ADVANCE(pos, len, 23);
|
|
|
|
/* Either non-zero withdrawn NLRI, non-zero reachable NLRI, or IPv4 End-of-RIB */
|
|
if ((wlen != 0) || (alen < len) || !alen)
|
|
return BGP_AF_IPV4;
|
|
|
|
if (alen > len)
|
|
return 0;
|
|
|
|
/* Process attribute list (alen == len) */
|
|
while (len)
|
|
{
|
|
if (len < 2)
|
|
return 0;
|
|
|
|
uint flags = pos[0];
|
|
uint code = pos[1];
|
|
ADVANCE(pos, len, 2);
|
|
|
|
uint ll = !(flags & BAF_EXT_LEN) ? 1 : 2;
|
|
if (len < ll)
|
|
return 0;
|
|
|
|
/* Read attribute length and move to attribute body */
|
|
alen = (ll == 1) ? get_u8(pos) : get_u16(pos);
|
|
ADVANCE(pos, len, ll);
|
|
|
|
if (len < alen)
|
|
return 0;
|
|
|
|
/* Found MP NLRI */
|
|
if ((code == BA_MP_REACH_NLRI) || (code == BA_MP_UNREACH_NLRI))
|
|
{
|
|
if (alen < 3)
|
|
return 0;
|
|
|
|
return BGP_AF(get_u16(pos), pos[2]);
|
|
}
|
|
|
|
/* Move to the next attribute */
|
|
ADVANCE(pos, len, alen);
|
|
}
|
|
|
|
/* No basic or MP NLRI, but there are some attributes -> error */
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* ROUTE-REFRESH
|
|
*/
|
|
|
|
static inline byte *
|
|
bgp_create_route_refresh(struct bgp_channel *c, byte *buf)
|
|
{
|
|
struct bgp_proto *p = (void *) c->c.proto;
|
|
|
|
BGP_TRACE(D_PACKETS, "Sending ROUTE-REFRESH");
|
|
|
|
/* Original route refresh request, RFC 2918 */
|
|
put_af4(buf, c->afi);
|
|
buf[2] = BGP_RR_REQUEST;
|
|
|
|
return buf+4;
|
|
}
|
|
|
|
static inline byte *
|
|
bgp_create_begin_refresh(struct bgp_channel *c, byte *buf)
|
|
{
|
|
struct bgp_proto *p = (void *) c->c.proto;
|
|
|
|
BGP_TRACE(D_PACKETS, "Sending BEGIN-OF-RR");
|
|
|
|
/* Demarcation of beginning of route refresh (BoRR), RFC 7313 */
|
|
put_af4(buf, c->afi);
|
|
buf[2] = BGP_RR_BEGIN;
|
|
|
|
return buf+4;
|
|
}
|
|
|
|
static inline byte *
|
|
bgp_create_end_refresh(struct bgp_channel *c, byte *buf)
|
|
{
|
|
struct bgp_proto *p = (void *) c->c.proto;
|
|
|
|
BGP_TRACE(D_PACKETS, "Sending END-OF-RR");
|
|
|
|
/* Demarcation of ending of route refresh (EoRR), RFC 7313 */
|
|
put_af4(buf, c->afi);
|
|
buf[2] = BGP_RR_END;
|
|
|
|
return buf+4;
|
|
}
|
|
|
|
static void
|
|
bgp_rx_route_refresh(struct bgp_conn *conn, byte *pkt, uint len)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
|
|
if (conn->state != BS_ESTABLISHED)
|
|
{ bgp_error(conn, 5, fsm_err_subcode[conn->state], NULL, 0); return; }
|
|
|
|
if (!conn->local_caps->route_refresh)
|
|
{ bgp_error(conn, 1, 3, pkt+18, 1); return; }
|
|
|
|
if (len < (BGP_HEADER_LENGTH + 4))
|
|
{ bgp_error(conn, 1, 2, pkt+16, 2); return; }
|
|
|
|
if (len > (BGP_HEADER_LENGTH + 4))
|
|
{ bgp_error(conn, 7, 1, pkt, MIN(len, 2048)); return; }
|
|
|
|
struct bgp_channel *c = bgp_get_channel(p, get_af4(pkt+19));
|
|
if (!c)
|
|
{
|
|
log(L_WARN "%s: Got ROUTE-REFRESH subtype %u for AF %u.%u, ignoring",
|
|
p->p.name, pkt[21], get_u16(pkt+19), pkt[22]);
|
|
return;
|
|
}
|
|
|
|
/* RFC 7313 redefined reserved field as RR message subtype */
|
|
uint subtype = p->enhanced_refresh ? pkt[21] : BGP_RR_REQUEST;
|
|
|
|
switch (subtype)
|
|
{
|
|
case BGP_RR_REQUEST:
|
|
BGP_TRACE(D_PACKETS, "Got ROUTE-REFRESH");
|
|
if (c->tx_keep)
|
|
bgp_tx_resend(p, c);
|
|
else
|
|
rt_export_refeed(&c->c.out_req, NULL);
|
|
break;
|
|
|
|
case BGP_RR_BEGIN:
|
|
BGP_TRACE(D_PACKETS, "Got BEGIN-OF-RR");
|
|
bgp_refresh_begin(c);
|
|
break;
|
|
|
|
case BGP_RR_END:
|
|
BGP_TRACE(D_PACKETS, "Got END-OF-RR");
|
|
bgp_refresh_end(c);
|
|
break;
|
|
|
|
default:
|
|
log(L_WARN "%s: Got ROUTE-REFRESH message with unknown subtype %u, ignoring",
|
|
p->p.name, subtype);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static inline struct bgp_channel *
|
|
bgp_get_channel_to_send(struct bgp_proto *p, struct bgp_conn *conn)
|
|
{
|
|
uint i = conn->last_channel;
|
|
|
|
/* Try the last channel, but at most several times */
|
|
if ((conn->channels_to_send & (1 << i)) &&
|
|
(conn->last_channel_count < 16))
|
|
goto found;
|
|
|
|
/* Find channel with non-zero channels_to_send */
|
|
do
|
|
{
|
|
i++;
|
|
if (i >= p->channel_count)
|
|
i = 0;
|
|
}
|
|
while (! (conn->channels_to_send & (1 << i)));
|
|
|
|
/* Use that channel */
|
|
conn->last_channel = i;
|
|
conn->last_channel_count = 0;
|
|
|
|
found:
|
|
conn->last_channel_count++;
|
|
return p->channel_map[i];
|
|
}
|
|
|
|
static inline int
|
|
bgp_send(struct bgp_conn *conn, uint type, uint len)
|
|
{
|
|
sock *sk = conn->sk;
|
|
byte *buf = sk->tbuf;
|
|
|
|
conn->bgp->stats.tx_messages++;
|
|
conn->bgp->stats.tx_bytes += len;
|
|
|
|
memset(buf, 0xff, BGP_HDR_MARKER_LENGTH);
|
|
put_u16(buf+16, len);
|
|
buf[18] = type;
|
|
|
|
int success = sk_send(sk, len);
|
|
if (success && ((conn->state == BS_ESTABLISHED) || (conn->state == BS_OPENCONFIRM)))
|
|
bgp_start_timer(conn->bgp, conn->send_hold_timer, conn->send_hold_time);
|
|
|
|
return success;
|
|
}
|
|
|
|
/**
|
|
* bgp_fire_tx - transmit packets
|
|
* @conn: connection
|
|
*
|
|
* Whenever the transmit buffers of the underlying TCP connection
|
|
* are free and we have any packets queued for sending, the socket functions
|
|
* call bgp_fire_tx() which takes care of selecting the highest priority packet
|
|
* queued (Notification > Keepalive > Open > Update), assembling its header
|
|
* and body and sending it to the connection.
|
|
*/
|
|
static int
|
|
bgp_fire_tx(struct bgp_conn *conn)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
struct bgp_channel *c;
|
|
byte *buf, *pkt, *end;
|
|
uint s;
|
|
|
|
if (!conn->sk)
|
|
return 0;
|
|
|
|
buf = conn->sk->tbuf;
|
|
pkt = buf + BGP_HEADER_LENGTH;
|
|
s = conn->packets_to_send;
|
|
|
|
if (s & (1 << PKT_SCHEDULE_CLOSE))
|
|
{
|
|
/* We can finally close connection and enter idle state */
|
|
bgp_conn_enter_idle_state(conn);
|
|
return 0;
|
|
}
|
|
if (s & (1 << PKT_NOTIFICATION))
|
|
{
|
|
conn->packets_to_send = 1 << PKT_SCHEDULE_CLOSE;
|
|
end = bgp_create_notification(conn, pkt);
|
|
return bgp_send(conn, PKT_NOTIFICATION, end - buf);
|
|
}
|
|
else if (s & (1 << PKT_OPEN))
|
|
{
|
|
conn->packets_to_send &= ~(1 << PKT_OPEN);
|
|
end = bgp_create_open(conn, pkt);
|
|
|
|
conn->local_open_msg = bgp_copy_open(p, buf, end - buf);
|
|
conn->local_open_length = end - buf - BGP_HEADER_LENGTH;
|
|
|
|
return bgp_send(conn, PKT_OPEN, end - buf);
|
|
}
|
|
else if (s & (1 << PKT_KEEPALIVE))
|
|
{
|
|
conn->packets_to_send &= ~(1 << PKT_KEEPALIVE);
|
|
BGP_TRACE(D_PACKETS, "Sending KEEPALIVE");
|
|
bgp_start_timer(p, conn->keepalive_timer, conn->keepalive_time);
|
|
return bgp_send(conn, PKT_KEEPALIVE, BGP_HEADER_LENGTH);
|
|
}
|
|
else while (conn->channels_to_send)
|
|
{
|
|
c = bgp_get_channel_to_send(p, conn);
|
|
s = c->packets_to_send;
|
|
|
|
if (s & (1 << PKT_ROUTE_REFRESH))
|
|
{
|
|
c->packets_to_send &= ~(1 << PKT_ROUTE_REFRESH);
|
|
end = bgp_create_route_refresh(c, pkt);
|
|
return bgp_send(conn, PKT_ROUTE_REFRESH, end - buf);
|
|
}
|
|
else if (s & (1 << PKT_BEGIN_REFRESH))
|
|
{
|
|
/* BoRR is a subtype of RR, but uses separate bit in packets_to_send */
|
|
c->packets_to_send &= ~(1 << PKT_BEGIN_REFRESH);
|
|
end = bgp_create_begin_refresh(c, pkt);
|
|
return bgp_send(conn, PKT_ROUTE_REFRESH, end - buf);
|
|
}
|
|
else if (s & (1 << PKT_UPDATE))
|
|
{
|
|
end = bgp_create_update(c, pkt);
|
|
if (end)
|
|
return bgp_send(conn, PKT_UPDATE, end - buf);
|
|
|
|
/* No update to send, perhaps we need to send End-of-RIB or EoRR */
|
|
c->packets_to_send = 0;
|
|
conn->channels_to_send &= ~(1 << c->index);
|
|
|
|
if (c->feed_state == BFS_LOADED)
|
|
{
|
|
c->feed_state = BFS_NONE;
|
|
end = bgp_create_end_mark(c, pkt);
|
|
return bgp_send(conn, PKT_UPDATE, end - buf);
|
|
}
|
|
|
|
else if (c->feed_state == BFS_REFRESHED)
|
|
{
|
|
c->feed_state = BFS_NONE;
|
|
end = bgp_create_end_refresh(c, pkt);
|
|
return bgp_send(conn, PKT_ROUTE_REFRESH, end - buf);
|
|
}
|
|
}
|
|
else if (s)
|
|
bug("Channel packets_to_send: %x", s);
|
|
|
|
c->packets_to_send = 0;
|
|
conn->channels_to_send &= ~(1 << c->index);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* bgp_schedule_packet - schedule a packet for transmission
|
|
* @conn: connection
|
|
* @c: channel
|
|
* @type: packet type
|
|
*
|
|
* Schedule a packet of type @type to be sent as soon as possible.
|
|
*/
|
|
void
|
|
bgp_schedule_packet(struct bgp_conn *conn, struct bgp_channel *c, int type)
|
|
{
|
|
ASSERT(conn->sk);
|
|
|
|
struct bgp_proto *p = conn->bgp;
|
|
if (c)
|
|
BGP_TRACE(D_PACKETS, "Scheduling packet type %d for channel %s", type, c->c.name);
|
|
else
|
|
BGP_TRACE(D_PACKETS, "Scheduling packet type %d", type);
|
|
|
|
if (c)
|
|
{
|
|
if (! conn->channels_to_send)
|
|
{
|
|
conn->last_channel = c->index;
|
|
conn->last_channel_count = 0;
|
|
}
|
|
|
|
c->packets_to_send |= 1 << type;
|
|
conn->channels_to_send |= 1 << c->index;
|
|
}
|
|
else
|
|
conn->packets_to_send |= 1 << type;
|
|
|
|
if ((conn->sk->tpos == conn->sk->tbuf) && !ev_active(conn->tx_ev))
|
|
proto_send_event(&p->p, conn->tx_ev);
|
|
}
|
|
void
|
|
bgp_kick_tx(void *vconn)
|
|
{
|
|
struct bgp_conn *conn = vconn;
|
|
|
|
DBG("BGP: kicking TX\n");
|
|
while (bgp_fire_tx(conn) > 0)
|
|
MAYBE_DEFER_TASK(proto_event_list(&conn->bgp->p), conn->tx_ev,
|
|
"BGP TX for %s", conn->bgp->p.name);
|
|
}
|
|
|
|
void
|
|
bgp_tx(sock *sk)
|
|
{
|
|
struct bgp_conn *conn = sk->data;
|
|
|
|
ASSERT_DIE(birdloop_inside(conn->bgp->p.loop));
|
|
|
|
/* Pending message was passed to kernel */
|
|
if ((conn->state == BS_ESTABLISHED) || (conn->state == BS_OPENCONFIRM))
|
|
bgp_start_timer(conn->bgp, conn->send_hold_timer, conn->send_hold_time);
|
|
|
|
DBG("BGP: TX hook\n");
|
|
while (bgp_fire_tx(conn) > 0)
|
|
MAYBE_DEFER_TASK(proto_event_list(&conn->bgp->p), conn->tx_ev,
|
|
"BGP TX for %s", conn->bgp->p.name);
|
|
}
|
|
|
|
|
|
static struct {
|
|
byte major, minor;
|
|
byte *msg;
|
|
} bgp_msg_table[] = {
|
|
{ 1, 0, "Invalid message header" },
|
|
{ 1, 1, "Connection not synchronized" },
|
|
{ 1, 2, "Bad message length" },
|
|
{ 1, 3, "Bad message type" },
|
|
{ 2, 0, "Invalid OPEN message" },
|
|
{ 2, 1, "Unsupported version number" },
|
|
{ 2, 2, "Bad peer AS" },
|
|
{ 2, 3, "Bad BGP identifier" },
|
|
{ 2, 4, "Unsupported optional parameter" },
|
|
{ 2, 5, "Authentication failure" },
|
|
{ 2, 6, "Unacceptable hold time" },
|
|
{ 2, 7, "Required capability missing" }, /* [RFC5492] */
|
|
{ 2, 8, "No supported AFI/SAFI" }, /* This error msg is nonstandard */
|
|
{ 2,11, "Role mismatch" }, /* From Open Policy, RFC 9234 */
|
|
{ 3, 0, "Invalid UPDATE message" },
|
|
{ 3, 1, "Malformed attribute list" },
|
|
{ 3, 2, "Unrecognized well-known attribute" },
|
|
{ 3, 3, "Missing mandatory attribute" },
|
|
{ 3, 4, "Invalid attribute flags" },
|
|
{ 3, 5, "Invalid attribute length" },
|
|
{ 3, 6, "Invalid ORIGIN attribute" },
|
|
{ 3, 7, "AS routing loop" }, /* Deprecated */
|
|
{ 3, 8, "Invalid NEXT_HOP attribute" },
|
|
{ 3, 9, "Optional attribute error" },
|
|
{ 3, 10, "Invalid network field" },
|
|
{ 3, 11, "Malformed AS_PATH" },
|
|
{ 4, 0, "Hold timer expired" },
|
|
{ 5, 0, "Finite state machine error" }, /* Subcodes are according to [RFC6608] */
|
|
{ 5, 1, "Unexpected message in OpenSent state" },
|
|
{ 5, 2, "Unexpected message in OpenConfirm state" },
|
|
{ 5, 3, "Unexpected message in Established state" },
|
|
{ 6, 0, "Cease" }, /* Subcodes are according to [RFC4486] */
|
|
{ 6, 1, "Maximum number of prefixes reached" },
|
|
{ 6, 2, "Administrative shutdown" },
|
|
{ 6, 3, "Peer de-configured" },
|
|
{ 6, 4, "Administrative reset" },
|
|
{ 6, 5, "Connection rejected" },
|
|
{ 6, 6, "Other configuration change" },
|
|
{ 6, 7, "Connection collision resolution" },
|
|
{ 6, 8, "Out of Resources" },
|
|
{ 7, 0, "Invalid ROUTE-REFRESH message" }, /* [RFC7313] */
|
|
{ 7, 1, "Invalid ROUTE-REFRESH message length" }, /* [RFC7313] */
|
|
{ 8, 0, "Send hold timer expired" }, /* [draft-ietf-idr-bgp-sendholdtimer] */
|
|
};
|
|
|
|
/**
|
|
* bgp_error_dsc - return BGP error description
|
|
* @code: BGP error code
|
|
* @subcode: BGP error subcode
|
|
*
|
|
* bgp_error_dsc() returns error description for BGP errors
|
|
* which might be static string or given temporary buffer.
|
|
*/
|
|
const char *
|
|
bgp_error_dsc(uint code, uint subcode)
|
|
{
|
|
static char buff[32];
|
|
uint i;
|
|
|
|
for (i=0; i < ARRAY_SIZE(bgp_msg_table); i++)
|
|
if (bgp_msg_table[i].major == code && bgp_msg_table[i].minor == subcode)
|
|
return bgp_msg_table[i].msg;
|
|
|
|
bsprintf(buff, "Unknown error %u.%u", code, subcode);
|
|
return buff;
|
|
}
|
|
|
|
/* RFC 8203 - shutdown communication message */
|
|
static int
|
|
bgp_handle_message(struct bgp_proto *p, byte *data, uint len, byte **bp)
|
|
{
|
|
byte *msg = data + 1;
|
|
uint msg_len = data[0];
|
|
uint i;
|
|
|
|
/* Handle zero length message */
|
|
if (msg_len == 0)
|
|
return 1;
|
|
|
|
/* Handle proper message */
|
|
if (msg_len + 1 > len)
|
|
return 0;
|
|
|
|
/* Some elementary cleanup */
|
|
for (i = 0; i < msg_len; i++)
|
|
if (msg[i] < ' ')
|
|
msg[i] = ' ';
|
|
|
|
proto_set_message(&p->p, msg, msg_len);
|
|
*bp += bsprintf(*bp, ": \"%s\"", p->p.message);
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
bgp_log_error(struct bgp_proto *p, u8 class, char *msg, uint code, uint subcode, byte *data, uint len)
|
|
{
|
|
byte argbuf[256+16], *t = argbuf;
|
|
uint i;
|
|
|
|
/* Don't report Cease messages generated by myself */
|
|
if (code == 6 && class == BE_BGP_TX)
|
|
return;
|
|
|
|
/* Reset shutdown message */
|
|
if ((code == 6) && ((subcode == 2) || (subcode == 4)))
|
|
proto_set_message(&p->p, NULL, 0);
|
|
|
|
if (len)
|
|
{
|
|
/* Bad peer AS / unacceptable hold time - print the value as decimal number */
|
|
if ((code == 2) && ((subcode == 2) || (subcode == 6)) && ((len == 2) || (len == 4)))
|
|
{
|
|
t += bsprintf(t, ": %u", (len == 2) ? get_u16(data) : get_u32(data));
|
|
goto done;
|
|
}
|
|
|
|
if ((code == 2) && (subcode == 11) && (len == 1))
|
|
{
|
|
t += bsprintf(t, " (%s)", bgp_format_role_name(get_u8(data)));
|
|
goto done;
|
|
}
|
|
|
|
/* RFC 8203 - shutdown communication */
|
|
if (((code == 6) && ((subcode == 2) || (subcode == 4))))
|
|
if (bgp_handle_message(p, data, len, &t))
|
|
goto done;
|
|
|
|
*t++ = ':';
|
|
*t++ = ' ';
|
|
if (len > 128)
|
|
len = 128;
|
|
for (i=0; i<len; i++)
|
|
t += bsprintf(t, "%02x", data[i]);
|
|
}
|
|
|
|
done:
|
|
*t = 0;
|
|
const byte *dsc = bgp_error_dsc(code, subcode);
|
|
log(L_REMOTE "%s: %s: %s%s", p->p.name, msg, dsc, argbuf);
|
|
}
|
|
|
|
static void
|
|
bgp_rx_notification(struct bgp_conn *conn, byte *pkt, uint len)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
|
|
if (len < 21)
|
|
{ bgp_error(conn, 1, 2, pkt+16, 2); return; }
|
|
|
|
uint code = pkt[19];
|
|
uint subcode = pkt[20];
|
|
int err = (code != 6);
|
|
|
|
bgp_log_error(p, BE_BGP_RX, "Received", code, subcode, pkt+21, len-21);
|
|
bgp_store_error(p, conn, BE_BGP_RX, (code << 16) | subcode);
|
|
|
|
conn->notify_code = code;
|
|
conn->notify_subcode = subcode;
|
|
conn->notify_data = pkt+21;
|
|
conn->notify_size = len-21;
|
|
|
|
bgp_conn_enter_close_state(conn);
|
|
bgp_schedule_packet(conn, NULL, PKT_SCHEDULE_CLOSE);
|
|
|
|
if (err)
|
|
{
|
|
bgp_update_startup_delay(p);
|
|
bgp_stop(p, 0, NULL, 0);
|
|
}
|
|
else
|
|
{
|
|
uint subcode_bit = 1 << ((subcode <= 8) ? subcode : 0);
|
|
if (p->cf->disable_after_cease & subcode_bit)
|
|
{
|
|
log(L_INFO "%s: Disabled after Cease notification", p->p.name);
|
|
p->startup_delay = 0;
|
|
p->p.disabled = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
bgp_rx_keepalive(struct bgp_conn *conn)
|
|
{
|
|
struct bgp_proto *p = conn->bgp;
|
|
|
|
BGP_TRACE(D_PACKETS, "Got KEEPALIVE");
|
|
bgp_start_timer(p, conn->hold_timer, conn->hold_time);
|
|
|
|
if (conn->state == BS_OPENCONFIRM)
|
|
{ bgp_conn_enter_established_state(conn); return; }
|
|
|
|
if (conn->state != BS_ESTABLISHED)
|
|
bgp_error(conn, 5, fsm_err_subcode[conn->state], NULL, 0);
|
|
}
|
|
|
|
|
|
/**
|
|
* bgp_rx_packet - handle a received packet
|
|
* @conn: BGP connection
|
|
* @pkt: start of the packet
|
|
* @len: packet size
|
|
*
|
|
* bgp_rx_packet() takes a newly received packet and calls the corresponding
|
|
* packet handler according to the packet type.
|
|
*/
|
|
static void
|
|
bgp_rx_packet(struct bgp_conn *conn, byte *pkt, uint len)
|
|
{
|
|
byte type = pkt[18];
|
|
|
|
DBG("BGP: Got packet %02x (%d bytes)\n", type, len);
|
|
conn->bgp->stats.rx_messages++;
|
|
conn->bgp->stats.rx_bytes += len;
|
|
|
|
if (conn->bgp->p.mrtdump & MD_MESSAGES)
|
|
bgp_dump_message(conn, pkt, len);
|
|
|
|
switch (type)
|
|
{
|
|
case PKT_OPEN: return bgp_rx_open(conn, pkt, len);
|
|
case PKT_UPDATE: return bgp_rx_update(conn, pkt, len);
|
|
case PKT_NOTIFICATION: return bgp_rx_notification(conn, pkt, len);
|
|
case PKT_KEEPALIVE: return bgp_rx_keepalive(conn);
|
|
case PKT_ROUTE_REFRESH: return bgp_rx_route_refresh(conn, pkt, len);
|
|
default: bgp_error(conn, 1, 3, pkt+18, 1);
|
|
}
|
|
}
|
|
|
|
void
|
|
bgp_do_uncork(void *vp)
|
|
{
|
|
struct bgp_proto *p = vp;
|
|
ASSERT_DIE(birdloop_inside(p->p.loop));
|
|
ASSERT_DIE(p->p.active_loops--);
|
|
|
|
if (p->p.proto_state == PS_DOWN)
|
|
ev_send_loop(p->p.loop, p->p.event);
|
|
else if (p->conn && (p->conn->state == BS_ESTABLISHED) && !p->conn->sk->rx_hook)
|
|
{
|
|
struct birdsock *sk = p->conn->sk;
|
|
ASSERT_DIE(sk->rpos > sk->rbuf);
|
|
sk_resume_rx(p->p.loop, sk, bgp_rx);
|
|
bgp_rx(sk, sk->rpos - sk->rbuf);
|
|
BGP_TRACE(D_PACKETS, "Uncorked");
|
|
}
|
|
}
|
|
|
|
void
|
|
bgp_uncork_main(void *vp)
|
|
{
|
|
/* The uncork event is run from &main_birdloop and there is no useful way how
|
|
* to assign the target loop to it, thus we have to lock it ourselves. */
|
|
|
|
struct bgp_proto *p = vp;
|
|
ev_send_loop(p->p.loop, p->uncork_do_ev);
|
|
}
|
|
|
|
/**
|
|
* bgp_rx - handle received data
|
|
* @sk: socket
|
|
* @size: amount of data received
|
|
*
|
|
* bgp_rx() is called by the socket layer whenever new data arrive from
|
|
* the underlying TCP connection. It assembles the data fragments to packets,
|
|
* checks their headers and framing and passes complete packets to
|
|
* bgp_rx_packet().
|
|
*/
|
|
int
|
|
bgp_rx(sock *sk, uint size)
|
|
{
|
|
struct bgp_conn *conn = sk->data;
|
|
struct bgp_proto *p = conn->bgp;
|
|
byte *pkt_start = sk->rbuf;
|
|
byte *end = pkt_start + size;
|
|
uint i, len;
|
|
|
|
DBG("BGP: RX hook: Got %d bytes\n", size);
|
|
while (end >= pkt_start + BGP_HEADER_LENGTH)
|
|
{
|
|
if ((conn->state == BS_CLOSE) || (conn->sk != sk))
|
|
return 0;
|
|
if ((conn->state == BS_ESTABLISHED) && rt_cork_check(conn->bgp->uncork_main_ev))
|
|
{
|
|
sk_pause_rx(p->p.loop, sk);
|
|
p->p.active_loops++;
|
|
BGP_TRACE(D_PACKETS, "Corked");
|
|
break;
|
|
}
|
|
for(i=0; i<16; i++)
|
|
if (pkt_start[i] != 0xff)
|
|
{
|
|
bgp_error(conn, 1, 1, NULL, 0);
|
|
break;
|
|
}
|
|
len = get_u16(pkt_start+16);
|
|
if ((len < BGP_HEADER_LENGTH) || (len > bgp_max_packet_length(conn)))
|
|
{
|
|
bgp_error(conn, 1, 2, pkt_start+16, 2);
|
|
break;
|
|
}
|
|
if (end < pkt_start + len)
|
|
break;
|
|
bgp_rx_packet(conn, pkt_start, len);
|
|
pkt_start += len;
|
|
}
|
|
if (pkt_start != sk->rbuf)
|
|
{
|
|
memmove(sk->rbuf, pkt_start, end - pkt_start);
|
|
sk->rpos = sk->rbuf + (end - pkt_start);
|
|
}
|
|
return 0;
|
|
}
|