mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-03 23:51:54 +00:00
493 lines
12 KiB
Plaintext
493 lines
12 KiB
Plaintext
/*
|
|
* BIRD -- Configuration Parser Top
|
|
*
|
|
* (c) 1998--2000 Martin Mares <mj@ucw.cz>
|
|
*
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
*/
|
|
|
|
CF_HDR
|
|
|
|
#define PARSER 1
|
|
|
|
#include "nest/bird.h"
|
|
#include "conf/conf.h"
|
|
#include "lib/resource.h"
|
|
#include "lib/socket.h"
|
|
#include "lib/settle.h"
|
|
#include "lib/timer.h"
|
|
#include "lib/string.h"
|
|
#include "nest/protocol.h"
|
|
#include "nest/iface.h"
|
|
#include "nest/route.h"
|
|
#include "nest/bfd.h"
|
|
#include "nest/cli.h"
|
|
#include "filter/filter.h"
|
|
|
|
/* FIXME: Turn on YYERROR_VERBOSE and work around lots of bison bugs? */
|
|
|
|
CF_DEFINES
|
|
|
|
static bool this_sadr_from_hack_active;
|
|
|
|
static void
|
|
check_u16(uint val)
|
|
{
|
|
if (val > 0xFFFF)
|
|
cf_error("Value %u out of range (0-65535)", val);
|
|
}
|
|
|
|
#define cf_assert(cond, ...) do { if (!(cond)) cf_error(__VA_ARGS__); } while (0)
|
|
static inline void cf_assert_symbol(const struct symbol *sym, uint class) {
|
|
switch (class) {
|
|
case SYM_PROTO: cf_assert(sym->class == SYM_PROTO, "Protocol name required"); break;
|
|
case SYM_TEMPLATE: cf_assert(sym->class == SYM_TEMPLATE, "Protocol template name required"); break;
|
|
case SYM_FUNCTION: cf_assert(sym->class == SYM_FUNCTION, "Function name required"); break;
|
|
case SYM_FILTER: cf_assert(sym->class == SYM_FILTER, "Filter name required"); break;
|
|
case SYM_TABLE: cf_assert(sym->class == SYM_TABLE, "Table name required"); break;
|
|
case SYM_ATTRIBUTE: cf_assert(sym->class == SYM_ATTRIBUTE, "Custom attribute name required"); break;
|
|
case SYM_MPLS_DOMAIN: cf_assert(sym->class == SYM_MPLS_DOMAIN, "MPLS domain name required"); break;
|
|
case SYM_MPLS_RANGE: cf_assert(sym->class == SYM_MPLS_RANGE, "MPLS range name required"); break;
|
|
case SYM_VARIABLE: cf_assert((sym->class & ~0xff) == SYM_VARIABLE, "Variable name required"); break;
|
|
case SYM_CONSTANT: cf_assert((sym->class & ~0xff) == SYM_CONSTANT, "Constant name required"); break;
|
|
default: bug("This shall not happen");
|
|
}
|
|
}
|
|
|
|
CF_DECLS
|
|
|
|
%union {
|
|
uint i;
|
|
u32 i32;
|
|
u64 i64;
|
|
ip_addr a;
|
|
ip4_addr ip4;
|
|
ip6_addr ip6;
|
|
net_addr net;
|
|
net_addr *net_ptr;
|
|
struct symbol *s;
|
|
const char *t;
|
|
struct rtable_config *r;
|
|
struct channel_config *cc;
|
|
struct channel *c;
|
|
struct f_inst *x;
|
|
struct {
|
|
struct f_inst *begin, *end;
|
|
} xp;
|
|
enum filter_return fret;
|
|
enum ec_subtype ecs;
|
|
struct ea_class *ea_class;
|
|
struct f_static_attr fsa;
|
|
struct f_attr_bit fab;
|
|
struct f_lval flv;
|
|
struct f_line *fl;
|
|
struct f_arg *fa;
|
|
const struct filter *f;
|
|
struct f_tree *e;
|
|
struct f_trie *trie;
|
|
const struct f_trie *const_trie;
|
|
struct f_val v;
|
|
struct password_item *p;
|
|
struct rt_show_data *ra;
|
|
struct sym_show_data *sd;
|
|
struct lsadb_show_data *ld;
|
|
struct mrt_dump_data *md;
|
|
struct mpls_show_ranges_cmd *msrc;
|
|
struct bfd_show_sessions_cmd *bssc;
|
|
struct iface *iface;
|
|
void *g;
|
|
btime time;
|
|
struct f_prefix px;
|
|
struct proto_spec ps;
|
|
struct channel_limit cl;
|
|
struct timeformat tf;
|
|
struct timeformat *tfp;
|
|
struct settle_config settle;
|
|
struct adata *ad;
|
|
const struct adata *bs;
|
|
struct aggr_item_node *ai;
|
|
}
|
|
|
|
%token END CLI_MARKER INVALID_TOKEN ELSECOL DDOT
|
|
%token GEQ LEQ NEQ AND OR IMP
|
|
%token PO PC
|
|
%token <i> NUM ENUM_TOKEN
|
|
%token <ip4> IP4
|
|
%token <ip6> IP6
|
|
%token <i64> VPN_RD
|
|
%token <s> CF_SYM_KNOWN CF_SYM_UNDEFINED CF_SYM_METHOD_BARE CF_SYM_METHOD_ARGS
|
|
%token <t> TEXT
|
|
%token <bs> BYTETEXT
|
|
%type <iface> ipa_scope
|
|
|
|
%type <i> expr bool pxlen4
|
|
%type <time> expr_us time
|
|
%type <settle> settle
|
|
%type <a> ipa net_ip6_slash
|
|
%type <net> net_ip4_ net_ip4 net_ip6_ net_ip6 net_ip_ net_ip net_or_ipa
|
|
%type <net_ptr> net_ net_any net_vpn4_ net_vpn6_ net_vpn_ net_roa4_ net_roa6_ net_roa_ net_ip6_sadr_ net_mpls_ net_aspa_
|
|
%type <ad> label_stack_start label_stack
|
|
|
|
%type <t> text opttext
|
|
%type <bs> bytestring
|
|
%type <s> symbol
|
|
|
|
%type <v> bytestring_text text_or_ipa
|
|
%type <x> bytestring_expr
|
|
|
|
%nonassoc PREFIX_DUMMY
|
|
%left AND OR
|
|
%nonassoc '=' '<' '>' '~' GEQ LEQ NEQ NMA IMP PO PC
|
|
%left '|' '&'
|
|
%left '+' '-'
|
|
%left '*' '/' '%'
|
|
%left '!'
|
|
%nonassoc '.'
|
|
|
|
%start config
|
|
|
|
CF_KEYWORDS(DEFINE, ON, OFF, YES, NO, S, MS, US, PORT, VPN, MPLS, FROM, MAX, AS)
|
|
|
|
CF_GRAMMAR
|
|
|
|
/* Basic config file structure */
|
|
|
|
config: conf_entries END { return 0; }
|
|
| CLI_MARKER cli_cmd { return 0; }
|
|
;
|
|
|
|
conf_entries:
|
|
/* EMPTY */
|
|
| conf_entries conf
|
|
;
|
|
|
|
conf: ';' ;
|
|
|
|
|
|
/* Constant expressions */
|
|
|
|
conf: definition ;
|
|
|
|
definition:
|
|
DEFINE symbol '=' term ';' {
|
|
struct f_val *val = cf_eval($4, T_VOID);
|
|
cf_define_symbol(new_config, $2, SYM_CONSTANT | val->type, val, val);
|
|
}
|
|
;
|
|
|
|
expr:
|
|
NUM
|
|
| '(' term ')' { $$ = cf_eval_int($2); }
|
|
| CF_SYM_KNOWN {
|
|
if ($1->class != (SYM_CONSTANT | T_INT)) cf_error("Number constant expected");
|
|
$$ = SYM_VAL($1).i; }
|
|
;
|
|
|
|
expr_us:
|
|
expr S { $$ = $1 S_; }
|
|
| expr MS { $$ = $1 MS_; }
|
|
| expr US { $$ = $1 US_; }
|
|
;
|
|
|
|
symbol: CF_SYM_UNDEFINED | CF_SYM_KNOWN ;
|
|
|
|
/* Switches */
|
|
|
|
bool:
|
|
expr { $$ = !!$1; }
|
|
| ON { $$ = 1; }
|
|
| YES { $$ = 1; }
|
|
| OFF { $$ = 0; }
|
|
| NO { $$ = 0; }
|
|
| /* Silence means agreement */ { $$ = 1; }
|
|
;
|
|
|
|
|
|
/* Addresses */
|
|
|
|
ipa:
|
|
IP4 { $$ = ipa_from_ip4($1); }
|
|
| IP6 { $$ = ipa_from_ip6($1); }
|
|
| CF_SYM_KNOWN {
|
|
if ($1->class != (SYM_CONSTANT | T_IP)) cf_error("IP address constant expected");
|
|
$$ = SYM_VAL($1).ip;
|
|
}
|
|
;
|
|
|
|
ipa_scope:
|
|
/* empty */ { $$ = NULL; }
|
|
| '%' symbol { $$ = if_get_by_name($2->name); }
|
|
;
|
|
|
|
|
|
/* Networks - internal */
|
|
|
|
pxlen4:
|
|
'/' NUM {
|
|
if ($2 > IP4_MAX_PREFIX_LENGTH) cf_error("Invalid prefix length %u", $2);
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
net_ip4_: IP4 pxlen4
|
|
{
|
|
net_fill_ip4(&($$), $1, $2);
|
|
|
|
net_addr_ip4 *n = (void *) &($$);
|
|
if (!net_validate_ip4(n))
|
|
cf_error("Invalid IPv4 prefix %I4/%d, maybe you wanted %I4/%d",
|
|
n->prefix, n->pxlen, ip4_and(n->prefix, ip4_mkmask(n->pxlen)), n->pxlen);
|
|
};
|
|
|
|
net_ip6_slash: IP6 '/'
|
|
{
|
|
this_sadr_from_hack_active = cf_maybe_exit_filters();
|
|
$$ = $1;
|
|
}
|
|
|
|
net_ip6_: net_ip6_slash NUM
|
|
{
|
|
if (this_sadr_from_hack_active)
|
|
{
|
|
cf_enter_filters();
|
|
this_sadr_from_hack_active = 0;
|
|
}
|
|
|
|
if ($2 > IP6_MAX_PREFIX_LENGTH)
|
|
cf_error("Invalid prefix length %u", $2);
|
|
|
|
net_fill_ip6(&($$), $1, $2);
|
|
|
|
net_addr_ip6 *n = (void *) &($$);
|
|
if (!net_validate_ip6(n))
|
|
cf_error("Invalid IPv6 prefix %I6/%d, maybe you wanted %I6/%d",
|
|
n->prefix, n->pxlen, ip6_and(n->prefix, ip6_mkmask(n->pxlen)), n->pxlen);
|
|
};
|
|
|
|
net_ip6_sadr_: net_ip6_slash NUM FROM IP6 '/' NUM
|
|
{
|
|
if (this_sadr_from_hack_active)
|
|
{
|
|
cf_enter_filters();
|
|
this_sadr_from_hack_active = 0;
|
|
}
|
|
|
|
if (($3->class != SYM_KEYWORD) || ($3->keyword->value != FROM))
|
|
cf_error("Expected FROM after %I6/%d", $1, $2);
|
|
|
|
if ($2 > IP6_MAX_PREFIX_LENGTH)
|
|
cf_error("Invalid prefix length %u", $2);
|
|
|
|
if ($6 > IP6_MAX_PREFIX_LENGTH)
|
|
cf_error("Invalid prefix length %u", $6);
|
|
|
|
$$ = cfg_alloc(sizeof(net_addr_ip6_sadr));
|
|
net_fill_ip6_sadr($$, $1, $2, $4, $6);
|
|
|
|
net_addr_ip6_sadr *n = (void *) $$;
|
|
if (!net_validate_ip6_sadr(n))
|
|
cf_error("Invalid SADR IPv6 prefix %I6/%d from %I6/%d, maybe you wanted %I6/%d from %I6/%d",
|
|
n->dst_prefix, n->dst_pxlen, n->src_prefix, n->src_pxlen,
|
|
ip6_and(n->dst_prefix, ip6_mkmask(n->dst_pxlen)), n->dst_pxlen,
|
|
ip6_and(n->src_prefix, ip6_mkmask(n->src_pxlen)), n->src_pxlen);
|
|
};
|
|
|
|
net_vpn4_: VPN_RD net_ip4_
|
|
{
|
|
$$ = cfg_alloc(sizeof(net_addr_vpn4));
|
|
net_fill_vpn4($$, net4_prefix(&$2), net4_pxlen(&$2), $1);
|
|
}
|
|
|
|
net_vpn6_: VPN_RD net_ip6_
|
|
{
|
|
$$ = cfg_alloc(sizeof(net_addr_vpn6));
|
|
net_fill_vpn6($$, net6_prefix(&$2), net6_pxlen(&$2), $1);
|
|
}
|
|
|
|
net_roa4_: net_ip4_ MAX NUM AS NUM
|
|
{
|
|
$$ = cfg_alloc(sizeof(net_addr_roa4));
|
|
net_fill_roa4($$, net4_prefix(&$1), net4_pxlen(&$1), $3, $5);
|
|
if ($3 < net4_pxlen(&$1) || $3 > IP4_MAX_PREFIX_LENGTH)
|
|
cf_error("Invalid max prefix length %u", $3);
|
|
};
|
|
|
|
net_roa6_: net_ip6_ MAX NUM AS NUM
|
|
{
|
|
$$ = cfg_alloc(sizeof(net_addr_roa6));
|
|
net_fill_roa6($$, net6_prefix(&$1), net6_pxlen(&$1), $3, $5);
|
|
if ($3 < net6_pxlen(&$1) || $3 > IP6_MAX_PREFIX_LENGTH)
|
|
cf_error("Invalid max prefix length %u", $3);
|
|
};
|
|
|
|
net_mpls_: MPLS NUM
|
|
{
|
|
$$ = cfg_alloc(sizeof(net_addr_mpls));
|
|
net_fill_mpls($$, $2);
|
|
}
|
|
|
|
net_aspa_: ASPA NUM
|
|
{
|
|
$$ = cfg_alloc(sizeof(net_addr_aspa));
|
|
net_fill_aspa($$, $2);
|
|
}
|
|
|
|
net_ip_: net_ip4_ | net_ip6_ ;
|
|
net_vpn_: net_vpn4_ | net_vpn6_ ;
|
|
net_roa_: net_roa4_ | net_roa6_ ;
|
|
|
|
net_:
|
|
net_ip_ { $$ = cfg_alloc($1.length); net_copy($$, &($1)); }
|
|
| net_vpn_
|
|
| net_roa_
|
|
| net_flow_
|
|
| net_ip6_sadr_
|
|
| net_mpls_
|
|
| net_aspa_
|
|
;
|
|
|
|
|
|
/* Networks - regular */
|
|
|
|
net_ip4:
|
|
net_ip4_
|
|
| CF_SYM_KNOWN {
|
|
if (($1->class != (SYM_CONSTANT | T_NET)) || (SYM_VAL($1).net->type != NET_IP4))
|
|
cf_error("IPv4 network constant expected");
|
|
$$ = * SYM_VAL($1).net;
|
|
}
|
|
;
|
|
|
|
net_ip6:
|
|
net_ip6_
|
|
| CF_SYM_KNOWN {
|
|
if (($1->class != (SYM_CONSTANT | T_NET)) || (SYM_VAL($1).net->type != NET_IP6))
|
|
cf_error("IPv6 network constant expected");
|
|
$$ = * SYM_VAL($1).net;
|
|
}
|
|
;
|
|
|
|
net_ip:
|
|
net_ip_
|
|
| CF_SYM_KNOWN {
|
|
if (($1->class != (SYM_CONSTANT | T_NET)) || !net_is_ip(SYM_VAL($1).net))
|
|
cf_error("IP network constant expected");
|
|
$$ = * SYM_VAL($1).net;
|
|
}
|
|
;
|
|
|
|
net_any:
|
|
net_
|
|
| CF_SYM_KNOWN {
|
|
if ($1->class != (SYM_CONSTANT | T_NET))
|
|
cf_error("Network constant expected");
|
|
$$ = (net_addr *) SYM_VAL($1).net; /* Avoid const warning */
|
|
}
|
|
;
|
|
|
|
net_or_ipa:
|
|
net_ip4_
|
|
| net_ip6_
|
|
| IP4 { net_fill_ip4(&($$), $1, IP4_MAX_PREFIX_LENGTH); }
|
|
| IP6 { net_fill_ip6(&($$), $1, IP6_MAX_PREFIX_LENGTH); }
|
|
| CF_SYM_KNOWN {
|
|
if ($1->class == (SYM_CONSTANT | T_IP))
|
|
net_fill_ip_host(&($$), SYM_VAL($1).ip);
|
|
else if (($1->class == (SYM_CONSTANT | T_NET)) && net_is_ip(SYM_VAL($1).net))
|
|
$$ = * SYM_VAL($1).net;
|
|
else
|
|
cf_error("IP address or network constant expected");
|
|
}
|
|
;
|
|
|
|
label_stack_start: NUM
|
|
{
|
|
$$ = cfg_allocz(ADATA_SIZE(MPLS_MAX_LABEL_STACK * sizeof(u32)));
|
|
$$->length = sizeof(u32);
|
|
*((u32 *)$$->data) = $1;
|
|
};
|
|
|
|
label_stack:
|
|
label_stack_start
|
|
| label_stack '/' NUM {
|
|
if ($1->length >= MPLS_MAX_LABEL_STACK * sizeof(u32))
|
|
cf_error("Too many labels in stack");
|
|
|
|
*((u32 *)($$->data + $1->length)) = $3;
|
|
$1->length += sizeof(u32);
|
|
$$ = $1;
|
|
}
|
|
;
|
|
|
|
time:
|
|
TEXT {
|
|
$$ = tm_parse_time($1);
|
|
if (!$$)
|
|
cf_error("Invalid date/time");
|
|
}
|
|
;
|
|
|
|
/* Settle timer configuration */
|
|
settle: expr_us expr_us {
|
|
if ($1 > $2) cf_error("Minimum settle time %t is bigger than maximum settle time %t", $1, $2);
|
|
$$.min = $1;
|
|
$$.max = $2;
|
|
};
|
|
|
|
text:
|
|
TEXT
|
|
| CF_SYM_KNOWN {
|
|
if ($1->class != (SYM_CONSTANT | T_STRING)) cf_error("String constant expected");
|
|
$$ = SYM_VAL($1).s;
|
|
}
|
|
;
|
|
|
|
opttext:
|
|
TEXT
|
|
| /* empty */ { $$ = NULL; }
|
|
;
|
|
|
|
text_or_ipa:
|
|
TEXT { $$.type = T_STRING; $$.val.s = $1; }
|
|
| IP4 { $$.type = T_IP; $$.val.ip = ipa_from_ip4($1); }
|
|
| IP6 { $$.type = T_IP; $$.val.ip = ipa_from_ip6($1); }
|
|
| CF_SYM_KNOWN {
|
|
if (($1->class == (SYM_CONSTANT | T_STRING)) ||
|
|
($1->class == (SYM_CONSTANT | T_IP)))
|
|
$$ = *($1->val);
|
|
else
|
|
cf_error("String or IP constant expected");
|
|
}
|
|
| '(' term ')' {
|
|
$$ = *cf_eval($2, T_VOID);
|
|
if (($$.type != T_BYTESTRING) && ($$.type != T_STRING))
|
|
cf_error("Bytestring or string value expected");
|
|
}
|
|
;
|
|
|
|
bytestring:
|
|
BYTETEXT
|
|
| bytestring_expr { $$ = cf_eval($1, T_BYTESTRING)->val.bs; }
|
|
;
|
|
|
|
bytestring_text:
|
|
BYTETEXT { $$.type = T_BYTESTRING; $$.val.bs = $1; }
|
|
| TEXT { $$.type = T_STRING; $$.val.s = $1; }
|
|
| bytestring_expr {
|
|
$$ = *cf_eval($1, T_VOID);
|
|
if (($$.type != T_BYTESTRING) && ($$.type != T_STRING))
|
|
cf_error("Bytestring or string value expected");
|
|
}
|
|
;
|
|
|
|
bytestring_expr:
|
|
lvalue { $$ = f_lval_getter(&$1); }
|
|
| term_bs
|
|
| '(' term ')' { $$ = $2; }
|
|
;
|
|
|
|
|
|
CF_CODE
|
|
|
|
CF_END
|