summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--etc/bash_completion.d/20vyatta-cfg17
-rw-r--r--src/cli_def.l717
-rw-r--r--src/cli_parse.y5
3 files changed, 405 insertions, 334 deletions
diff --git a/etc/bash_completion.d/20vyatta-cfg b/etc/bash_completion.d/20vyatta-cfg
index bc9088f..6a411de 100644
--- a/etc/bash_completion.d/20vyatta-cfg
+++ b/etc/bash_completion.d/20vyatta-cfg
@@ -354,14 +354,13 @@ vyatta_parse_tmpl_comp_fields ()
# $1: tmpl
# $2: field name
sed -n '
- /^#'"$2"':/,$ {
- s/^#'"$2"'://
+ /^'"$2"':/,$ {
+ s/^'"$2"':[ ]*//
h
:b
$ { x; p; q }
n
- /^#[-_a-z]\+:/ { x; p; q }
- s/^#//
+ /^\([-_a-z]\+:\|#\)/ { x; p; q }
H
bb
}
@@ -387,12 +386,8 @@ vyatta_parse_tmpl ()
return
fi
eval `sed -n '
- /^help:[ ]\+/,/^[a-z]\+:/ {
- s/^help:[ ]\+/vyatta_cfg_help=/p
- /^ /p
- }
- /^syntax:[ ]\+\$(@)[ ]\+in[ ]\+/ {
- s/^syntax:[ ]\+\$(@)[ ]\+in[ ]\+/vyatta_cfg_allowed=( /
+ /^syntax:expression:[ ]\+\$(@)[ ]\+in[ ]\+/ {
+ s/^syntax:expression:[ ]\+\$(@)[ ]\+in[ ]\+/vyatta_cfg_allowed=( /
s/^\([^;]\+\);.*$/\1 )/
s/[ ]*,[ ]*/ /gp
}
@@ -401,6 +396,8 @@ vyatta_parse_tmpl ()
s/^type:[ ]\+\([^ ;]\+\)\(;.*\)\?/vyatta_cfg_type=\1/p
' $1`
+ vyatta_cfg_help=$(vyatta_parse_tmpl_comp_fields $1 "help")
+
local acmd=$(vyatta_parse_tmpl_comp_fields $1 "allowed")
vyatta_cfg_comp_help=$(vyatta_parse_tmpl_comp_fields $1 "comp_help")
diff --git a/src/cli_def.l b/src/cli_def.l
index 41994d6..20fb07d 100644
--- a/src/cli_def.l
+++ b/src/cli_def.l
@@ -1,23 +1,164 @@
+%x action
+%x expression
+%x str
+%option noyywrap
%{
#include "cli_val.h"
#include "cli_parse.h"
-FILE *yy_cli_def_in;
-static void make_def_value(vtw_type_e type);
-static int cli_last_nl_returned=0;
-#define STR_DELTA 2
-%}
-%x str
-%option noyywrap
-%option nounput
-%option never-interactive
-/*
- * Regular expressions of IP and MAC addresses, URLs, etc.
- */
+#define YY_DECL int real_yy_cli_def_lex(void)
+
+#define BUF_INCREMENT 4096
+static int action_buflen = 0;
+static char *action_buf = NULL;
+static char *action_ptr = NULL;
+static int str_buflen = 0;
+static char *str_buf = NULL;
+static char *str_ptr = NULL;
+static char str_delim = 0;
+static int eof_seen = 0;
+static int pre_str_state = 0;
+
+static char *reg_fields[] = { "default", "tag", "type", "multi", NULL };
+static int reg_fields_t[] = { DEFAULT, TAG, TYPE, MULTI, 0 };
+static char *act_fields[] = { "help", "syntax", "commit",
+ "delete", "update", "activate", "create",
+ "begin", "end",
+ "comp_help", "allowed",
+ NULL };
+static int act_fields_t[] = { HELP, SYNTAX, COMMIT,
+ ACTION, ACTION, ACTION, ACTION,
+ ACTION, ACTION,
+ DUMMY, DUMMY,
+ 0 };
+static int act_types[] = { -1, -1, -1,
+ delete_act, update_act, activate_act, create_act,
+ begin_act, end_act,
+ -1, -1,
+ -1 };
+
+static char *type_names[] = { "txt", "u32", "ipv4", "ipv4net",
+ "ipv6", "ipv6net", "bool", "macaddr",
+ NULL };
+static int type_t[] = { TEXT_TYPE, INT_TYPE, IPV4_TYPE, IPV4NET_TYPE,
+ IPV6_TYPE, IPV6NET_TYPE, BOOL_TYPE, MACADDR_TYPE, 0 };
+
+static char *op_cond_strs[] = { "in", "==", "!=", "<", ">", "<=", ">=", NULL };
+static int op_cond_types[] = { IN_COND, EQ_COND, NE_COND, LT_COND,
+ GT_COND, LE_COND, GE_COND, 0 };
+
+static char *op_strs[] = { "pattern", "exec", ",", "||", "&&", "=", "!",
+ "(", ")", ";", NULL };
+static int op_types[] = { PATTERN, EXEC, COMMA, OR, AND, ASSIGN,
+ NOT, LP, RP, SEMI, 0 };
+
+static void
+append_action(char *text)
+{
+ int tlen = strlen(text);
+ if (((action_ptr + tlen) - action_buf) >= action_buflen) {
+ action_buflen += BUF_INCREMENT;
+ action_buf = realloc(action_buf, action_buflen);
+ if (!action_buf) {
+ printf("Failed to allocate memory\n");
+ exit(-1);
+ }
+ }
+ strcpy(action_ptr, text);
+ action_ptr += tlen;
+}
-/*
- * IPv4 address representation.
- */
+static void
+append_str(char *text)
+{
+ int tlen = strlen(text);
+ if (((str_ptr + tlen) - str_buf) >= str_buflen) {
+ str_buflen += BUF_INCREMENT;
+ str_buf = realloc(str_buf, str_buflen);
+ if (!str_buf) {
+ printf("Failed to allocate memory\n");
+ exit(-1);
+ }
+ }
+ strcpy(str_ptr, text);
+ str_ptr += tlen;
+}
+
+static int
+return_action()
+{
+ *action_ptr = 0;
+ yy_cli_parse_lval.strp = strdup(action_buf);
+ action_ptr = action_buf;
+ return STRING;
+}
+
+static int
+return_str(char str_delim)
+{
+ *str_ptr = 0;
+ yy_cli_parse_lval.strp = strdup(str_buf);
+ str_ptr = str_buf;
+ return ((str_delim == '"') ? STRING : EX_STRING);
+}
+
+static int
+return_act_field(char *name)
+{
+ int idx = 0, ret = 0;
+ char *fname = NULL;
+ char *dname = strdup(name);
+ char *ptr = strchr(dname, ':');
+ if (ptr) {
+ *ptr = 0;
+ }
+ while ((fname = act_fields[idx])) {
+ if (strcmp(dname, fname) == 0) {
+ if (act_types[idx] >= 0) {
+ yy_cli_parse_lval.action = act_types[idx];
+ }
+ ret = act_fields_t[idx];
+ break;
+ }
+ ++idx;
+ }
+ free(dname);
+ return ret;
+}
+
+static int
+return_reg_field(char *name)
+{
+ int idx = 0, ret = 0;
+ char *fname = NULL;
+ char *dname = strdup(name);
+ char *ptr = strchr(dname, ':');
+ if (ptr) {
+ *ptr = 0;
+ }
+ while ((fname = reg_fields[idx])) {
+ if (strcmp(dname, fname) == 0) {
+ ret = reg_fields_t[idx];
+ break;
+ }
+ ++idx;
+ }
+ free(dname);
+ return ret;
+}
+
+static int
+return_value(vtw_type_e type)
+{
+ yy_cli_parse_lval.val.free_me = TRUE;
+ yy_cli_parse_lval.val.val = strdup(yy_cli_def_text);
+ yy_cli_parse_lval.val.val_type = type;
+ return VALUE;
+}
+
+%}
+
+/* IPv4 address representation. */
RE_IPV4_BYTE 25[0-5]|2[0-4][0-9]|[01][0-9][0-9]|([0-9]{1,2})
RE_IPV4 {RE_IPV4_BYTE}(\.{RE_IPV4_BYTE}){3}
RE_IPV4_PREFIXLEN (3[012]|[12][0-9]|[0-9])
@@ -26,28 +167,8 @@ RE_IPV4NET {RE_IPV4}"/"{RE_IPV4_PREFIXLEN}
/*
* IPv6 address representation in Augmented Backus-Naur Form (ABNF)
* as defined in RFC-2234.
- * IPv6 address representation taken from RFC-3986:
- *
- * IPv6address = 6( h16 ":" ) ls32
- * / "::" 5( h16 ":" ) ls32
- * / [ h16 ] "::" 4( h16 ":" ) ls32
- * / [ *1( h16 ":" ) h16 ] "::" 3( h16 ":" ) ls32
- * / [ *2( h16 ":" ) h16 ] "::" 2( h16 ":" ) ls32
- * / [ *3( h16 ":" ) h16 ] "::" h16 ":" ls32
- * / [ *4( h16 ":" ) h16 ] "::" ls32
- * / [ *5( h16 ":" ) h16 ] "::" h16
- * / [ *6( h16 ":" ) h16 ] "::"
- *
- * h16 = 1*4HEXDIG
- * ls32 = ( h16 ":" h16 ) / IPv4address
- * IPv4address = dec-octet "." dec-octet "." dec-octet "." dec-octet
- * dec-octet = DIGIT ; 0-9
- * / %x31-39 DIGIT ; 10-99
- * / "1" 2DIGIT ; 100-199
- * / "2" %x30-34 DIGIT ; 200-249
- * / "25" %x30-35 ; 250-255
+ * IPv6 address representation taken from RFC-3986.
*/
-
RE_H16 [a-fA-F0-9]{1,4}
RE_H16_COLON {RE_H16}":"
RE_LS32 (({RE_H16}":"{RE_H16})|{RE_IPV4})
@@ -64,14 +185,11 @@ RE_IPV6 {RE_IPV6_P1}|{RE_IPV6_P2}|{RE_IPV6_P3}|{RE_IPV6_P4}|{RE_IPV6_P5}|{RE_IP
RE_IPV6_PREFIXLEN 12[0-8]|1[01][0-9]|[0-9][0-9]?
RE_IPV6NET {RE_IPV6}"/"{RE_IPV6_PREFIXLEN}
-/*
- * Ethernet MAC address representation.
- */
+/* Ethernet MAC address representation. */
RE_MACADDR [a-fA-F0-9]{1,2}(:[a-fA-F0-9]{1,2}){5}
/*
* URL-related regular expressions.
- *
* Implementation is based on the BNF-like specification from:
* - RFC-1738: HTTP, FTP, FILE
* - RFC-3617: TFTP
@@ -79,22 +197,7 @@ RE_MACADDR [a-fA-F0-9]{1,2}(:[a-fA-F0-9]{1,2}){5}
*/
RE_URL {RE_URL_FILE}|{RE_URL_FTP}|{RE_URL_HTTP}|{RE_URL_TFTP}
-/*
- * URL schemeparts for IP based protocols.
- * Representation taken from RFC-1738, and some of it is updated by RFC-3986.
- *
- * login = [ user [ ":" password ] "@" ] hostport
- * hostport = host [ ":" port ]
- * host = hostname | hostnumber
- * hostname = *[ domainlabel "." ] toplabel
- * domainlabel = alphadigit | alphadigit *[ alphadigit | "-" ] alphadigit
- * toplabel = alpha | alpha *[ alphadigit | "-" ] alphadigit
- * alphadigit = alpha | digit
- * hostnumber = digits "." digits "." digits "." digits
- * port = digits
- * user = *[ uchar | ";" | "?" | "&" | "=" ]
- * password = *[ uchar | ";" | "?" | "&" | "=" ]
- */
+/* URL schemeparts for IP based protocols. */
RE_URL_LOGIN ({RE_URL_USER}(":"{RE_URL_PASSWORD})?"@")?{RE_URL_HOSTPORT}
RE_URL_HOSTPORT {RE_URL_HOST}(":"{RE_URL_PORT})?
RE_URL_HOST {RE_URL_HOSTNAME}|{RE_IPV4}|{RE_URL_IP_LITERAL}
@@ -109,90 +212,27 @@ RE_URL_PORT {RE_URL_DIGITS}
RE_URL_USER ({RE_URL_UCHAR}|";"|"?"|"&"|"=")*
RE_URL_PASSWORD ({RE_URL_UCHAR}|";"|"?"|"&"|"=")*
-/*
- * FILE URL regular expression.
- * Representation taken from RFC-1738.
- *
- * fileurl = "file://" [ host | "localhost" ] "/" fpath
- */
+/* FILE URL regular expression. */
RE_URL_FILE "file://"({RE_URL_HOST}|"localhost")?"/"{RE_URL_FPATH}
-/*
- * FTP URL regular expression.
- * Representation taken from RFC-1738.
- *
- * ftpurl = "ftp://" login [ "/" fpath [ ";type=" ftptype ] ]
- * fpath = fsegment *[ "/" fsegment ]
- * fsegment = *[ uchar | "?" | ":" | "@" | "&" | "=" ]
- * ftptype = "A" | "I" | "D" | "a" | "i" | "d"
- */
+/* FTP URL regular expression. */
RE_URL_FTP "ftp://"{RE_URL_LOGIN}("/"{RE_URL_FPATH}(";type="{RE_URL_FTPTYPE})?)?
RE_URL_FPATH {RE_URL_FSEGMENT}("/"{RE_URL_FSEGMENT})*
RE_URL_FSEGMENT ({RE_URL_UCHAR}|"?"|":"|"@"|"&"|"=")*
RE_URL_FTPTYPE "A"|"I"|"D"|"a"|"i"|"d"
-/*
- * HTTP URL regular expression.
- * Representation taken from RFC-1738.
- *
- * httpurl = "http://" hostport [ "/" hpath [ "?" search ] ]
- * hpath = hsegment *[ "/" hsegment ]
- * hsegment = *[ uchar | ";" | ":" | "@" | "&" | "=" ]
- * search = *[ uchar | ";" | ":" | "@" | "&" | "=" ]
- */
+/* HTTP URL regular expression. */
RE_URL_HTTP "http://"{RE_URL_HOSTPORT}("/"{RE_URL_HPATH}("?"{RE_URL_SEARCH})?)?
RE_URL_HPATH {RE_URL_HSEGMENT}("/"{RE_URL_HSEGMENT})*
RE_URL_HSEGMENT ({RE_URL_UCHAR}|";"|":"|"@"|"&"|"=")*
RE_URL_SEARCH ({RE_URL_UCHAR}|";"|":"|"@"|"&"|"=")*
-/*
- * TFTP URL regular expression.
- * Representation taken from RFC-3617.
- *
- * tftpURI = "tftp://" host "/" file [ mode ]
- * mode = ";" "mode=" ( "netascii" / "octet" )
- * file = *( unreserved / escaped )
- * host = <as specified by RFC 2732 [3]>
- * unreserved = <as specified in RFC 2396 [4]>
- * escaped = <as specified in RFC 2396>
- */
+/* TFTP URL regular expression. */
RE_URL_TFTP "tftp://"{RE_URL_HOST}"/"{RE_URL_TFTP_FILE}({RE_URL_TFTP_MODE})?
RE_URL_TFTP_MODE ";""mode="("netascii"|"octet")
RE_URL_TFTP_FILE ({RE_URL_UNRESERVED}|{RE_URL_ESCAPE})*
-/*
- * URL-related miscellaneous definitions.
- * Representation taken from RFC-1738 and from RFC-3986.
- *
- * lowalpha = "a" | "b" | "c" | "d" | "e" | "f" | "g" | "h" |
- * "i" | "j" | "k" | "l" | "m" | "n" | "o" | "p" |
- * "q" | "r" | "s" | "t" | "u" | "v" | "w" | "x" |
- * "y" | "z"
- * hialpha = "A" | "B" | "C" | "D" | "E" | "F" | "G" | "H" | "I" |
- * "J" | "K" | "L" | "M" | "N" | "O" | "P" | "Q" | "R" |
- * "S" | "T" | "U" | "V" | "W" | "X" | "Y" | "Z"
- * alpha = lowalpha | hialpha
- * digit = "0" | "1" | "2" | "3" | "4" | "5" | "6" | "7" |
- * "8" | "9"
- * safe = "$" | "-" | "_" | "." | "+"
- * extra = "!" | "*" | "'" | "(" | ")" | ","
- * national = "{" | "}" | "|" | "\" | "^" | "~" | "[" | "]" | "`"
- * punctuation = "<" | ">" | "#" | "%" | <">
- *
- *
- * reserved = ";" | "/" | "?" | ":" | "@" | "&" | "="
- * hex = digit | "A" | "B" | "C" | "D" | "E" | "F" |
- * "a" | "b" | "c" | "d" | "e" | "f"
- * escape = "%" hex hex
- *
- * unreserved = alpha | digit | safe | extra
- * uchar = unreserved | escape
- * xchar = unreserved | reserved | escape
- * digits = 1*digit
- *
- * sub-delims = "!" / "$" / "&" / "'" / "(" / ")"
- * / "*" / "+" / "," / ";" / "="
- */
+/* URL-related miscellaneous definitions. */
RE_URL_LOWALPHA [a-z]
RE_URL_HIALPHA [A-Z]
RE_URL_ALPHA {RE_URL_LOWALPHA}|{RE_URL_HIALPHA}
@@ -210,215 +250,244 @@ RE_URL_XCHAR {RE_URL_UNRESERVED}|{RE_URL_RESERVED}|{RE_URL_ESCAPE}
RE_URL_DIGITS {RE_URL_DIGIT}{1,}
RE_URL_SUBDELIMS "!"|"$"|"&"|"'"|"("|")"|"*"|"+"|","|";"|"="
+/* type names */
+RE_TYPE_NAME (txt|u32|ipv4|ipv4net|ipv6|ipv6net|bool|macaddr)
+
+/* values */
+RE_VAL_U32 [0-9]+
+RE_VAL_BOOL (true|false)
+
+/* operators */
+RE_OP_COND (==|!=|<|>|<=|>=|in)
+RE_OP_OTHER (pattern|exec|,|\|\||&&|=|!|\(|\)|;)
+
+/* template fields */
+RE_REG_FIELD (default|tag|type|multi)
+RE_ACT_FIELD (help|syntax|commit|delete|update|activate|create|begin|end|comp_help|allowed)
%%
- char *string_buf=NULL;
- char *string_buf_ptr=NULL;
- int string_len=0, string_buf_len=0;
- char str_delim=0;
-
-#[^\n]*(\\[ \t]*\n[^\n]*)*\n { /* comment */ }
-\n {yy_cli_def_lineno++;return EOL;}
-<<EOF>> { if(!cli_last_nl_returned) {cli_last_nl_returned=1;return EOL;} yyterminate(); }
-[\`\"] {
- BEGIN(str);
- str_delim = yy_cli_def_text[0];
- string_buf_ptr = string_buf = my_malloc(STR_DELTA + 1, "");
- string_len = STR_DELTA;
- string_buf_len = STR_DELTA;
- }
+
+#[^\n]*\n {
+ /* comment */
+ }
+
+\n {
+ return EOL;
+ }
+
+{RE_REG_FIELD}:[ \t]* {
+ return return_reg_field(yy_cli_def_text);
+ }
+
+<INITIAL>[\`\"] {
+ BEGIN(str);
+ pre_str_state = INITIAL;
+ str_delim = yy_cli_def_text[0];
+ }
+
+<expression>[\`\"] {
+ BEGIN(str);
+ pre_str_state = expression;
+ str_delim = yy_cli_def_text[0];
+ }
+
<str>[\"\`] {
- /* maybe a closing quote - all done */
- if(str_delim == yy_cli_def_text[0]) {
- BEGIN(INITIAL);
- *string_buf_ptr = 0;
- /* return string constant token type and
- * value to parser
- */
- yy_cli_parse_lval.strp = string_buf;
- return str_delim == '"'?STRING:EX_STRING;
- }else{
- *string_buf_ptr++ = yy_cli_def_text[0];
- goto string_too_long;
- }
- }
-
-<str>\n {
- /* error - unterminated string constant */
- yy_cli_def_lineno++;
- }
-
-<str>\\n {*string_buf_ptr++ = '\n';goto string_too_long;}
-<str>\\t {*string_buf_ptr++ = '\t';goto string_too_long;}
-<str>\\r {*string_buf_ptr++ = '\r';goto string_too_long;}
-<str>\\b {*string_buf_ptr++ = '\b';goto string_too_long;}
-<str>\\f {*string_buf_ptr++ = '\f';goto string_too_long;}
-<str>\\\n { yy_cli_def_lineno++; /* continuation => ignore */ }
-
-<str>\\. {
- *string_buf_ptr++ = yy_cli_def_text[1];
-string_too_long:
- *string_buf_ptr = 0;
- /* printf("Cur string |%s|\n", string_buf); */
- if (!--string_len) {
- string_buf = my_realloc(string_buf,
- string_buf_len + STR_DELTA + 1,
- "cli_def STRING");
- string_buf_ptr = string_buf + string_buf_len;
- string_buf_len += STR_DELTA;
- string_len = STR_DELTA;
+ if (str_delim == yy_cli_def_text[0]) {
+ BEGIN(pre_str_state);
+ return return_str(str_delim);
+ } else {
+ char tmp[2] = { yy_cli_def_text[0], 0 };
+ append_str(tmp);
+ }
+ }
+
+<str>\\\n {
+ /* continuation */
+ }
+
+<str>\\. {
+ unsigned int i = 0;
+ unsigned char tbl[256];
+ unsigned char c = 0;
+ char tmp[2] = { 0, 0 };
+ for (i = 0; i < 256; i++) {
+ tbl[i] = i;
+ }
+ c = 'n'; tbl[c] = '\n';
+ c = 't'; tbl[c] = '\t';
+ c = 'r'; tbl[c] = '\r';
+ c = 'b'; tbl[c] = '\b';
+ c = 'f'; tbl[c] = '\f';
+ /* note: can't have "[[" or "]]" (confuses m4) */
+ tmp[0] = tbl[ (int) yy_cli_def_text[1] ];
+ append_str(tmp);
+ }
+
+<str>[^\"\`\\]+ {
+ append_str(yy_cli_def_text);
+ }
+
+<str><<EOF>> {
+ BEGIN(INITIAL);
+ return return_str(str_delim);
+ }
+
+{RE_ACT_FIELD}:expression:[ \t]* {
+ BEGIN(expression);
+ return return_act_field(yy_cli_def_text);
+ }
+
+{RE_ACT_FIELD}:[ \t]* {
+ BEGIN(action);
+ return return_act_field(yy_cli_def_text);
+ }
+
+<expression>\n(({RE_REG_FIELD}|{RE_ACT_FIELD}):|#).* {
+ int i = 0;
+ char *tmp = strdup(yy_cli_def_text);
+ BEGIN(INITIAL);
+ for (i = yy_cli_def_leng - 1; i >= 0; --i) {
+ unput( tmp[i] );
+ }
+ free(tmp);
+ }
+
+<expression>\n { /* skip the \n */ }
+
+<expression><<EOF>> {
+ BEGIN(INITIAL);
+ eof_seen = 1;
+ return EOL;
+ }
+
+<action>\n(({RE_REG_FIELD}|{RE_ACT_FIELD}):|#).* {
+ int i = 0;
+ char *tmp = strdup(yy_cli_def_text);
+ BEGIN(INITIAL);
+ for (i = yy_cli_def_leng - 1; i >= 0; --i) {
+ unput( tmp[i] );
+ }
+ free(tmp);
+ return return_action();
+ }
+
+<action>\n?.* {
+ append_action(yy_cli_def_text);
+ }
+
+<action><<EOF>> {
+ BEGIN(INITIAL);
+ return return_action();
+ }
+
+<<EOF>> {
+ if (eof_seen) {
+ eof_seen = 0;
+ yyterminate();
+ }
+ eof_seen = 1;
+ return EOL;
+ }
+
+<INITIAL,expression>[ \t]+ { /* space */ }
+
+<expression>\\\n { /* continuation */ }
+
+{RE_TYPE_NAME} {
+ int i = 0;
+ while (type_names[i]) {
+ if (strcmp(type_names[i], yy_cli_def_text) == 0) {
+ yy_cli_parse_lval.type = type_t[i];
+ return TYPE_DEF;
+ }
+ i++;
}
}
-<str>[^\\\n\"\`]+ {
- char *yptr = yy_cli_def_text;
-
- while ( *yptr ){
- *string_buf_ptr++ = *yptr++;
- if (!--string_len) {
- string_buf = my_realloc(string_buf,
- string_buf_len + STR_DELTA + 1,
- "cli_def STRING");
- string_buf_ptr = string_buf + string_buf_len;
- string_buf_len += STR_DELTA;
- string_len = STR_DELTA;
- }
- }
- *string_buf_ptr = 0;
+
+<expression>{RE_OP_COND} {
+ int i = 0;
+ while (op_cond_strs[i]) {
+ if (strcmp(op_cond_strs[i], yy_cli_def_text) == 0) {
+ yy_cli_parse_lval.cond = op_cond_types[i];
+ return COND;
+ }
+ i++;
}
+ }
-default: return DEFAULT;
-tag: return TAG;
-type: return TYPE;
-help: return HELP;
-syntax: return SYNTAX;
-commit: return COMMIT;
-check: yy_cli_parse_lval.action = syntax_act; return ACTION;
-delete: yy_cli_parse_lval.action = delete_act; return ACTION;
-update: yy_cli_parse_lval.action = update_act; return ACTION;
-activate: yy_cli_parse_lval.action = activate_act; return ACTION;
-create: yy_cli_parse_lval.action = create_act; return ACTION;
-begin: yy_cli_parse_lval.action = begin_act; return ACTION;
-end: yy_cli_parse_lval.action = end_act; return ACTION;
-multi: return MULTI;
-
-:: {
- make_def_value(IPV6_TYPE);
- return VALUE;
- }
-txt {
- yy_cli_parse_lval.type = TEXT_TYPE;
- return TYPE_DEF;
- }
-
-pattern return PATTERN;
-
-exec return EXEC;
-
-, return COMMA;
-\|\| return OR;
-\&\& return AND;
-\= return ASSIGN;
-\=\= {yy_cli_parse_lval.cond = EQ_COND; return COND;}
-\!\= {yy_cli_parse_lval.cond = NE_COND; return COND;}
-\< {yy_cli_parse_lval.cond = LT_COND; return COND;}
-\> {yy_cli_parse_lval.cond = GT_COND; return COND;}
-\<\= {yy_cli_parse_lval.cond = LE_COND; return COND;}
-\>\= {yy_cli_parse_lval.cond = GE_COND; return COND;}
-in {yy_cli_parse_lval.cond = IN_COND; return COND;}
-\! { return NOT; }
-\$\([^)]+\) {
- yy_cli_parse_lval.strp = my_strdup(yy_cli_def_text, "TEXT");
- return VAR;
- }
-true {
- make_def_value(BOOL_TYPE);
- return VALUE;
- }
-
-false {
- make_def_value(BOOL_TYPE);
- return VALUE;
- }
-
-[0-9]+ {
- make_def_value(INT_TYPE);
- return VALUE;
- }
-
-{RE_IPV4} {
- make_def_value(IPV4_TYPE);
- return VALUE;
- }
-
-{RE_IPV4NET} {
- make_def_value(IPV4NET_TYPE);
- return VALUE;
- }
-
-{RE_IPV6} {
- make_def_value(IPV6_TYPE);
- return VALUE;
- }
-
-{RE_IPV6NET} {
- make_def_value(IPV6NET_TYPE);
- return VALUE;
- }
-
-{RE_MACADDR} {
- make_def_value(MACADDR_TYPE);
- return VALUE;
- }
-u32 {
- yy_cli_parse_lval.type = INT_TYPE;
- return TYPE_DEF;
- }
-
-ipv4 {
- yy_cli_parse_lval.type = IPV4_TYPE;
- return TYPE_DEF;
- }
-
-ipv4net {
- yy_cli_parse_lval.type = IPV4NET_TYPE;
- return TYPE_DEF;
- }
-ipv6 {
- yy_cli_parse_lval.type = IPV6_TYPE;
- return TYPE_DEF;
- }
-
-ipv6net {
- yy_cli_parse_lval.type = IPV6NET_TYPE;
- return TYPE_DEF;
- }
-bool {
- yy_cli_parse_lval.type = BOOL_TYPE;
- return TYPE_DEF;
- }
-macaddr {
- yy_cli_parse_lval.type = MACADDR_TYPE;
- return TYPE_DEF;
- }
-\( return LP;
-\) return RP;
-; return SEMI;
-
-\\\n { yy_cli_def_lineno++; /*whitespace -- continuation => ignore */ }
-
-[ \t]+ /* whitespace */
-
-. {
- /* everything else is a syntax error */
- return SYNTAX_ERROR;
- }
+<expression>{RE_OP_OTHER} {
+ int i = 0;
+ while (op_strs[i]) {
+ if (strcmp(op_strs[i], yy_cli_def_text) == 0) {
+ return op_types[i];
+ }
+ i++;
+ }
+ }
+
+<expression>\$\([^)]+\) {
+ yy_cli_parse_lval.strp = strdup(yy_cli_def_text);
+ return VAR;
+ }
+
+<expression>{RE_VAL_U32} { return return_value(INT_TYPE); }
+<expression>{RE_IPV4} { return return_value(IPV4_TYPE); }
+<expression>{RE_IPV4NET} { return return_value(IPV4NET_TYPE); }
+<expression>{RE_IPV6} { return return_value(IPV6_TYPE); }
+<expression>{RE_IPV6NET} { return return_value(IPV6NET_TYPE); }
+<expression>{RE_VAL_BOOL} { return return_value(BOOL_TYPE); }
+<expression>{RE_MACADDR} { return return_value(MACADDR_TYPE); }
+
+<*>. {
+ return SYNTAX_ERROR;
+ }
%%
-static void make_def_value(vtw_type_e type)
+static void
+init_bufs()
{
- memset(&yy_cli_parse_lval.val.val, 0, sizeof(yy_cli_parse_lval.val.val));
- yy_cli_parse_lval.val.free_me = TRUE;
- yy_cli_parse_lval.val.val = my_strdup(yy_cli_def_text, "cli_parse.l");
- yy_cli_parse_lval.val.val_type = type;
+ action_buf = malloc(BUF_INCREMENT);
+ action_ptr = action_buf;
+ action_buflen = BUF_INCREMENT;
+
+ str_buf = malloc(BUF_INCREMENT);
+ str_ptr = str_buf;
+ str_buflen = BUF_INCREMENT;
+
+ if (!action_buf || !str_buf) {
+ printf("Failed to allocate memory\n");
+ exit(-1);
+ }
}
+
+int
+yy_cli_def_lex()
+{
+ if (!action_buf) {
+ init_bufs();
+ }
+ return real_yy_cli_def_lex();
+}
+
+#if 0
+#define STANDALONE_TEST
+#endif
+#ifdef STANDALONE_TEST
+/* build:
+ flex -d --prefix=yy_cli_def_ -o cli_def.c cli_def.l && gcc -o test_def
+ cli_def.c cli_parse.o cli_new.o cli_objects.o cli_path_utils.o
+ cli_val_engine.o cli_val.o
+ */
+int
+main(int argc, char *argv[])
+{
+ int token = 0;
+ yy_cli_def_in = fopen(argv[1], "r");
+ while((token = yy_cli_def_lex()) > 0) {
+ printf("token[%d]\n", token);
+ }
+ return 0;
+}
+#endif
+
diff --git a/src/cli_parse.y b/src/cli_parse.y
index 372f3a9..150fe71 100644
--- a/src/cli_parse.y
+++ b/src/cli_parse.y
@@ -34,6 +34,7 @@ static void cli_deferror(const char *);
%token SYNTAX
%token COMMIT
%token CHECK
+%token DUMMY
%left SEMI
%token <val>VALUE
%token <type>TYPE_DEF
@@ -100,8 +101,12 @@ cause: help_cause
| default_cause
| syntax_cause
| ACTION action { append(parse_defp->actions + $1, $2, 0);}
+ | dummy_stmt
;
+dummy_stmt: DUMMY STRING { /* ignored */ }
+ ;
+
help_cause: HELP STRING
{ parse_defp->def_node_help = $2; /* no semantics for now */
}