diff options
author | Michele Marchetto <michele@cvs.openbsd.org> | 2009-06-01 20:59:46 +0000 |
---|---|---|
committer | Michele Marchetto <michele@cvs.openbsd.org> | 2009-06-01 20:59:46 +0000 |
commit | 5de1d23aeb3294b99aea577259871a4a7686e9b4 (patch) | |
tree | cc434b947408b0844b7b33a08358359751c9a7e4 /usr.sbin/ldpd/parse.y | |
parent | 19268a0235511684a7ee9e71b0c3dd8170ff3762 (diff) |
Welcome ldpd, the Label Distribution Protocol daemon.
Built using the imsg/three process framework, its main aim
is to redistribute MPLS labels between peers.
Right now it has some really basic functionalities,
the basic protocol works and peers are able to exchange
labels and insert them in the kernel.
It still does not react to changes of topology.
Not yet connected to the builds.
ok claudio@ deraadt@
Diffstat (limited to 'usr.sbin/ldpd/parse.y')
-rw-r--r-- | usr.sbin/ldpd/parse.y | 888 |
1 files changed, 888 insertions, 0 deletions
diff --git a/usr.sbin/ldpd/parse.y b/usr.sbin/ldpd/parse.y new file mode 100644 index 00000000000..534e59a0dcf --- /dev/null +++ b/usr.sbin/ldpd/parse.y @@ -0,0 +1,888 @@ +/* $OpenBSD: parse.y,v 1.1 2009/06/01 20:59:45 michele Exp $ */ + +/* + * Copyright (c) 2004, 2005, 2008 Esben Norby <norby@openbsd.org> + * Copyright (c) 2004 Ryan McBride <mcbride@openbsd.org> + * Copyright (c) 2002, 2003, 2004 Henning Brauer <henning@openbsd.org> + * Copyright (c) 2001 Markus Friedl. All rights reserved. + * Copyright (c) 2001 Daniel Hartmeier. All rights reserved. + * Copyright (c) 2001 Theo de Raadt. All rights reserved. + * + * Permission to use, copy, modify, and distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +%{ +#include <sys/types.h> +#include <sys/socket.h> +#include <sys/stat.h> +#include <netinet/in.h> +#include <arpa/inet.h> +#include <ctype.h> +#include <err.h> +#include <errno.h> +#include <unistd.h> +#include <ifaddrs.h> +#include <limits.h> +#include <stdarg.h> +#include <stdio.h> +#include <string.h> + +#include "ldp.h" +#include "ldpd.h" +#include "ldpe.h" +#include "log.h" + +TAILQ_HEAD(files, file) files = TAILQ_HEAD_INITIALIZER(files); +static struct file { + TAILQ_ENTRY(file) entry; + FILE *stream; + char *name; + int lineno; + int errors; +} *file, *topfile; +struct file *pushfile(const char *, int); +int popfile(void); +int check_file_secrecy(int, const char *); +int yyparse(void); +int yylex(void); +int yyerror(const char *, ...); +int kw_cmp(const void *, const void *); +int lookup(char *); +int lgetc(int); +int lungetc(int); +int findeol(void); + +TAILQ_HEAD(symhead, sym) symhead = TAILQ_HEAD_INITIALIZER(symhead); +struct sym { + TAILQ_ENTRY(sym) entry; + int used; + int persist; + char *nam; + char *val; +}; + +int symset(const char *, const char *, int); +char *symget(const char *); + +void clear_config(struct ldpd_conf *xconf); +u_int32_t get_rtr_id(void); +int host(const char *, struct in_addr *, struct in_addr *); + +static struct ldpd_conf *conf; +static int errors = 0; + +struct iface *iface = NULL; + +struct config_defaults { + u_int16_t holdtime; + u_int16_t keepalive; + u_int16_t hello_interval; + u_int8_t mode; +}; + +struct config_defaults globaldefs; +struct config_defaults lspacedefs; +struct config_defaults ifacedefs; +struct config_defaults *defs; + +struct iface *conf_get_if(struct kif *, struct kif_addr *); + +typedef struct { + union { + int64_t number; + char *string; + } v; + int lineno; +} YYSTYPE; + +%} + +%token LSPACE INTERFACE ROUTERID LFIBUPDATE RTLABEL +%token HOLDTIME HELLOINTERVAL KEEPALIVE +%token DISTRIBUTION RETENTION ADVERTISEMENT +%token EXTTAG PASSIVE +%token HELLOINTERVAL +%token YES NO +%token ERROR +%token <v.string> STRING +%token <v.number> NUMBER +%type <v.number> yesno +%type <v.string> string + +%% + +grammar : /* empty */ + | grammar '\n' + | grammar conf_main '\n' + | grammar varset '\n' + | grammar interface '\n' + | grammar error '\n' { file->errors++; } + ; + +string : string STRING { + if (asprintf(&$$, "%s %s", $1, $2) == -1) { + free($1); + free($2); + yyerror("string: asprintf"); + YYERROR; + } + free($1); + free($2); + } + | STRING + ; + +yesno : YES { $$ = 1; } + | NO { $$ = 0; } + ; + +varset : STRING '=' string { + if (conf->opts & LDPD_OPT_VERBOSE) + printf("%s = \"%s\"\n", $1, $3); + if (symset($1, $3, 0) == -1) + fatal("cannot store variable"); + free($1); + free($3); + } + ; + +conf_main : ROUTERID STRING { + if (!inet_aton($2, &conf->rtr_id)) { + yyerror("error parsing router-id"); + free($2); + YYERROR; + } + free($2); + } + | LFIBUPDATE yesno { + if ($2 == 0) + conf->flags |= LDPD_FLAG_NO_LFIB_UPDATE; + else + conf->flags &= ~LDPD_FLAG_NO_LFIB_UPDATE; + } + | DISTRIBUTION STRING { + conf->mode &= ~(MODE_DIST_INDEPENDENT | + MODE_DIST_ORDERED); + + if (!strcmp($2, "independent")) + conf->mode |= MODE_DIST_INDEPENDENT; + else if (!strcmp($2, "ordered")) + conf->mode |= MODE_DIST_ORDERED; + else { + yyerror("unknown distribution type"); + free($2); + YYERROR; + } + } + | RETENTION STRING { + conf->mode &= ~(MODE_RET_CONSERVATIVE | + MODE_RET_LIBERAL); + + if (!strcmp($2, "conservative")) + conf->mode |= MODE_RET_CONSERVATIVE; + else if (!strcmp($2, "liberal")) + conf->mode |= MODE_RET_LIBERAL; + else { + yyerror("unknown retention type"); + free($2); + YYERROR; + } + } + | ADVERTISEMENT STRING { + conf->mode &= ~(MODE_ADV_ONDEMAND | + MODE_ADV_UNSOLICITED); + + if (!strcmp($2, "ondemand")) + conf->mode |= MODE_ADV_ONDEMAND; + else if (!strcmp($2, "unsolicited")) + conf->mode |= MODE_ADV_UNSOLICITED; + else { + yyerror("unknown retention type"); + free($2); + YYERROR; + } + } + | RTLABEL STRING EXTTAG NUMBER { + if ($4 < 0 || $4 > UINT_MAX) { + yyerror("invalid external route tag"); + free($2); + YYERROR; + } + rtlabel_tag(rtlabel_name2id($2), $4); + free($2); + } + | defaults + ; +defaults : HOLDTIME NUMBER { + if ($2 < MIN_HOLDTIME || + $2 > MAX_HOLDTIME) { + yyerror("holdtime out of range (%d-%d)", + MIN_HOLDTIME, MAX_HOLDTIME); + YYERROR; + } + defs->holdtime = $2; + } + | KEEPALIVE NUMBER { + if ($2 < MIN_KEEPALIVE || + $2 > MAX_KEEPALIVE) { + yyerror("keepalive out of range (%d-%d)", + MIN_KEEPALIVE, MAX_KEEPALIVE); + YYERROR; + } + defs->keepalive = $2; + } + | HELLOINTERVAL NUMBER { + if ($2 < MIN_HELLO_INTERVAL || + $2 > MAX_HELLO_INTERVAL) { + yyerror("hello-interval out of range (%d-%d)", + MIN_HELLO_INTERVAL, MAX_HELLO_INTERVAL); + YYERROR; + } + defs->hello_interval = $2; + } + ; + +optnl : '\n' optnl + | + ; + +nl : '\n' optnl /* one newline or more */ + ; + +interface : INTERFACE STRING { + struct kif *kif; + struct kif_addr *ka = NULL; + char *s; + struct in_addr addr; + + s = strchr($2, ':'); + if (s) { + *s++ = '\0'; + if (inet_aton(s, &addr) == 0) { + yyerror( + "error parsing interface address"); + free($2); + YYERROR; + } + } else + addr.s_addr = 0; + + if ((kif = kif_findname($2, addr, &ka)) == NULL) { + yyerror("unknown interface %s", $2); + free($2); + YYERROR; + } + if (ka == NULL) { + if (s) + yyerror("address %s not configured on " + "interface %s", s, $2); + else + yyerror("unnumbered interface %s", $2); + free($2); + YYERROR; + } + free($2); + iface = conf_get_if(kif, ka); + if (iface == NULL) + YYERROR; + LIST_INSERT_HEAD(&conf->iface_list, iface, entry); + + memcpy(&ifacedefs, defs, sizeof(ifacedefs)); + defs = &ifacedefs; + } interface_block { + iface->holdtime = defs->holdtime; + iface->keepalive = defs->keepalive; + iface->hello_interval = defs->hello_interval; + iface = NULL; + + defs = &globaldefs; + } + ; + +interface_block : '{' optnl interfaceopts_l '}' + | '{' optnl '}' + ; + +interfaceopts_l : interfaceopts_l interfaceoptsl nl + | interfaceoptsl optnl + ; + +interfaceoptsl : PASSIVE { iface->passive = 1; } + | defaults + ; + +%% + +struct keywords { + const char *k_name; + int k_val; +}; + +int +yyerror(const char *fmt, ...) +{ + va_list ap; + + file->errors++; + va_start(ap, fmt); + fprintf(stderr, "%s:%d: ", file->name, yylval.lineno); + vfprintf(stderr, fmt, ap); + fprintf(stderr, "\n"); + va_end(ap); + return (0); +} + +int +kw_cmp(const void *k, const void *e) +{ + return (strcmp(k, ((const struct keywords *)e)->k_name)); +} + +int +lookup(char *s) +{ + /* this has to be sorted always */ + static const struct keywords keywords[] = { + {"advertisement", ADVERTISEMENT}, + {"distribution", DISTRIBUTION}, + {"external-tag", EXTTAG}, + {"hello-interval", HELLOINTERVAL}, + {"holdtime", HOLDTIME}, + {"interface", INTERFACE}, + {"keepalive", KEEPALIVE}, + {"labelspace", LSPACE}, + {"lfib-update", LFIBUPDATE}, + {"passive", PASSIVE}, + {"retention", RETENTION}, + {"router-id", ROUTERID}, + {"rtlabel", RTLABEL}, + {"yes", YES} + }; + const struct keywords *p; + + p = bsearch(s, keywords, sizeof(keywords)/sizeof(keywords[0]), + sizeof(keywords[0]), kw_cmp); + + if (p) + return (p->k_val); + else + return (STRING); +} + +#define MAXPUSHBACK 128 + +char *parsebuf; +int parseindex; +char pushback_buffer[MAXPUSHBACK]; +int pushback_index = 0; + +int +lgetc(int quotec) +{ + int c, next; + + if (parsebuf) { + /* Read character from the parsebuffer instead of input. */ + if (parseindex >= 0) { + c = parsebuf[parseindex++]; + if (c != '\0') + return (c); + parsebuf = NULL; + } else + parseindex++; + } + + if (pushback_index) + return (pushback_buffer[--pushback_index]); + + if (quotec) { + if ((c = getc(file->stream)) == EOF) { + yyerror("reached end of file while parsing " + "quoted string"); + if (file == topfile || popfile() == EOF) + return (EOF); + return (quotec); + } + return (c); + } + + while ((c = getc(file->stream)) == '\\') { + next = getc(file->stream); + if (next != '\n') { + c = next; + break; + } + yylval.lineno = file->lineno; + file->lineno++; + } + + while (c == EOF) { + if (file == topfile || popfile() == EOF) + return (EOF); + c = getc(file->stream); + } + return (c); +} + +int +lungetc(int c) +{ + if (c == EOF) + return (EOF); + if (parsebuf) { + parseindex--; + if (parseindex >= 0) + return (c); + } + if (pushback_index < MAXPUSHBACK-1) + return (pushback_buffer[pushback_index++] = c); + else + return (EOF); +} + +int +findeol(void) +{ + int c; + + parsebuf = NULL; + pushback_index = 0; + + /* skip to either EOF or the first real EOL */ + while (1) { + c = lgetc(0); + if (c == '\n') { + file->lineno++; + break; + } + if (c == EOF) + break; + } + return (ERROR); +} + +int +yylex(void) +{ + char buf[8096]; + char *p, *val; + int quotec, next, c; + int token; + +top: + p = buf; + while ((c = lgetc(0)) == ' ' || c == '\t') + ; /* nothing */ + + yylval.lineno = file->lineno; + if (c == '#') + while ((c = lgetc(0)) != '\n' && c != EOF) + ; /* nothing */ + if (c == '$' && parsebuf == NULL) { + while (1) { + if ((c = lgetc(0)) == EOF) + return (0); + + if (p + 1 >= buf + sizeof(buf) - 1) { + yyerror("string too long"); + return (findeol()); + } + if (isalnum(c) || c == '_') { + *p++ = (char)c; + continue; + } + *p = '\0'; + lungetc(c); + break; + } + val = symget(buf); + if (val == NULL) { + yyerror("macro '%s' not defined", buf); + return (findeol()); + } + parsebuf = val; + parseindex = 0; + goto top; + } + + switch (c) { + case '\'': + case '"': + quotec = c; + while (1) { + if ((c = lgetc(quotec)) == EOF) + return (0); + if (c == '\n') { + file->lineno++; + continue; + } else if (c == '\\') { + if ((next = lgetc(quotec)) == EOF) + return (0); + if (next == quotec || c == ' ' || c == '\t') + c = next; + else if (next == '\n') + continue; + else + lungetc(next); + } else if (c == quotec) { + *p = '\0'; + break; + } + if (p + 1 >= buf + sizeof(buf) - 1) { + yyerror("string too long"); + return (findeol()); + } + *p++ = (char)c; + } + yylval.v.string = strdup(buf); + if (yylval.v.string == NULL) + err(1, "yylex: strdup"); + return (STRING); + } + +#define allowed_to_end_number(x) \ + (isspace(x) || x == ')' || x ==',' || x == '/' || x == '}' || x == '=') + + if (c == '-' || isdigit(c)) { + do { + *p++ = c; + if ((unsigned)(p-buf) >= sizeof(buf)) { + yyerror("string too long"); + return (findeol()); + } + } while ((c = lgetc(0)) != EOF && isdigit(c)); + lungetc(c); + if (p == buf + 1 && buf[0] == '-') + goto nodigits; + if (c == EOF || allowed_to_end_number(c)) { + const char *errstr = NULL; + + *p = '\0'; + yylval.v.number = strtonum(buf, LLONG_MIN, + LLONG_MAX, &errstr); + if (errstr) { + yyerror("\"%s\" invalid number: %s", + buf, errstr); + return (findeol()); + } + return (NUMBER); + } else { +nodigits: + while (p > buf + 1) + lungetc(*--p); + c = *--p; + if (c == '-') + return (c); + } + } + +#define allowed_in_string(x) \ + (isalnum(x) || (ispunct(x) && x != '(' && x != ')' && \ + x != '{' && x != '}' && \ + x != '!' && x != '=' && x != '#' && \ + x != ',')) + + if (isalnum(c) || c == ':' || c == '_') { + do { + *p++ = c; + if ((unsigned)(p-buf) >= sizeof(buf)) { + yyerror("string too long"); + return (findeol()); + } + } while ((c = lgetc(0)) != EOF && (allowed_in_string(c))); + lungetc(c); + *p = '\0'; + if ((token = lookup(buf)) == STRING) + if ((yylval.v.string = strdup(buf)) == NULL) + err(1, "yylex: strdup"); + return (token); + } + if (c == '\n') { + yylval.lineno = file->lineno; + file->lineno++; + } + if (c == EOF) + return (0); + return (c); +} + +int +check_file_secrecy(int fd, const char *fname) +{ + struct stat st; + + if (fstat(fd, &st)) { + log_warn("cannot stat %s", fname); + return (-1); + } + if (st.st_uid != 0 && st.st_uid != getuid()) { + log_warnx("%s: owner not root or current user", fname); + return (-1); + } + if (st.st_mode & (S_IRWXG | S_IRWXO)) { + log_warnx("%s: group/world readable/writeable", fname); + return (-1); + } + return (0); +} + +struct file * +pushfile(const char *name, int secret) +{ + struct file *nfile; + + if ((nfile = calloc(1, sizeof(struct file))) == NULL || + (nfile->name = strdup(name)) == NULL) { + log_warn("malloc"); + return (NULL); + } + if ((nfile->stream = fopen(nfile->name, "r")) == NULL) { + log_warn("%s", nfile->name); + free(nfile->name); + free(nfile); + return (NULL); + } else if (secret && + check_file_secrecy(fileno(nfile->stream), nfile->name)) { + fclose(nfile->stream); + free(nfile->name); + free(nfile); + return (NULL); + } + nfile->lineno = 1; + TAILQ_INSERT_TAIL(&files, nfile, entry); + return (nfile); +} + +int +popfile(void) +{ + struct file *prev; + + if ((prev = TAILQ_PREV(file, files, entry)) != NULL) + prev->errors += file->errors; + + TAILQ_REMOVE(&files, file, entry); + fclose(file->stream); + free(file->name); + free(file); + file = prev; + return (file ? 0 : EOF); +} + +struct ldpd_conf * +parse_config(char *filename, int opts) +{ + struct sym *sym, *next; + + if ((conf = calloc(1, sizeof(struct ldpd_conf))) == NULL) + fatal("parse_config"); + conf->opts = opts; + + bzero(&globaldefs, sizeof(globaldefs)); + defs = &globaldefs; + defs->holdtime = DEFAULT_HOLDTIME; + defs->keepalive = DEFAULT_KEEPALIVE; + defs->hello_interval = DEFAULT_HELLO_INTERVAL; + defs->mode = (MODE_DIST_INDEPENDENT | MODE_RET_LIBERAL | + MODE_ADV_UNSOLICITED); + + if ((file = pushfile(filename, !(conf->opts & LDPD_OPT_NOACTION))) == NULL) { + free(conf); + return (NULL); + } + topfile = file; + + yyparse(); + errors = file->errors; + popfile(); + + /* Free macros and check which have not been used. */ + for (sym = TAILQ_FIRST(&symhead); sym != NULL; sym = next) { + next = TAILQ_NEXT(sym, entry); + if ((conf->opts & LDPD_OPT_VERBOSE2) && !sym->used) + fprintf(stderr, "warning: macro '%s' not " + "used\n", sym->nam); + if (!sym->persist) { + free(sym->nam); + free(sym->val); + TAILQ_REMOVE(&symhead, sym, entry); + free(sym); + } + } + + /* free global config defaults */ + if (errors) { + clear_config(conf); + return (NULL); + } + + if (conf->rtr_id.s_addr == 0) + conf->rtr_id.s_addr = get_rtr_id(); + + return (conf); +} + +int +symset(const char *nam, const char *val, int persist) +{ + struct sym *sym; + + for (sym = TAILQ_FIRST(&symhead); sym && strcmp(nam, sym->nam); + sym = TAILQ_NEXT(sym, entry)) + ; /* nothing */ + + if (sym != NULL) { + if (sym->persist == 1) + return (0); + else { + free(sym->nam); + free(sym->val); + TAILQ_REMOVE(&symhead, sym, entry); + free(sym); + } + } + if ((sym = calloc(1, sizeof(*sym))) == NULL) + return (-1); + + sym->nam = strdup(nam); + if (sym->nam == NULL) { + free(sym); + return (-1); + } + sym->val = strdup(val); + if (sym->val == NULL) { + free(sym->nam); + free(sym); + return (-1); + } + sym->used = 0; + sym->persist = persist; + TAILQ_INSERT_TAIL(&symhead, sym, entry); + return (0); +} + +int +cmdline_symset(char *s) +{ + char *sym, *val; + int ret; + size_t len; + + if ((val = strrchr(s, '=')) == NULL) + return (-1); + + len = strlen(s) - strlen(val) + 1; + if ((sym = malloc(len)) == NULL) + errx(1, "cmdline_symset: malloc"); + + strlcpy(sym, s, len); + + ret = symset(sym, val + 1, 1); + free(sym); + + return (ret); +} + +char * +symget(const char *nam) +{ + struct sym *sym; + + TAILQ_FOREACH(sym, &symhead, entry) + if (strcmp(nam, sym->nam) == 0) { + sym->used = 1; + return (sym->val); + } + return (NULL); +} + +struct iface * +conf_get_if(struct kif *kif, struct kif_addr *ka) +{ + struct iface *i; + + LIST_FOREACH(i, &conf->iface_list, entry) { + if (i->ifindex == kif->ifindex && + i->addr.s_addr == ka->addr.s_addr) { + yyerror("interface %s already configured", + kif->ifname); + return (NULL); + } + } + + i = if_new(kif, ka); + + return (i); +} + +void +clear_config(struct ldpd_conf *xconf) +{ + struct iface *i; + + while ((i = LIST_FIRST(&conf->iface_list)) != NULL) { + LIST_REMOVE(i, entry); + if_del(i); + } + + free(xconf); +} + +u_int32_t +get_rtr_id(void) +{ + struct ifaddrs *ifap, *ifa; + u_int32_t ip = 0, cur, localnet; + + localnet = htonl(INADDR_LOOPBACK & IN_CLASSA_NET); + + if (getifaddrs(&ifap) == -1) + fatal("getifaddrs"); + + for (ifa = ifap; ifa; ifa = ifa->ifa_next) { + if (ifa->ifa_addr->sa_family != AF_INET) + continue; + cur = ((struct sockaddr_in *)ifa->ifa_addr)->sin_addr.s_addr; + if ((cur & localnet) == localnet) /* skip 127/8 */ + continue; + if (cur > ip || ip == 0) + ip = cur; + } + freeifaddrs(ifap); + + if (ip == 0) + fatal("router-id is 0.0.0.0"); + + return (ip); +} + +int +host(const char *s, struct in_addr *addr, struct in_addr *mask) +{ + struct in_addr ina; + int bits = 32; + + bzero(&ina, sizeof(struct in_addr)); + if (strrchr(s, '/') != NULL) { + if ((bits = inet_net_pton(AF_INET, s, &ina, sizeof(ina))) == -1) + return (0); + } else { + if (inet_pton(AF_INET, s, &ina) != 1) + return (0); + } + + addr->s_addr = ina.s_addr; + mask->s_addr = prefixlen2mask(bits); + + return (1); +} |