代码拉取完成,页面将自动刷新
/*
CLI generator. Take idl as input and generate a tree for use in cli.
***** BEGIN LICENSE BLOCK *****
Copyright (C) 2001-2021 Olof Hagsand
This file is part of CLIgen.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
Alternatively, the contents of this file may be used under the terms of
the GNU General Public License Version 2 or later (the "GPL"),
in which case the provisions of the GPL are applicable instead
of those above. If you wish to allow use of your version of this file only
under the terms of the GPL, and not to allow others to
use your version of this file under the terms of Apache License version 2, indicate
your decision by deleting the provisions above and replace them with the
notice and other provisions required by the GPL. If you do not delete
the provisions above, a recipient may use your version of this file under
the terms of any one of the Apache License version 2 or the GPL.
***** END LICENSE BLOCK *****
Choice:
*/
%start file
%union {
int intval;
char *string;
}
%token MY_EOF
%token V_RANGE V_LENGTH V_CHOICE V_KEYWORD V_REGEXP V_FRACTION_DIGITS V_SHOW V_TREENAME V_TRANSLATE
%token DOUBLEPARENT /* (( */
%token DQ /* " */
%token DQP /* ") */
%token PDQ /* (" */
%token SETS /* @{ SETS */
%token <string> NAME /* in variables: <NAME type:NAME> */
%token <string> NUMBER /* In variables */
%token <string> DECIMAL /* In variables */
%token <string> CHARS
%type <string> charseq
%type <string> choices
%type <string> numdec
%type <string> arg1
%type <string> typecast
%type <intval> preline
%lex-param {void *_cy} /* Add this argument to parse() and lex() function */
%parse-param {void *_cy}
%{
/* Here starts user C-code */
/* typecast macro */
#define _CY ((cligen_yacc *)_cy)
/* add _cy to error paramaters */
#define YY_(msgid) msgid
#include "cligen_config.h"
#include <stdio.h>
#include <string.h>
#include <stdint.h>
#include <stdlib.h>
#include <errno.h>
#include <limits.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <net/if.h>
#include "cligen_buf.h"
#include "cligen_cv.h"
#include "cligen_cvec.h"
#include "cligen_parsetree.h"
#include "cligen_pt_head.h"
#include "cligen_object.h"
#include "cligen_match.h"
#include "cligen_syntax.h"
#include "cligen_handle.h"
#include "cligen_parse.h"
static int debug = 0;
/* Enable for debugging, steals some cycles otherwise */
#if 0
#define _PARSE_DEBUG(s) fprintf(stderr, s "\n")
#else
#define _PARSE_DEBUG(s)
#endif
extern int cligen_parseget_lineno (void);
int
cligen_parse_debug(int d)
{
debug = d;
return 0;
}
/*! CLIGEN parse error routine
* Also called from yacc generated code *
* @param[in] cy CLIgen yacc parse struct
*/
void cligen_parseerror(void *_cy,
char *s)
{
fprintf(stderr, "%s%s%d: Error: %s: at or before: '%s'\n",
_CY->cy_name,
":" ,
_CY->cy_linenum ,
s,
cligen_parsetext);
return;
}
#define cligen_parseerror1(cy, s) cligen_parseerror(cy, s)
/*! Create a CLIgen variable (cv) and store it in the current variable object
* Note that only one such cv can be stored.
* @param[in] cy CLIgen yacc parse struct
*/
static cg_var *
create_cv(cligen_yacc *cy,
char *type,
char *str)
{
cg_var *cv = NULL;
if ((cv = cv_new(CGV_STRING)) == NULL){
fprintf(stderr, "malloc: %s\n", strerror(errno));
goto done;
}
if (type){
if (cv_type_set(cv, cv_str2type(type)) == CGV_ERR){
fprintf(stderr, "%s:%d: error: No such type: %s\n",
cy->cy_name, cy->cy_linenum, type);
cv_free(cv); cv = NULL;
goto done;
}
}
if (cv_parse(str, cv) < 0){ /* parse str into cgv */
cv_free(cv); cv = NULL;
goto done;
}
done:
return cv;
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
static int
cgy_flag(cligen_yacc *cy,
char *var)
{
struct cgy_stack *cs = cy->cy_stack;
cg_var *cv;
int retval = -1;
if (debug)
fprintf(stderr, "%s: %s 1\n", __FUNCTION__, var);
if (cs){ /* XXX: why cs? */
if (cy->cy_cvec == NULL){
if ((cy->cy_cvec = cvec_new(0)) == NULL){
fprintf(stderr, "%s: cvec_new:%s\n", __FUNCTION__, strerror(errno));
goto done;
}
}
if ((cv = cvec_add(cy->cy_cvec, CGV_INT32)) == NULL){
fprintf(stderr, "%s: realloc:%s\n", __FUNCTION__, strerror(errno));
goto done;
}
cv_name_set(cv, var);
cv_int32_set(cv, 1);
}
retval = 0;
done:
return retval;
}
/*! Set a new treename. In fact registers the previous tree and creates a new .
* Note that one could have used an assignment: treename = <name>; for this but
* I decided to create special syntax for this so that assignments can use any
* variable names.
* @param[in] cy CLIgen yacc parse struct
* @param[in] name Name of tree
*/
static int
cgy_treename(cligen_yacc *cy,
char *name)
{
cg_obj *co = NULL;
cg_obj *cot;
struct cgy_list *cl;
int retval = -1;
int i;
parse_tree *pt;
pt_head *ph;
/* Get the first object */
for (cl=cy->cy_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
break;
}
/* Get the top object */
cot = co_top(co); /* co and cot can be same object */
pt = co_pt_get(cot);
/* If anything parsed */
if (pt_len_get(pt)){
/* 2. Add the old parse-tree with old name*/
for (i=0; i<pt_len_get(pt); i++){
if ((co=pt_vec_i_get(pt, i)) != NULL)
co_up_set(co, NULL);
}
if ((ph = cligen_ph_add(cy->cy_handle, cy->cy_treename)) == NULL)
goto done;
if (cligen_ph_parsetree_set(ph, pt) < 0)
goto done;
/* 3. Create new parse-tree XXX */
if ((pt = pt_new()) == NULL)
goto done;
co_pt_clear(cot);
co_pt_set(cot, pt);
}
/* 4. Set the new name */
if (cy->cy_treename)
free(cy->cy_treename);
if ((cy->cy_treename = strdup(name)) == NULL){
fprintf(stderr, "%s: strdup: %s\n", __FUNCTION__, strerror(errno));
goto done;
}
retval = 0;
done:
return retval;
}
/*! Variable assignment
* Only string type supported for now
* @param[in] cy CLIgen yacc parse struct
*/
static int
cgy_assignment(cligen_yacc *cy,
char *var,
char *val)
{
struct cgy_stack *cs = cy->cy_stack;
int retval = -1;
cg_var *cv;
char *treename_keyword;
cligen_handle h = cy->cy_handle;
if (debug)
fprintf(stderr, "%s: %s=%s\n", __FUNCTION__, var, val);
if (cs == NULL){
fprintf(stderr, "%s: Error, stack should not be NULL\n", __FUNCTION__);
}
if (cs->cs_next != NULL){ /* local */
if (cy->cy_cvec == NULL)
if ((cy->cy_cvec = cvec_new(0)) == NULL){
fprintf(stderr, "%s: cvec_new:%s\n", __FUNCTION__, strerror(errno));
goto done;
}
if ((cv = cvec_add(cy->cy_cvec, CGV_STRING)) == NULL){
fprintf(stderr, "%s: realloc:%s\n", __FUNCTION__, strerror(errno));
goto done;
}
cv_name_set(cv, var);
if (cv_parse(val, cv) < 0)
goto done;
}
else{ /* global */
treename_keyword = cligen_treename_keyword(h); /* typically: "treename" */
if (strcmp(var, treename_keyword) == 0){
if (cgy_treename(cy, val) < 0)
goto done;
}
else {
if ((cv = cvec_add(cy->cy_globals, CGV_STRING)) == NULL){
fprintf(stderr, "%s: realloc:%s\n", __FUNCTION__, strerror(errno));
goto done;
}
cv_name_set(cv, var);
if (cv_parse(val, cv) < 0) /* May be wrong type */
goto done;
}
}
retval = 0;
done:
return retval;
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
int
cgy_callback(cligen_yacc *cy,
char *cb_str)
{
struct cgy_stack *cs = cy->cy_stack;
struct cg_callback *cc, **ccp;
if (debug)
fprintf(stderr, "%s: %s\n", __FUNCTION__, cb_str);
if (cs == NULL)
return 0;
ccp = &cy->cy_callbacks;
while (*ccp != NULL)
ccp = &((*ccp)->cc_next);
if ((cc = malloc(sizeof(*cc))) == NULL){
fprintf(stderr, "%s: malloc: %s\n", __FUNCTION__, strerror(errno));
cligen_parseerror1(cy, "Allocating cligen callback");
return -1;
}
memset(cc, 0, sizeof(*cc));
cc->cc_fn_str = cb_str;
*ccp = cc;
return 0;
}
/*! Create a callback argument and store it in the current callback
* @param[in] cy CLIgen yacc parse struct
*/
static int
cgy_callback_arg(cligen_yacc *cy,
char *type,
char *arg)
{
int retval = -1;
struct cg_callback *cc;
struct cg_callback *cclast;
cg_var *cv = NULL;
cclast = NULL;
for (cc=cy->cy_callbacks; cc; cc=cc->cc_next)
cclast = cc;
if (cclast){
if ((cv = create_cv(cy, type, arg)) == NULL)
goto done;
if (cclast->cc_cvec)
cvec_append_var(cclast->cc_cvec, cv);
else
cclast->cc_cvec = cvec_from_var(cv);
}
retval = 0;
done:
if (cv)
cv_free(cv);
return retval;
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
static int
expand_arg(cligen_yacc *cy,
char *type,
char *arg)
{
int retval = -1;
cg_var *cv = NULL;
if ((cv = create_cv(cy, type, arg)) == NULL)
goto done;
if (cy->cy_var->co_expand_fn_vec)
cvec_append_var(cy->cy_var->co_expand_fn_vec, cv);
else
cy->cy_var->co_expand_fn_vec = cvec_from_var(cv);
retval = 0;
done:
if (cv)
cv_free(cv);
return retval;
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
static int
expand_fn(cligen_yacc *cy,
char *fn)
{
cy->cy_var->co_expand_fn_str = fn;
return 0;
}
static int
cg_translate(cligen_yacc *cy,
char *fn)
{
cy->cy_var->co_translate_fn_str = fn;
return 0;
}
static int
cgy_list_push(cg_obj *co,
struct cgy_list **cl0)
{
struct cgy_list *cl;
if (debug)
fprintf(stderr, "%s\n", __FUNCTION__);
if ((cl = malloc(sizeof(*cl))) == NULL) {
fprintf(stderr, "%s: malloc: %s\n", __FUNCTION__, strerror(errno));
return -1;
}
cl->cl_next = *cl0;
cl->cl_obj = co;
*cl0 = cl;
return 0;
}
/*! Delete whole list
* @param[in] cy CLIgen yacc parse struct
*/
static int
cgy_list_delete(struct cgy_list **cl0)
{
struct cgy_list *cl;
while ((cl = *cl0) != NULL){
*cl0 = cl->cl_next;
free(cl);
}
return 0;
}
/*! Create new tmp variable cligen object
* It must be filled in by later functions.
* The reason is, the variable must be completely parsed by successive syntax
* (eg <type:> stuff) and we cant insert it into the object tree until that is done.
* And completed by the '_post' function
* @param[in] cy CLIgen yacc parse struct
* @retval tmp variable object
* @see cgy_var_post
*/
static cg_obj *
cgy_var_create(cligen_yacc *cy)
{
cg_obj *co;
/* Create unassigned variable object */
if ((co = cov_new(CGV_ERR, NULL)) == NULL){
fprintf(stderr, "%s: malloc: %s\n", __FUNCTION__, strerror(errno));
cligen_parseerror1(cy, "Allocating cligen object");
return NULL;
}
if (debug)
fprintf(stderr, "%s: pre\n", __FUNCTION__);
return co;
}
/*! Set name and type on a (previously created) variable
* @param[in] cy CLIgen yacc parse struct
* @see cgy_var_create
*/
static int
cgy_var_name_type(cligen_yacc *cy,
char *name,
char *type)
{
cy->cy_var->co_command = name;
if ((cy->cy_var->co_vtype = cv_str2type(type)) == CGV_ERR){
cligen_parseerror1(cy, "Invalid type");
fprintf(stderr, "%s: Invalid type: %s\n", __FUNCTION__, type);
return -1;
}
return 0;
}
/*! Complete variable cligen object after parsing is complete,
* And insert it into object hierarchies.
* That is, insert a variable in each hieracrhy.
* @param[in] cy CLIgen yacc parse struct
* @retval 0 on OK
* @retval -1 on error
*/
static int
cgy_var_post(cligen_yacc *cy)
{
struct cgy_list *cl;
cg_obj *coc = NULL; /* variable copy object */
cg_obj *coparent; /* parent */
cg_obj *co; /* new obj/sister */
cg_obj *coy = cy->cy_var;
#if 0
if (coy->co_vtype == CGV_ERR) /* unassigned */
coy->co_vtype = cv_str2type(coy->co_command);
#endif
if (debug)
fprintf(stderr, "%s: cmd:%s vtype:%d\n", __FUNCTION__,
coy->co_command,
coy->co_vtype );
if (coy->co_vtype == CGV_ERR){
cligen_parseerror1(cy, "Wrong or unassigned variable type");
return -1;
}
#if 0 /* XXX dont really know what i am doing but variables dont behave nice in choice */
if (cy->cy_opt){ /* get coparent from stack */
if (cy->cy_stack == NULL){
fprintf(stderr, "Option allowed only within () or []\n");
return -1;
}
cl = cy->cy_stack->cs_list;
}
else
#endif
cl = cy->cy_list;
for (; cl; cl = cl->cl_next){
coparent = cl->cl_obj;
if (cl->cl_next){
if (co_copy(coy, coparent, &coc) < 0) /* duplicate coy to coc */
return -1;
}
else
coc = coy; /* Dont copy if last in list */
co_up_set(coc, coparent);
if ((co = co_insert(co_pt_get(coparent), coc)) == NULL) /* coc may be deleted */
return -1;
cl->cl_obj = co;
}
return 0;
}
/*! Create a new command object.
* Actually, create a new for every tree in the list
* and replace the old with the new object.
* @param[in] cy CLIgen yacc parse struct
* @param[in] cmd the command string
* @retval 0 OK
* @retval -1 Error
*/
static int
cgy_cmd(cligen_yacc *cy,
char *cmd)
{
struct cgy_list *cl;
cg_obj *cop; /* parent */
cg_obj *conew; /* new obj */
cg_obj *co; /* new/sister */
for (cl=cy->cy_list; cl; cl = cl->cl_next){
cop = cl->cl_obj;
if (debug)
fprintf(stderr, "%s: %s parent:%s\n",
__FUNCTION__, cmd, cop->co_command);
if ((conew = co_new(cmd, cop)) == NULL) {
cligen_parseerror1(cy, "Allocating cligen object");
return -1;
}
if ((co = co_insert(co_pt_get(cop), conew)) == NULL) /* co_new may be deleted */
return -1;
cl->cl_obj = co; /* Replace parent in cgy_list */
}
return 0;
}
/*! Create a REFERENCE node that references another tree.
* This is evaluated in runtime by pt_expand().
* @param[in] cy CLIgen yacc parse struct
* @see also db2tree() in clicon/apps/cli_main.c on how to create such a tree
* @see pt_expand_treeref()/pt_callback_reference() how it is expanded
*/
static int
cgy_reference(cligen_yacc *cy,
char *name)
{
struct cgy_list *cl;
cg_obj *cop; /* parent */
cg_obj *cot; /* tree */
for (cl=cy->cy_list; cl; cl = cl->cl_next){
/* Add a treeref 'stub' which is expanded in pt_expand to a sub-tree */
cop = cl->cl_obj;
if ((cot = co_new(name, cop)) == NULL) {
cligen_parseerror1(cy, "Allocating cligen object");
return -1;
}
cot->co_type = CO_REFERENCE;
if ((cot = co_insert(co_pt_get(cop), cot)) == NULL) /* cot may be deleted */
return -1;
/* Replace parent in cgy_list: not allowed after ref?
but only way to add callbacks to it.
*/
cl->cl_obj = cot;
}
return 0;
}
/*! Add comment
* Assume comment is malloced and not freed by parser
* @param[in] cy CLIgen yacc parse struct
* @param[in] comment Help string, inside ("...")
*/
static int
cgy_comment(cligen_yacc *cy,
char *comment)
{
struct cgy_list *cl;
cg_obj *co;
for (cl = cy->cy_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (co->co_helpvec == NULL && /* Why would it already have a comment? */
cligen_txt2cvv(comment, &co->co_helpvec) < 0){ /* Or just append to existing? */
cligen_parseerror1(cy, "Allocating comment");
return -1;
}
}
return 0;
}
/*! Append a new choice option to a choice variable string
* @param[in] cy CLIgen yacc parse struct
* @param[in] str Accumulated choice string on the form: "a|..|z"
* @param[in] app Choice option to append
* @retval s New string created by appending: "<str>|<app>"
* This is just string manipulation, the complete string is linked into CLIgen structs by upper rule
* Note that this is variable choice. There is also command choice, eg [a|b] which is different.
*/
static char *
cgy_var_choice_append(cligen_yacc *cy,
char *str,
char *app)
{
int len;
char *s;
len = strlen(str)+strlen(app) + 2;
if ((s = realloc(str, len)) == NULL) {
fprintf(stderr, "%s: realloc: %s\n", __FUNCTION__, strerror(errno));
return NULL;
}
strncat(s, "|", len-1);
strncat(s, app, len-1);
return s;
}
/*! Post-processing of commands, eg at ';':
* a cmd;<--
* But only if parsing succesful.
* 1. Add callback and args to every list
* 2. Add empty child unless already empty child
* @param[in] cy CLIgen yacc parse struct
*/
int
cgy_terminal(cligen_yacc *cy)
{
struct cgy_list *cl;
cg_obj *co;
int i;
struct cg_callback *cc;
struct cg_callback **ccp;
int retval = -1;
parse_tree *ptc;
for (cl = cy->cy_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (cy->cy_callbacks){ /* callbacks */
ccp = &co->co_callbacks;
while (*ccp != NULL)
ccp = &((*ccp)->cc_next);
if (co_callback_copy(cy->cy_callbacks, ccp) < 0)
goto done;
}
/* variables: special case "hide" */
if (cy->cy_cvec){
if (cvec_find(cy->cy_cvec, "hide") != NULL)
co_flags_set(co, CO_FLAGS_HIDE);
/* generic variables */
if ((co->co_cvec = cvec_dup(cy->cy_cvec)) == NULL){
fprintf(stderr, "%s: cvec_dup: %s\n", __FUNCTION__, strerror(errno));
goto done;
}
}
/* misc */
if ((ptc = co_pt_get(co)) != NULL){
for (i=0; i<pt_len_get(ptc); i++){
if (pt_vec_i_get(ptc, i) == NULL)
break;
}
if (i == pt_len_get(ptc)) /* Insert empty child if ';' */
co_insert(co_pt_get(co), NULL);
}
else
co_insert(co_pt_get(co), NULL);
}
/* cleanup */
while ((cc = cy->cy_callbacks) != NULL){
if (cc->cc_cvec)
cvec_free(cc->cc_cvec);
if (cc->cc_fn_str)
free(cc->cc_fn_str);
cy->cy_callbacks = cc->cc_next;
free(cc);
}
if (cy->cy_cvec){
cvec_free(cy->cy_cvec);
cy->cy_cvec = NULL;
}
retval = 0;
done:
return retval;
}
/*! Take the whole cgy_list and push it to the stack
* @param[in] cy CLIgen yacc parse struct
*/
static int
ctx_push(cligen_yacc *cy,
int sets)
{
struct cgy_list *cl;
struct cgy_stack *cs;
cg_obj *co;
if (debug)
fprintf(stderr, "%s sets:%d\n", __FUNCTION__, sets);
/* Create new stack element */
if ((cs = malloc(sizeof(*cs))) == NULL) {
fprintf(stderr, "%s: malloc: %s\n", __FUNCTION__, strerror(errno));
return -1;
}
memset(cs, 0, sizeof(*cs));
cs->cs_next = cy->cy_stack;
cy->cy_stack = cs; /* Push the new stack element */
for (cl = cy->cy_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (cvec_find(cy->cy_cvec, "hide") != NULL)
co_flags_set(co, CO_FLAGS_HIDE);
if (sets)
co_sets_set(co, 1);
if (cgy_list_push(co, &cs->cs_list) < 0)
return -1;
}
return 0;
}
/*! Peek context from stack and replace the object list with it
* Typically done in a command choice, eg "(c1|c2)" at c2.
* Dont pop context
* @param[in] cy CLIgen yacc parse struct
* @see ctx_peek_swap2
*/
static int
ctx_peek_swap(cligen_yacc *cy)
{
struct cgy_stack *cs;
struct cgy_list *cl;
cg_obj *co;
if (debug)
fprintf(stderr, "%s\n", __FUNCTION__);
if ((cs = cy->cy_stack) == NULL){
#if 1
cligen_parseerror1(cy, "No surrounding () or []");
return -1; /* e.g a|b instead of (a|b) */
#else
cgy_list_delete(&cy->cy_list);
return 0;
#endif
}
for (cl = cy->cy_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
co_flags_set(co, CO_FLAGS_OPTION);
if (cgy_list_push(co, &cs->cs_saved) < 0)
return -1;
}
cgy_list_delete(&cy->cy_list);
for (cl = cs->cs_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (cgy_list_push(co, &cy->cy_list) < 0)
return -1;
}
return 0;
}
/*! Peek context from stack and replace the object list with it
* Typically done in a command choice, eg "(c1|c2)" at c2.
* Dont pop context
* @param[in] cy CLIgen yacc parse struct
* @see ctx_peek_swap
*/
static int
ctx_peek_swap2(cligen_yacc *cy)
{
struct cgy_stack *cs;
struct cgy_list *cl;
cg_obj *co;
if (debug)
fprintf(stderr, "%s\n", __FUNCTION__);
if ((cs = cy->cy_stack) == NULL){
#if 1
cligen_parseerror1(cy, "No surrounding () or []");
return -1; /* e.g a|b instead of (a|b) */
#else
cgy_list_delete(&cy->cy_list);
return 0;
#endif
}
cgy_list_delete(&cy->cy_list);
for (cl = cs->cs_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (cgy_list_push(co, &cy->cy_list) < 0)
return -1;
}
return 0;
}
static int
delete_stack_element(struct cgy_stack *cs)
{
cgy_list_delete(&cs->cs_list);
cgy_list_delete(&cs->cs_saved);
free(cs);
return 0;
}
/*! Pop context from stack and add it to object list
* Done after an option, eg "cmd [opt]"
* "cmd <push> [opt] <pop>". At pop, all objects saved at push
* are added to the object list.
* @param[in] cy CLIgen yacc parse struct
*/
static int
ctx_pop_add(cligen_yacc *cy)
{
struct cgy_stack *cs;
struct cgy_list *cl;
cg_obj *co;
if (debug)
fprintf(stderr, "%s\n", __FUNCTION__);
for (cl = cy->cy_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
co_flags_set(co, CO_FLAGS_OPTION);
}
if ((cs = cy->cy_stack) == NULL){
fprintf(stderr, "%s: cgy_stack empty\n", __FUNCTION__);
return -1; /* shouldnt happen */
}
for (cl = cs->cs_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
}
cy->cy_stack = cs->cs_next;
/* We could have saved some heap work by moving the cs_list,... */
for (cl = cs->cs_list; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (cgy_list_push(co, &cy->cy_list) < 0)
return -1;
}
for (cl = cs->cs_saved; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (cgy_list_push(co, &cy->cy_list) < 0)
return -1;
}
delete_stack_element(cs);
return 0;
}
/*! Pop context from stack and discard it.
* Typically done after a grouping, eg "cmd (opt1|opt2)"
* @param[in] cy CLIgen yacc parse struct
*/
static int
ctx_pop(cligen_yacc *cy)
{
struct cgy_stack *cs;
struct cgy_list *cl;
cg_obj *co;
if (debug)
fprintf(stderr, "%s\n", __FUNCTION__);
if ((cs = cy->cy_stack) == NULL){
fprintf(stderr, "%s: cgy_stack empty\n", __FUNCTION__);
return -1; /* shouldnt happen */
}
for (cl = cs->cs_list; cl; cl = cl->cl_next)
co = cl->cl_obj;
cy->cy_stack = cs->cs_next;
for (cl = cs->cs_saved; cl; cl = cl->cl_next){
co = cl->cl_obj;
if (cgy_list_push(co, &cy->cy_list) < 0)
return -1;
}
delete_stack_element(cs);
return 0;
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
static int
cg_regexp(cligen_yacc *cy,
char *rx,
int invert)
{
int retval = -1;
cg_var *cv;
if (cy->cy_var->co_regex == NULL &&
(cy->cy_var->co_regex = cvec_new(0)) == NULL)
goto done;
if ((cv = cvec_add(cy->cy_var->co_regex, CGV_STRING)) == NULL)
goto done;
if (invert)
cv_flag_set(cv, V_INVERT);
cv_string_set(cv, rx);
if (cy->cy_var->co_vtype != CGV_STRING && cy->cy_var->co_vtype != CGV_REST)
cy->cy_var->co_vtype = CGV_STRING;
retval = 0;
done:
return retval;
}
/*! Given an optional min and a max, create low and upper bounds on cv values
*
* @param[in] cy CLIgen yacc parse struct
* @param[in] lowstr low bound of interval (can be NULL)
* @param[in] uppstr upper bound of interval
* @param[in] yv The CLIgen syntax object
* @param[in] cvtype Type of variable
* Supported for ints, decimal64 and strings.
* <x:int length[min:max]> or <x:int length[max]>
* @note: decimal64 fraction-digits must be given before range:
* <x:decimal64 fraction-digits:4 range[20.0]>
* if you want any other fraction-digit than 2
*/
static int
cg_range_create(cligen_yacc *cy,
char *lowstr,
char *uppstr,
cg_obj *yv,
enum cv_type cvtype)
{
int retval = -1;
char *reason = NULL;
cg_var *cv1 = NULL;
cg_var *cv2 = NULL;
int cvret;
/* First create low bound cv */
if ((cv1 = cv_new(cvtype)) == NULL){
fprintf(stderr, "cv_new %s\n", strerror(errno));
goto done;
}
if (cv_name_set(cv1, "range_low") == NULL){
fprintf(stderr, "cv_name_set %s\n", strerror(errno));
goto done;
}
if (lowstr == NULL){
cv_type_set(cv1, CGV_EMPTY);
}
else{
if (yv->co_vtype == CGV_DEC64) /* XXX: Seems misplaced? / too specific */
cv_dec64_n_set(cv1, yv->co_dec64_n);
if ((cvret = cv_parse1(lowstr, cv1, &reason)) < 0){
fprintf(stderr, "cv_parse1 %s\n", strerror(errno));
goto done;
}
if (cvret == 0){ /* parsing failed */
cligen_parseerror1(cy, reason);
free(reason);
goto done;
}
}
/* Then append it to the lowbound cvec, create if NULL */
if (yv->co_rangecvv_low == NULL){
if ((yv->co_rangecvv_low = cvec_from_var(cv1)) == NULL)
goto done;
}
else if (cvec_append_var(yv->co_rangecvv_low, cv1) < 0)
goto done;
/* Then create upper bound cv */
if ((cv2 = cv_new(cvtype)) == NULL){
fprintf(stderr, "cv_new %s\n", strerror(errno));
goto done;
}
if (cv_name_set(cv2, "range_high") == NULL){
fprintf(stderr, "cv_name_set %s\n", strerror(errno));
goto done;
}
if (yv->co_vtype == CGV_DEC64) /* XXX: Seems misplaced? / too specific */
cv_dec64_n_set(cv2, yv->co_dec64_n);
if ((cvret = cv_parse1(uppstr, cv2, &reason)) < 0){
fprintf(stderr, "cv_parse1 %s\n", strerror(errno));
goto done;
}
if (cvret == 0){ /* parsing failed */
cligen_parseerror1(cy, reason);
free(reason);
goto done;
}
/* Append it to the upper bound cvec, create if NULL */
if (yv->co_rangecvv_upp == NULL){
if ((yv->co_rangecvv_upp = cvec_from_var(cv2)) == NULL)
goto done;
}
else if (cvec_append_var(yv->co_rangecvv_upp, cv2) < 0)
goto done;
/* Then increment range vector length */
yv->co_rangelen++;
retval = 0;
done:
if (cv1)
cv_free(cv1);
if (cv2)
cv_free(cv2);
return retval;
}
/*! A length statement has been parsed. Create length/range cv
*
* @param[in] cy CLIgen yacc parse struct
* @param[in] lowstr low bound of interval (can be NULL)
* @param[in] uppstr upper bound of interval
* Examples:
* <x:string length[min:max]> <x:string length[max]>
* @note that the co_range structure fields are re-used for string length restrictions.
* but the range type is uint64, not depending on cv type as int:s
* @see cg_range
*/
static int
cg_length(cligen_yacc *cy,
char *lowstr,
char *uppstr)
{
cg_obj *yv;
if ((yv = cy->cy_var) == NULL){
fprintf(stderr, "No var obj");
return -1;
}
return cg_range_create(cy, lowstr, uppstr, yv, CGV_UINT64);
}
/*! A range statement has been parsed. Create range cv
* @param[in] cy CLIgen yacc parse struct
* @param[in] lowstr low bound of interval (can be NULL)
* @param[in] uppstr upper bound of interval
* Examples:
* <x:int32 range[min:max]> <x:int32 range[max]>
* @see cg_length
*/
static int
cg_range(cligen_yacc *cy,
char *lowstr,
char *uppstr)
{
cg_obj *yv;
if ((yv = cy->cy_var) == NULL){
fprintf(stderr, "No var obj");
return -1;
}
return cg_range_create(cy, lowstr, uppstr, yv, yv->co_vtype);
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
static int
cg_dec64_n(cligen_yacc *cy,
char *fraction_digits)
{
cg_obj *yv;
char *reason = NULL;
if ((yv = cy->cy_var) == NULL){
fprintf(stderr, "No var obj");
return -1;
}
if (parse_uint8(fraction_digits, &yv->co_dec64_n, NULL) != 1){
cligen_parseerror1(cy, reason);
return -1;
}
return 0;
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
int
cgy_init(cligen_yacc *cy,
cg_obj *co_top)
{
if (debug)
fprintf(stderr, "%s\n", __FUNCTION__);
/* Add top-level object */
if (cgy_list_push(co_top, &cy->cy_list) < 0)
return -1;
if (ctx_push(cy, 0) < 0)
return -1;
return 0;
}
/*!
* @param[in] cy CLIgen yacc parse struct
*/
int
cgy_exit(cligen_yacc *cy)
{
struct cgy_stack *cs;
if (debug)
fprintf(stderr, "%s\n", __FUNCTION__);
cy->cy_var = NULL;
cgy_list_delete(&cy->cy_list);
if((cs = cy->cy_stack) != NULL){
delete_stack_element(cs);
#if 0
fprintf(stderr, "%s:%d: error: lacking () or [] at or before: '%s'\n",
cy->cy_name,
cy->cy_linenum,
cy->cy_parse_string
);
return -1;
#endif
}
return 0;
}
%}
%%
file : lines MY_EOF{_PARSE_DEBUG("file->lines"); YYACCEPT;}
;
lines : lines line { _PARSE_DEBUG("lines->lines line"); }
| { _PARSE_DEBUG("lines->"); }
;
line : decltop line1 { _PARSE_DEBUG("line->decltop line1"); }
| assignment ';' { _PARSE_DEBUG("line->assignment ;"); }
;
line1 : line2 { _PARSE_DEBUG("line1->line2"); }
| ',' options line2 { _PARSE_DEBUG("line1->',' options line2"); }
;
preline : '{' { $$ = 0; }
| SETS { $$ = 1; } /* SETS */
;
line2 : ';' {
_PARSE_DEBUG("line2->';'");
if (cgy_terminal(_cy) < 0) YYERROR;
if (ctx_peek_swap2(_cy) < 0) YYERROR;
}
| preline {
if (ctx_push(_cy, $1) < 0) YYERROR;
}
lines
'}' {
_PARSE_DEBUG("line2->'{' lines '}'");
if (ctx_pop(_cy) < 0) YYERROR;
if (ctx_peek_swap2(_cy) < 0) YYERROR;
}
| ';'
preline {
if (cgy_terminal(_cy) < 0) YYERROR;
if (ctx_push(_cy, $2) < 0) YYERROR;
}
lines
'}' { _PARSE_DEBUG("line2->';' '{' lines '}'");
if (ctx_pop(_cy) < 0) YYERROR;
if (ctx_peek_swap2(_cy) < 0) YYERROR; }
;
options : options ',' option {_PARSE_DEBUG("options->options , option");}
| option {_PARSE_DEBUG("options->option");}
;
option : callback { _PARSE_DEBUG("option->callback");}
| flag { _PARSE_DEBUG("option->flag");}
| assignment { _PARSE_DEBUG("option->assignment");}
;
assignment : NAME '=' DQ charseq DQ {_PARSE_DEBUG("assignment->\" charseq \"");
cgy_assignment(_cy, $1,$4);free($1); free($4);}
;
flag : NAME {cgy_flag(_cy, $1);free($1);}
;
callback : NAME {if (cgy_callback(_cy, $1) < 0) YYERROR;} '(' arglist ')'
;
arglist : arglist1
|
;
arglist1 : arglist1 ',' arg
| arg
;
arg : typecast arg1 {
if ($2 && cgy_callback_arg(_cy, $1, $2) < 0) YYERROR;
if ($1 != NULL) free($1);
if ($2 != NULL) free($2);
}
;
arg1 : DQ DQ { $$=NULL; }
| DQ charseq DQ { $$=$2; }
| NAME { $$=$1; }
;
typecast : '(' NAME ')' { $$ = $2; }
| { $$ = NULL; }
;
decltop : decllist { _PARSE_DEBUG("decltop->decllist");}
| declcomp { _PARSE_DEBUG("decltop->declcomp");}
;
decllist : decltop
declcomp { _PARSE_DEBUG("decllist->decltop declcomp");}
| decltop '|' { if (ctx_peek_swap(_cy) < 0) YYERROR;}
declcomp { _PARSE_DEBUG("decllist->decltop | declcomp");}
;
declcomp : '(' { if (ctx_push(_cy, 0) < 0) YYERROR; }
decltop ')' { if (ctx_pop(_cy) < 0) YYERROR;
_PARSE_DEBUG("declcomp->(decltop)");}
| '[' { if (ctx_push(_cy, 0) < 0) YYERROR; }
decltop ']' { if (ctx_pop_add(_cy) < 0) YYERROR; } {
_PARSE_DEBUG("declcomp->[decltop]");}
| decl { _PARSE_DEBUG("declcomp->decl");}
;
decl : cmd { _PARSE_DEBUG("decl->cmd");}
| cmd PDQ charseq DQP { _PARSE_DEBUG("decl->cmd (\" comment \")");
if (cgy_comment(_cy, $3) < 0) YYERROR; free($3);}
| cmd PDQ DQP { _PARSE_DEBUG("decl->cmd (\"\")");}
;
cmd : NAME { _PARSE_DEBUG("cmd->NAME");
if (cgy_cmd(_cy, $1) < 0) YYERROR; free($1); }
| '@' NAME { _PARSE_DEBUG("cmd->@NAME");
if (cgy_reference(_cy, $2) < 0) YYERROR; free($2); }
| '<' { if ((_CY->cy_var = cgy_var_create(_CY)) == NULL) YYERROR; }
variable '>' { if (cgy_var_post(_cy) < 0) YYERROR; }
;
variable : NAME { if (cgy_var_name_type(_cy, $1, $1)<0) YYERROR; }
| NAME ':' NAME { if (cgy_var_name_type(_cy, $1, $3)<0) YYERROR; free($3); }
| NAME ' ' { if (cgy_var_name_type(_cy, $1, $1) < 0) YYERROR; }
keypairs
| NAME ':' NAME ' ' { if (cgy_var_name_type(_cy, $1, $3) < 0) YYERROR; free($3); }
keypairs
;
keypairs : keypair
| keypairs ' ' keypair
;
numdec : NUMBER { $$ = $1; }
| DECIMAL
;
keypair : NAME '(' ')' { expand_fn(_cy, $1); }
| NAME '(' exparglist ')' { expand_fn(_cy, $1); }
| V_SHOW ':' NAME {
_CY->cy_var->co_show = $3;
}
| V_SHOW ':' DQ charseq DQ {
_CY->cy_var->co_show = $4;
}
| V_RANGE '[' numdec ':' numdec ']' {
if (cg_range(_cy, $3, $5) < 0) YYERROR; free($3); free($5);
}
| V_RANGE '[' numdec ']' {
if (cg_range(_cy, NULL, $3) < 0) YYERROR; free($3);
}
| V_LENGTH '[' NUMBER ':' NUMBER ']' {
if (cg_length(_cy, $3, $5) < 0) YYERROR; free($3); free($5);
}
| V_LENGTH '[' NUMBER ']' {
if (cg_length(_cy, NULL, $3) < 0) YYERROR; free($3);
}
| V_FRACTION_DIGITS ':' NUMBER {
if (cg_dec64_n(_cy, $3) < 0) YYERROR; free($3);
}
| V_CHOICE choices { _CY->cy_var->co_choice = $2; }
| V_KEYWORD ':' NAME {
_CY->cy_var->co_keyword = $3;
_CY->cy_var->co_vtype=CGV_STRING;
}
| V_REGEXP ':' DQ charseq DQ { if (cg_regexp(_cy, $4, 0) < 0) YYERROR; free($4); }
| V_REGEXP ':' '!' DQ charseq DQ { if (cg_regexp(_cy, $5, 1) < 0) YYERROR; free($5);}
| V_TRANSLATE ':' NAME '(' ')' { cg_translate(_cy, $3); }
;
exparglist : exparglist ',' exparg
| exparg
;
exparg : DQ DQ
| DQ charseq DQ { expand_arg(_cy, "string", $2); free($2); }
;
exparg : typecast arg1 {
if ($2 && cgy_callback_arg(_cy, $1, $2) < 0) YYERROR;
if ($1) free($1);
if ($2) free($2);
}
;
choices : { $$ = NULL;}
| NUMBER { $$ = $1;}
| NAME { $$ = $1;}
| DECIMAL { $$ = $1;}
| choices '|' NUMBER { $$ = cgy_var_choice_append(_cy, $1, $3); free($3);}
| choices '|' NAME { $$ = cgy_var_choice_append(_cy, $1, $3); free($3);}
| choices '|' DECIMAL { $$ = cgy_var_choice_append(_cy, $1, $3); free($3);}
;
charseq : charseq CHARS
{
int len = strlen($1);
$$ = realloc($1, len+strlen($2) +1);
sprintf($$+len, "%s", $2);
free($2);
}
| CHARS {$$=$1;}
;
%%
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。