0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2025-01-10 11:01:54 +00:00
bird/conf/confbase.Y

317 lines
6.3 KiB
Plaintext
Raw Normal View History

1998-11-27 19:36:06 +00:00
/*
* BIRD -- Configuration Parser Top
*
* (c) 1998--2000 Martin Mares <mj@ucw.cz>
1998-11-27 19:36:06 +00:00
*
* Can be freely distributed and used under the terms of the GNU GPL.
*/
CF_HDR
#define PARSER 1
#include <stdio.h>
1998-11-27 19:36:06 +00:00
#include "nest/bird.h"
#include "conf/conf.h"
#include "lib/resource.h"
#include "lib/socket.h"
#include "lib/timer.h"
#include "lib/string.h"
#include "nest/protocol.h"
#include "nest/iface.h"
#include "nest/route.h"
#include "nest/cli.h"
#include "filter/filter.h"
1998-11-27 19:36:06 +00:00
/* FIXME: Turn on YYERROR_VERBOSE and work around lots of bison bugs? */
CF_DEFINES
static void
check_u8(unsigned val)
{
if (val > 0xFF)
cf_error("Value %d out of range (0-255)", val);
}
static void
check_u16(unsigned val)
{
if (val > 0xFFFF)
cf_error("Value %d out of range (0-65535)", val);
}
static void
check_file_readability(const char *file_path)
{
FILE *file = fopen(file_path, "r");
if (file)
fclose(file);
else
cf_error("File '%s' cannot be open for read: %m", file_path);
}
1998-11-27 19:36:06 +00:00
CF_DECLS
%union {
int i;
u32 i32;
1998-11-27 19:36:06 +00:00
ip_addr a;
ip4_addr ip4;
ip6_addr ip6;
2015-12-24 14:52:03 +00:00
net_addr net;
net_addr *net_ptr;
1998-11-27 19:36:06 +00:00
struct symbol *s;
char *t;
struct rtable_config *r;
struct f_inst *x;
struct filter *f;
struct f_tree *e;
struct f_trie *trie;
struct f_val v;
struct f_path_mask *h;
1999-05-26 14:24:32 +00:00
struct password_item *p;
struct rt_show_data *ra;
struct sym_show_data *sd;
struct lsadb_show_data *ld;
struct iface *iface;
void *g;
bird_clock_t time;
struct f_prefix px;
struct proto_spec ps;
2010-02-02 23:19:24 +00:00
struct timeformat *tf;
1998-11-27 19:36:06 +00:00
}
%token END CLI_MARKER INVALID_TOKEN ELSECOL DDOT
2000-06-01 08:43:29 +00:00
%token GEQ LEQ NEQ AND OR
2009-03-14 11:43:10 +00:00
%token PO PC
1999-11-15 11:35:41 +00:00
%token <i> NUM ENUM
%token <ip4> IP4
%token <ip6> IP6
1998-11-27 19:36:06 +00:00
%token <s> SYM
%token <t> TEXT
%type <iface> ipa_scope
1998-11-27 19:36:06 +00:00
2015-12-24 14:52:03 +00:00
%type <i> expr bool pxlen4
%type <i32> expr_us
%type <time> datetime
2015-12-24 14:52:03 +00:00
%type <a> ipa
%type <net> net_ip4_ net_ip6_ net_ip6 net_ip_ net_ip net_or_ipa
%type <net_ptr> net_ net_any net_roa4_ net_roa6_ net_roa_
%type <t> text opttext
2000-05-16 14:24:33 +00:00
%nonassoc PREFIX_DUMMY
2010-01-27 22:45:36 +00:00
%left AND OR
%nonassoc '=' '<' '>' '~' GEQ LEQ NEQ PO PC
2010-01-27 22:45:36 +00:00
%left '+' '-'
%left '*' '/' '%'
%left '!'
%nonassoc '.'
CF_KEYWORDS(DEFINE, ON, OFF, YES, NO, S, MS, US, PORT)
1998-11-27 19:36:06 +00:00
CF_GRAMMAR
/* Basic config file structure */
config: conf_entries END { return 0; }
1999-11-17 12:00:21 +00:00
| CLI_MARKER cli_cmd { return 0; }
1998-11-27 19:36:06 +00:00
;
conf_entries:
/* EMPTY */
| conf_entries conf
1998-11-27 19:36:06 +00:00
;
CF_ADDTO(conf, ';')
1998-11-27 19:36:06 +00:00
/* Constant expressions */
CF_ADDTO(conf, definition)
definition:
DEFINE SYM '=' term ';' {
struct f_val *val = cfg_alloc(sizeof(struct f_val));
*val = f_eval($4, cfg_mem);
if (val->type == T_RETURN) cf_error("Runtime error");
cf_define_symbol($2, SYM_CONSTANT | val->type, val);
}
;
expr:
NUM
| '(' term ')' { $$ = f_eval_int($2); }
| SYM {
if ($1->class != (SYM_CONSTANT | T_INT)) cf_error("Number expected");
$$ = SYM_VAL($1).i; }
;
expr_us:
expr S { $$ = (u32) $1 * 1000000; }
| expr MS { $$ = (u32) $1 * 1000; }
| expr US { $$ = (u32) $1 * 1; }
;
/* expr_u16: expr { check_u16($1); $$ = $1; }; */
/* Switches */
bool:
expr {$$ = !!$1; }
| ON { $$ = 1; }
| YES { $$ = 1; }
| OFF { $$ = 0; }
| NO { $$ = 0; }
| /* Silence means agreement */ { $$ = 1; }
;
2015-12-24 14:52:03 +00:00
/* Addresses */
ipa:
2015-12-24 14:52:03 +00:00
IP4 { $$ = ipa_from_ip4($1); }
| IP6 { $$ = ipa_from_ip6($1); }
| SYM {
if ($1->class != (SYM_CONSTANT | T_IP)) cf_error("IP address expected");
$$ = SYM_VAL($1).ip;
}
;
ipa_scope:
/* empty */ { $$ = NULL; }
| '%' SYM { $$ = if_get_by_name($2->name); }
;
2015-12-24 14:52:03 +00:00
/* Networks - internal */
2015-12-24 14:52:03 +00:00
pxlen4:
'/' NUM {
if ($2 < 0 || $2 > IP4_MAX_PREFIX_LENGTH) cf_error("Invalid prefix length %d", $2);
$$ = $2;
}
| ':' IP4 {
$$ = ip4_masklen($2);
if ($$ == 255) cf_error("Invalid netmask %I4", $2);
2015-12-24 14:52:03 +00:00
}
;
2015-12-24 14:52:03 +00:00
net_ip4_: IP4 pxlen4
{
net_fill_ip4(&($$), $1, $2);
if (!net_validate_ip4((net_addr_ip4 *) &($$)))
cf_error("Invalid IPv4 prefix");
};
2015-12-24 14:52:03 +00:00
net_ip6_: IP6 '/' NUM
{
net_fill_ip6(&($$), $1, $3);
if ($3 < 0 || $3 > IP6_MAX_PREFIX_LENGTH)
cf_error("Invalid prefix length %d", $3);
if (!net_validate_ip6((net_addr_ip6 *) &($$)))
cf_error("Invalid IPv6 prefix");
};
net_roa4_: net_ip4_ MAX NUM AS NUM
{
$$ = cfg_alloc(sizeof(net_addr_roa4));
net_fill_roa4($$, ((net_addr_ip4 *)&$1)->prefix, $1.pxlen, $3, $5);
if ($3 < 0 || $3 > IP4_MAX_PREFIX_LENGTH)
cf_error("Invalid max prefix length %d", $3);
if (((net_addr_roa4 *) $$)->max_pxlen < ($$)->pxlen)
cf_error("Maximum prefix length %d must be >= prefix length %d", ((net_addr_roa4 *) $$)->max_pxlen, ($$)->pxlen);
};
net_roa6_: net_ip6_ MAX NUM AS NUM
{
$$ = cfg_alloc(sizeof(net_addr_roa6));
net_fill_roa6($$, ((net_addr_ip6 *)&$1)->prefix, $1.pxlen, $3, $5);
if ($3 < 0 || $3 > IP6_MAX_PREFIX_LENGTH)
cf_error("Invalid max prefix length %d", $3);
if (((net_addr_roa6 *) $$)->max_pxlen < ($$)->pxlen)
cf_error("Maximum prefix length %d must be >= prefix length %d", ((net_addr_roa6 *) $$)->max_pxlen, ($$)->pxlen);
};
2015-12-24 14:52:03 +00:00
net_ip_: net_ip4_ | net_ip6_ ;
net_roa_: net_roa4_ | net_roa6_ ;
net_:
net_ip_ { $$ = cfg_alloc($1.length); net_copy($$, &($1)); }
| net_roa_
;
2015-12-24 14:52:03 +00:00
/* Networks - regular */
net_ip6:
net_ip6_
| SYM {
if (($1->class != (SYM_CONSTANT | T_NET)) || (SYM_VAL($1).net->type != NET_IP6))
cf_error("IPv6 network expected");
$$ = * SYM_VAL($1).net;
}
2015-12-24 14:52:03 +00:00
;
net_ip:
net_ip_
| SYM {
if (($1->class != (SYM_CONSTANT | T_NET)) || !net_is_ip(SYM_VAL($1).net))
cf_error("IP network expected");
$$ = * SYM_VAL($1).net;
}
;
2015-12-24 14:52:03 +00:00
net_any:
net_
| SYM {
if ($1->class != (SYM_CONSTANT | T_NET))
cf_error("Network 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); }
| SYM {
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 expected");
}
;
2015-12-24 14:52:03 +00:00
2000-03-04 21:27:57 +00:00
datetime:
TEXT {
2005-02-12 22:27:55 +00:00
$$ = tm_parse_datetime($1);
if (!$$)
2005-02-12 22:27:55 +00:00
cf_error("Invalid date and time");
}
1999-05-26 14:24:32 +00:00
;
text:
TEXT
| SYM {
if ($1->class != (SYM_CONSTANT | T_STRING)) cf_error("String expected");
$$ = SYM_VAL($1).s;
}
;
opttext:
TEXT
| /* empty */ { $$ = NULL; }
;
1998-11-27 19:36:06 +00:00
CF_CODE
CF_END