mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-15 05:21:54 +00:00
e0ed978e75
On BSD, the onlink flag is not tracked or reported by kernel. We are using an heuristic that assigns the onlink flag to routes scanned from the kernel. We should use the same heuristic even in BSD-Netlink case, as the onlink flag is not reported here too. Thanks to Björn König for the original patch.
168 lines
4.8 KiB
C
168 lines
4.8 KiB
C
/*
|
|
* BIRD -- UNIX Kernel Route Syncer
|
|
*
|
|
* (c) 1998--2000 Martin Mares <mj@ucw.cz>
|
|
*
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
*/
|
|
|
|
#ifndef _BIRD_KRT_H_
|
|
#define _BIRD_KRT_H_
|
|
|
|
struct config;
|
|
struct krt_config;
|
|
struct krt_proto;
|
|
struct kif_config;
|
|
struct kif_proto;
|
|
|
|
#include "nest/iface.h"
|
|
#include "sysdep/config.h"
|
|
#include CONFIG_INCLUDE_KRTSYS_H
|
|
|
|
#define KRT_DEFAULT_ECMP_LIMIT 16
|
|
|
|
#define EA_KRT_SOURCE EA_CODE(PROTOCOL_KERNEL, 0)
|
|
#define EA_KRT_METRIC EA_CODE(PROTOCOL_KERNEL, 1)
|
|
|
|
#define KRT_REF_SEEN 0x1 /* Seen in table */
|
|
#define KRT_REF_BEST 0x2 /* Best in table */
|
|
|
|
#define KRT_LEARN_NONE 0 /* Do not learn */
|
|
#define KRT_LEARN_ALIEN 1 /* Learn KRT_SRC_ALIEN routes */
|
|
#define KRT_LEARN_ALL 2 /* Learn both KRT_SRC_ALIEN and KRT_SRC_KERNEL routes */
|
|
|
|
/* Whenever we recognize our own routes, we allow learing of foreign routes */
|
|
|
|
#ifdef CONFIG_SELF_CONSCIOUS
|
|
#define KRT_ALLOW_LEARN
|
|
#endif
|
|
|
|
/* krt.c */
|
|
|
|
extern struct protocol proto_unix_kernel;
|
|
|
|
struct krt_config {
|
|
struct proto_config c;
|
|
struct krt_params sys; /* Sysdep params */
|
|
btime scan_time; /* How often we re-scan routes */
|
|
int persist; /* Keep routes when we exit */
|
|
int learn; /* Learn routes from other sources */
|
|
int graceful_restart; /* Regard graceful restart recovery */
|
|
int merge_paths; /* Exported routes are merged for ECMP */
|
|
};
|
|
|
|
struct krt_proto {
|
|
struct proto p;
|
|
struct krt_state sys; /* Sysdep state */
|
|
|
|
#ifdef KRT_ALLOW_LEARN
|
|
struct rtable *krt_table; /* Internal table of inherited routes */
|
|
#endif
|
|
|
|
timer *scan_timer;
|
|
struct bmap sync_map; /* Keeps track which exported routes were successfully written to kernel */
|
|
struct bmap seen_map; /* Routes seen during last periodic scan */
|
|
node krt_node; /* Node in krt_proto_list */
|
|
byte af; /* Kernel address family (AF_*) */
|
|
byte ready; /* Initial feed has been finished */
|
|
byte initialized; /* First scan has been finished */
|
|
byte reload; /* Next scan is doing reload */
|
|
};
|
|
|
|
extern pool *krt_pool;
|
|
|
|
#define KRT_CF ((struct krt_config *)p->p.cf)
|
|
|
|
#define KRT_TRACE(pr, fl, msg, args...) do { \
|
|
DBG("KRT: " msg "\n" , ## args); \
|
|
if (pr->p.debug & fl) \
|
|
{ log(L_TRACE "%s: " msg, pr->p.name , ## args); } } while(0)
|
|
|
|
struct proto_config * kif_init_config(int class);
|
|
void kif_request_scan(void);
|
|
void krt_use_shared_scan(void);
|
|
void krt_got_route(struct krt_proto *p, struct rte *e, s8 src);
|
|
void krt_got_route_async(struct krt_proto *p, struct rte *e, int new, s8 src);
|
|
int krt_assume_onlink(struct iface *iface, int ipv6);
|
|
|
|
static inline int
|
|
krt_get_sync_error(struct krt_proto *p, struct rte *e)
|
|
{
|
|
return (p->p.proto_state == PS_UP) &&
|
|
bmap_test(&p->p.main_channel->export_map, e->id) &&
|
|
!bmap_test(&p->sync_map, e->id);
|
|
}
|
|
|
|
/* Values for rte->u.krt_sync.src */
|
|
#define KRT_SRC_UNKNOWN -1 /* Nobody knows */
|
|
#define KRT_SRC_BIRD 0 /* Our route (not passed in async mode) */
|
|
#define KRT_SRC_REDIRECT 1 /* Redirect route, delete it */
|
|
#define KRT_SRC_ALIEN 2 /* Route installed by someone else */
|
|
#define KRT_SRC_KERNEL 3 /* Kernel routes, are ignored by krt syncer */
|
|
|
|
extern struct protocol proto_unix_iface;
|
|
|
|
struct kif_config {
|
|
struct proto_config c;
|
|
struct kif_params sys; /* Sysdep params */
|
|
|
|
list iface_list; /* List of iface configs (struct kif_iface_config) */
|
|
btime scan_time; /* How often we re-scan interfaces */
|
|
};
|
|
|
|
struct kif_iface_config {
|
|
struct iface_patt i;
|
|
|
|
ip_addr pref_v4;
|
|
ip_addr pref_v6;
|
|
ip_addr pref_ll;
|
|
};
|
|
|
|
struct kif_proto {
|
|
struct proto p;
|
|
struct kif_state sys; /* Sysdep state */
|
|
};
|
|
|
|
extern struct kif_proto *kif_proto;
|
|
|
|
#define KIF_CF ((struct kif_config *)p->p.cf)
|
|
|
|
struct kif_iface_config * kif_get_iface_config(struct iface *iface);
|
|
struct proto_config * krt_init_config(int class);
|
|
|
|
|
|
/* krt sysdep */
|
|
|
|
void krt_sys_io_init(void);
|
|
void krt_sys_init(struct krt_proto *);
|
|
int krt_sys_start(struct krt_proto *);
|
|
void krt_sys_shutdown(struct krt_proto *);
|
|
int krt_sys_reconfigure(struct krt_proto *p UNUSED, struct krt_config *n, struct krt_config *o);
|
|
|
|
void krt_sys_preconfig(struct config *);
|
|
void krt_sys_postconfig(struct krt_config *);
|
|
void krt_sys_init_config(struct krt_config *);
|
|
void krt_sys_copy_config(struct krt_config *, struct krt_config *);
|
|
|
|
int krt_capable(rte *e);
|
|
void krt_do_scan(struct krt_proto *);
|
|
void krt_replace_rte(struct krt_proto *p, net *n, rte *new, rte *old);
|
|
int krt_sys_get_attr(const eattr *a, byte *buf, int buflen);
|
|
|
|
|
|
/* kif sysdep */
|
|
|
|
void kif_sys_init(struct kif_proto *);
|
|
void kif_sys_start(struct kif_proto *);
|
|
void kif_sys_shutdown(struct kif_proto *);
|
|
int kif_sys_reconfigure(struct kif_proto *, struct kif_config *, struct kif_config *);
|
|
|
|
void kif_sys_init_config(struct kif_config *);
|
|
void kif_sys_copy_config(struct kif_config *, struct kif_config *);
|
|
|
|
void kif_do_scan(struct kif_proto *);
|
|
|
|
int kif_update_sysdep_addr(struct iface *i);
|
|
|
|
#endif
|