2000-03-20 06:09:07 +08:00
|
|
|
/*
|
|
|
|
* BIRD -- The Border Gateway Protocol
|
|
|
|
*
|
|
|
|
* (c) 2000 Martin Mares <mj@ucw.cz>
|
|
|
|
*
|
|
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _BIRD_BGP_H_
|
|
|
|
#define _BIRD_BGP_H_
|
|
|
|
|
2000-04-12 22:09:26 +08:00
|
|
|
#include "nest/route.h"
|
|
|
|
|
2000-04-01 07:21:37 +08:00
|
|
|
struct linpool;
|
2000-04-01 18:21:11 +08:00
|
|
|
struct eattr;
|
2000-04-01 07:21:37 +08:00
|
|
|
|
2000-03-20 06:09:07 +08:00
|
|
|
struct bgp_config {
|
|
|
|
struct proto_config c;
|
2008-10-27 05:36:08 +08:00
|
|
|
u32 local_as, remote_as;
|
2000-03-20 06:09:07 +08:00
|
|
|
ip_addr remote_ip;
|
|
|
|
int multihop; /* Number of hops if multihop */
|
2000-04-01 07:21:37 +08:00
|
|
|
ip_addr multihop_via; /* Multihop: address to route to */
|
2000-04-21 21:01:28 +08:00
|
|
|
ip_addr source_addr; /* Source address to use */
|
2000-04-10 19:21:40 +08:00
|
|
|
int next_hop_self; /* Always set next hop to local IP address */
|
2000-04-17 20:46:07 +08:00
|
|
|
int compare_path_lengths; /* Use path lengths when selecting best route */
|
|
|
|
u32 default_local_pref; /* Default value for LOCAL_PREF attribute */
|
|
|
|
u32 default_med; /* Default value for MULTI_EXIT_DISC attribute */
|
2008-10-27 05:48:02 +08:00
|
|
|
int enable_as4; /* Enable local support for 4B AS numbers [RFC4893] */
|
2008-10-27 05:45:09 +08:00
|
|
|
u32 rr_cluster_id; /* Route reflector cluster ID, if different from local ID */
|
|
|
|
int rr_client; /* Whether neighbor is RR client of me */
|
2008-11-01 19:55:43 +08:00
|
|
|
int rs_client; /* Whether neighbor is RS client of me */
|
2000-03-30 18:44:20 +08:00
|
|
|
unsigned connect_retry_time;
|
|
|
|
unsigned hold_time, initial_hold_time;
|
|
|
|
unsigned keepalive_time;
|
2000-04-26 07:08:31 +08:00
|
|
|
unsigned start_delay_time; /* Minimum delay between connects */
|
|
|
|
unsigned error_amnesia_time; /* Errors are forgotten after */
|
|
|
|
unsigned error_delay_time_min; /* Time to wait after an error is detected */
|
|
|
|
unsigned error_delay_time_max;
|
|
|
|
unsigned disable_after_error; /* Disable the protocol when error is detected */
|
2008-10-27 05:42:39 +08:00
|
|
|
char *password; /* Password used for MD5 authentication */
|
2000-03-21 05:50:17 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct bgp_conn {
|
|
|
|
struct bgp_proto *bgp;
|
|
|
|
struct birdsock *sk;
|
2000-03-21 23:53:50 +08:00
|
|
|
unsigned int state; /* State of connection state machine */
|
2000-03-21 05:50:17 +08:00
|
|
|
struct timer *connect_retry_timer;
|
|
|
|
struct timer *hold_timer;
|
|
|
|
struct timer *keepalive_timer;
|
2000-03-30 18:44:20 +08:00
|
|
|
int packets_to_send; /* Bitmap of packet types to be sent */
|
2000-04-26 05:13:25 +08:00
|
|
|
int notify_code, notify_subcode, notify_size;
|
|
|
|
byte *notify_data;
|
2000-03-30 18:44:20 +08:00
|
|
|
int error_flag; /* Error state, ignore all input */
|
2000-03-31 01:39:48 +08:00
|
|
|
int primary; /* This connection is primary */
|
2008-10-27 05:36:08 +08:00
|
|
|
u32 advertised_as; /* Temporary value for AS number received */
|
2000-03-30 18:44:20 +08:00
|
|
|
unsigned hold_time, keepalive_time; /* Times calculated from my and neighbor's requirements */
|
2000-03-20 06:09:07 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct bgp_proto {
|
|
|
|
struct proto p;
|
2000-03-21 05:50:17 +08:00
|
|
|
struct bgp_config *cf; /* Shortcut to BGP configuration */
|
2008-10-27 05:36:08 +08:00
|
|
|
u32 local_as, remote_as;
|
2000-03-21 05:50:17 +08:00
|
|
|
int is_internal; /* Internal BGP connection (local_as == remote_as) */
|
2008-10-27 05:36:08 +08:00
|
|
|
int as4_support; /* Peer supports 4B AS numbers [RFC4893] */
|
2008-10-27 05:48:02 +08:00
|
|
|
int as4_session; /* Session uses 4B AS numbers in AS_PATH (both sides support it) */
|
2000-03-21 05:50:17 +08:00
|
|
|
u32 local_id; /* BGP identifier of this router */
|
|
|
|
u32 remote_id; /* BGP identifier of the neighbor */
|
2008-10-27 05:45:09 +08:00
|
|
|
u32 rr_cluster_id; /* Route reflector cluster ID */
|
|
|
|
int rr_client; /* Whether neighbor is RR client of me */
|
2008-11-01 19:55:43 +08:00
|
|
|
int rs_client; /* Whether neighbor is RS client of me */
|
2000-03-31 01:39:48 +08:00
|
|
|
struct bgp_conn *conn; /* Connection we have established */
|
|
|
|
struct bgp_conn outgoing_conn; /* Outgoing connection we're working with */
|
2000-03-21 05:50:17 +08:00
|
|
|
struct bgp_conn incoming_conn; /* Incoming connection we have neither accepted nor rejected yet */
|
|
|
|
struct object_lock *lock; /* Lock for neighbor connection */
|
2000-04-10 19:21:40 +08:00
|
|
|
ip_addr next_hop; /* Either the peer or multihop_via */
|
|
|
|
struct neighbor *neigh; /* Neighbor entry corresponding to next_hop */
|
|
|
|
ip_addr local_addr; /* Address of the local end of the link to next_hop */
|
2000-04-12 22:09:26 +08:00
|
|
|
struct bgp_bucket **bucket_hash; /* Hash table of attribute buckets */
|
2000-04-10 20:39:51 +08:00
|
|
|
unsigned int hash_size, hash_count, hash_limit;
|
2000-04-12 22:09:26 +08:00
|
|
|
struct fib prefix_fib; /* Prefixes to be sent */
|
|
|
|
list bucket_queue; /* Queue of buckets to send */
|
|
|
|
struct bgp_bucket *withdraw_bucket; /* Withdrawn routes */
|
2000-04-26 07:08:31 +08:00
|
|
|
unsigned startup_delay; /* Time to delay protocol startup by due to errors */
|
|
|
|
bird_clock_t last_connect; /* Time of last connect attempt */
|
2000-05-04 17:03:31 +08:00
|
|
|
#ifdef IPV6
|
|
|
|
byte *mp_reach_start, *mp_unreach_start; /* Multiprotocol BGP attribute notes */
|
|
|
|
unsigned mp_reach_len, mp_unreach_len;
|
2003-02-23 06:47:45 +08:00
|
|
|
ip_addr local_link; /* Link-level version of local_addr */
|
2000-05-04 17:03:31 +08:00
|
|
|
#endif
|
2000-04-12 22:09:26 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct bgp_prefix {
|
|
|
|
struct fib_node n; /* Node in prefix fib */
|
|
|
|
node bucket_node; /* Node in per-bucket list */
|
|
|
|
};
|
|
|
|
|
|
|
|
struct bgp_bucket {
|
|
|
|
node send_node; /* Node in send queue */
|
2000-04-17 17:37:31 +08:00
|
|
|
struct bgp_bucket *hash_next, *hash_prev; /* Node in bucket hash table */
|
2000-04-12 22:09:26 +08:00
|
|
|
unsigned hash; /* Hash over extended attributes */
|
|
|
|
list prefixes; /* Prefixes in this buckets */
|
|
|
|
ea_list eattrs[0]; /* Per-bucket extended attributes */
|
2000-03-20 06:09:07 +08:00
|
|
|
};
|
|
|
|
|
2000-03-21 23:53:50 +08:00
|
|
|
#define BGP_PORT 179
|
|
|
|
#define BGP_VERSION 4
|
|
|
|
#define BGP_HEADER_LENGTH 19
|
|
|
|
#define BGP_MAX_PACKET_LENGTH 4096
|
|
|
|
#define BGP_RX_BUFFER_SIZE 4096
|
|
|
|
#define BGP_TX_BUFFER_SIZE BGP_MAX_PACKET_LENGTH
|
2000-03-20 06:09:07 +08:00
|
|
|
|
2000-03-31 02:44:23 +08:00
|
|
|
extern struct linpool *bgp_linpool;
|
|
|
|
|
2008-10-27 05:36:08 +08:00
|
|
|
extern int bgp_as4_support;
|
|
|
|
|
|
|
|
|
2000-03-30 18:44:20 +08:00
|
|
|
void bgp_start_timer(struct timer *t, int value);
|
2000-03-20 06:09:07 +08:00
|
|
|
void bgp_check(struct bgp_config *c);
|
2000-04-26 05:13:25 +08:00
|
|
|
void bgp_error(struct bgp_conn *c, unsigned code, unsigned subcode, byte *data, int len);
|
2000-03-31 01:39:48 +08:00
|
|
|
void bgp_close_conn(struct bgp_conn *c);
|
2000-03-20 06:09:07 +08:00
|
|
|
|
2000-05-03 00:07:41 +08:00
|
|
|
#ifdef LOCAL_DEBUG
|
|
|
|
#define BGP_FORCE_DEBUG 1
|
|
|
|
#else
|
|
|
|
#define BGP_FORCE_DEBUG 0
|
|
|
|
#endif
|
|
|
|
#define BGP_TRACE(flags, msg, args...) do { if ((p->p.debug & flags) || BGP_FORCE_DEBUG) \
|
|
|
|
log(L_TRACE "%s: " msg, p->p.name , ## args ); } while(0)
|
|
|
|
|
2000-03-21 05:50:17 +08:00
|
|
|
/* attrs.c */
|
|
|
|
|
2008-10-27 05:45:09 +08:00
|
|
|
void bgp_attach_attr(struct ea_list **to, struct linpool *pool, unsigned attr, uintptr_t val);
|
|
|
|
byte *bgp_attach_attr_wa(struct ea_list **to, struct linpool *pool, unsigned attr, unsigned len);
|
2000-04-21 20:25:35 +08:00
|
|
|
struct rta *bgp_decode_attrs(struct bgp_conn *conn, byte *a, unsigned int len, struct linpool *pool, int mandatory);
|
2000-04-01 18:21:11 +08:00
|
|
|
int bgp_get_attr(struct eattr *e, byte *buf);
|
2000-04-10 06:05:02 +08:00
|
|
|
int bgp_rte_better(struct rte *, struct rte *);
|
|
|
|
void bgp_rt_notify(struct proto *, struct network *, struct rte *, struct rte *, struct ea_list *);
|
|
|
|
int bgp_import_control(struct proto *, struct rte **, struct ea_list **, struct linpool *);
|
2000-04-10 20:39:51 +08:00
|
|
|
void bgp_attr_init(struct bgp_proto *);
|
2008-10-27 05:36:08 +08:00
|
|
|
unsigned int bgp_encode_attrs(struct bgp_proto *p, byte *w, ea_list *attrs, int remains);
|
2000-04-17 17:37:31 +08:00
|
|
|
void bgp_free_bucket(struct bgp_proto *p, struct bgp_bucket *buck);
|
2000-05-19 19:01:41 +08:00
|
|
|
void bgp_get_route_info(struct rte *, byte *buf, struct ea_list *attrs);
|
2000-04-01 07:21:37 +08:00
|
|
|
|
2008-10-27 05:45:09 +08:00
|
|
|
inline static void bgp_attach_attr_ip(struct ea_list **to, struct linpool *pool, unsigned attr, ip_addr a)
|
|
|
|
{ *(ip_addr *) bgp_attach_attr_wa(to, pool, attr, sizeof(ip_addr)) = a; }
|
|
|
|
|
2000-03-21 05:50:17 +08:00
|
|
|
/* packets.c */
|
|
|
|
|
2000-03-21 23:53:50 +08:00
|
|
|
void bgp_schedule_packet(struct bgp_conn *conn, int type);
|
|
|
|
void bgp_tx(struct birdsock *sk);
|
|
|
|
int bgp_rx(struct birdsock *sk, int size);
|
2000-04-26 05:13:25 +08:00
|
|
|
void bgp_log_error(struct bgp_proto *p, char *msg, unsigned code, unsigned subcode, byte *data, unsigned len);
|
2000-03-21 23:53:50 +08:00
|
|
|
|
2000-03-21 05:50:17 +08:00
|
|
|
/* Packet types */
|
|
|
|
|
|
|
|
#define PKT_OPEN 0x01
|
|
|
|
#define PKT_UPDATE 0x02
|
|
|
|
#define PKT_NOTIFICATION 0x03
|
|
|
|
#define PKT_KEEPALIVE 0x04
|
2000-03-21 23:53:50 +08:00
|
|
|
#define PKT_SCHEDULE_CLOSE 0x1f /* Used internally to schedule socket close */
|
2000-03-21 05:50:17 +08:00
|
|
|
|
|
|
|
/* Attributes */
|
|
|
|
|
|
|
|
#define BAF_OPTIONAL 0x80
|
|
|
|
#define BAF_TRANSITIVE 0x40
|
|
|
|
#define BAF_PARTIAL 0x20
|
|
|
|
#define BAF_EXT_LEN 0x10
|
|
|
|
|
|
|
|
#define BA_ORIGIN 0x01 /* [RFC1771] */ /* WM */
|
|
|
|
#define BA_AS_PATH 0x02 /* WM */
|
|
|
|
#define BA_NEXT_HOP 0x03 /* WM */
|
|
|
|
#define BA_MULTI_EXIT_DISC 0x04 /* ON */
|
2000-04-01 07:21:37 +08:00
|
|
|
#define BA_LOCAL_PREF 0x05 /* WD */
|
2000-03-21 05:50:17 +08:00
|
|
|
#define BA_ATOMIC_AGGR 0x06 /* WD */
|
|
|
|
#define BA_AGGREGATOR 0x07 /* OT */
|
|
|
|
#define BA_COMMUNITY 0x08 /* [RFC1997] */ /* OT */
|
|
|
|
#define BA_ORIGINATOR_ID 0x09 /* [RFC1966] */ /* ON */
|
|
|
|
#define BA_CLUSTER_LIST 0x0a /* ON */
|
|
|
|
/* We don't support these: */
|
|
|
|
#define BA_DPA 0x0b /* ??? */
|
|
|
|
#define BA_ADVERTISER 0x0c /* [RFC1863] */
|
|
|
|
#define BA_RCID_PATH 0x0d
|
|
|
|
#define BA_MP_REACH_NLRI 0x0e /* [RFC2283] */
|
|
|
|
#define BA_MP_UNREACH_NLRI 0x0f
|
|
|
|
#define BA_EXTENDED_COMM 0x10 /* draft-ramachandra-bgp-ext-communities */
|
2008-10-27 05:36:08 +08:00
|
|
|
#define BA_AS4_PATH 0x11 /* [RFC4893] */
|
|
|
|
#define BA_AS4_AGGREGATOR 0x12
|
2000-03-21 05:50:17 +08:00
|
|
|
|
|
|
|
/* BGP states */
|
|
|
|
|
|
|
|
#define BS_IDLE 0
|
|
|
|
#define BS_CONNECT 1 /* Attempting to connect */
|
|
|
|
#define BS_ACTIVE 2 /* Waiting for connection retry & listening */
|
|
|
|
#define BS_OPENSENT 3
|
|
|
|
#define BS_OPENCONFIRM 4
|
|
|
|
#define BS_ESTABLISHED 5
|
|
|
|
|
2000-04-17 21:13:08 +08:00
|
|
|
/* Well-known communities */
|
|
|
|
|
|
|
|
#define BGP_COMM_NO_EXPORT 0xffffff01 /* Don't export outside local AS / confed. */
|
|
|
|
#define BGP_COMM_NO_ADVERTISE 0xffffff02 /* Don't export at all */
|
|
|
|
#define BGP_COMM_NO_EXPORT_SUBCONFED 0xffffff03 /* NO_EXPORT even in local confederation */
|
|
|
|
|
2000-04-28 23:13:29 +08:00
|
|
|
/* Origins */
|
|
|
|
|
|
|
|
#define ORIGIN_IGP 0
|
|
|
|
#define ORIGIN_EGP 1
|
|
|
|
#define ORIGIN_INCOMPLETE 2
|
|
|
|
|
2000-05-04 17:03:31 +08:00
|
|
|
/* Address families */
|
|
|
|
|
|
|
|
#define BGP_AF_IPV6 2
|
|
|
|
|
2000-03-20 06:09:07 +08:00
|
|
|
#endif
|