mirror of https://github.com/Icinga/icinga2.git
564 lines
11 KiB
Plaintext
564 lines
11 KiB
Plaintext
%{
|
|
/******************************************************************************
|
|
* Icinga 2 *
|
|
* Copyright (C) 2012-2018 Icinga Development Team (https://www.icinga.com/) *
|
|
* *
|
|
* This program is free software; you can redistribute it and/or *
|
|
* modify it under the terms of the GNU General Public License *
|
|
* as published by the Free Software Foundation; either version 2 *
|
|
* of the License, or (at your option) any later version. *
|
|
* *
|
|
* This program is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU General Public License *
|
|
* along with this program; if not, write to the Free Software Foundation *
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
******************************************************************************/
|
|
|
|
#include "classcompiler.hpp"
|
|
#include <iostream>
|
|
#include <vector>
|
|
#include <cstring>
|
|
|
|
using std::malloc;
|
|
using std::free;
|
|
using std::exit;
|
|
|
|
using namespace icinga;
|
|
|
|
#define YYLTYPE icinga::ClassDebugInfo
|
|
|
|
%}
|
|
|
|
%pure-parser
|
|
|
|
%locations
|
|
%defines
|
|
%error-verbose
|
|
|
|
%parse-param { ClassCompiler *context }
|
|
%lex-param { void *scanner }
|
|
|
|
%union {
|
|
char *text;
|
|
int num;
|
|
FieldType *type;
|
|
Field *field;
|
|
std::vector<Field> *fields;
|
|
Klass *klass;
|
|
FieldAccessor *fieldaccessor;
|
|
std::vector<FieldAccessor> *fieldaccessors;
|
|
Rule *rule;
|
|
std::vector<Rule> *rules;
|
|
Validator *validator;
|
|
}
|
|
|
|
%token T_INCLUDE "#include (T_INCLUDE)"
|
|
%token T_IMPL_INCLUDE "#impl_include (T_IMPL_INCLUDE)"
|
|
%token T_CLASS "class (T_CLASS)"
|
|
%token T_CODE "code (T_CODE)"
|
|
%token T_LOAD_AFTER "load_after (T_LOAD_AFTER)"
|
|
%token T_LIBRARY "library (T_LIBRARY)"
|
|
%token T_NAMESPACE "namespace (T_NAMESPACE)"
|
|
%token T_VALIDATOR "validator (T_VALIDATOR)"
|
|
%token T_REQUIRED "required (T_REQUIRED)"
|
|
%token T_NAVIGATION "navigation (T_NAVIGATION)"
|
|
%token T_NAME "name (T_NAME)"
|
|
%token T_ARRAY "array (T_ARRAY)"
|
|
%token T_STRING "string (T_STRING)"
|
|
%token T_ANGLE_STRING "angle_string (T_ANGLE_STRING)"
|
|
%token T_FIELD_ATTRIBUTE "field_attribute (T_FIELD_ATTRIBUTE)"
|
|
%token T_CLASS_ATTRIBUTE "class_attribute (T_CLASS_ATTRIBUTE)"
|
|
%token T_IDENTIFIER "identifier (T_IDENTIFIER)"
|
|
%token T_GET "get (T_GET)"
|
|
%token T_SET "set (T_SET)"
|
|
%token T_DEFAULT "default (T_DEFAULT)"
|
|
%token T_FIELD_ACCESSOR_TYPE "field_accessor_type (T_FIELD_ACCESSOR_TYPE)"
|
|
%type <text> T_IDENTIFIER
|
|
%type <text> T_STRING
|
|
%type <text> T_ANGLE_STRING
|
|
%type <text> identifier
|
|
%type <text> alternative_name_specifier
|
|
%type <text> inherits_specifier
|
|
%type <text> type_base_specifier
|
|
%type <text> include
|
|
%type <text> angle_include
|
|
%type <text> impl_include
|
|
%type <text> angle_impl_include
|
|
%type <text> code
|
|
%type <num> T_FIELD_ATTRIBUTE
|
|
%type <field> field_attribute
|
|
%type <field> field_attributes
|
|
%type <field> field_attribute_list
|
|
%type <num> T_FIELD_ACCESSOR_TYPE
|
|
%type <num> T_CLASS_ATTRIBUTE
|
|
%type <num> class_attribute_list
|
|
%type <type> field_type
|
|
%type <field> class_field
|
|
%type <fields> class_fields
|
|
%type <klass> class
|
|
%type <fieldaccessors> field_accessor_list
|
|
%type <fieldaccessors> field_accessors
|
|
%type <fieldaccessor> field_accessor
|
|
%type <rule> validator_rule
|
|
%type <rules> validator_rules
|
|
%type <validator> validator
|
|
|
|
%{
|
|
|
|
int yylex(YYSTYPE *lvalp, YYLTYPE *llocp, void *scanner);
|
|
|
|
void yyerror(YYLTYPE *locp, ClassCompiler *, const char *err)
|
|
{
|
|
std::cerr << "in " << locp->path << " at " << locp->first_line << ":" << locp->first_column << "-"
|
|
<< locp->last_line << ":" << locp->last_column << ": " << err << std::endl;
|
|
std::exit(1);
|
|
}
|
|
|
|
int yyparse(ClassCompiler *context);
|
|
|
|
void ClassCompiler::Compile(void)
|
|
{
|
|
try {
|
|
yyparse(this);
|
|
} catch (const std::exception& ex) {
|
|
std::cerr << "Exception: " << ex.what();
|
|
}
|
|
|
|
HandleMissingValidators();
|
|
}
|
|
|
|
#define scanner (context->GetScanner())
|
|
|
|
%}
|
|
|
|
%%
|
|
|
|
statements: /* empty */
|
|
| statements statement
|
|
;
|
|
|
|
statement: include
|
|
{
|
|
context->HandleInclude($1, yylloc);
|
|
std::free($1);
|
|
}
|
|
| angle_include
|
|
{
|
|
context->HandleAngleInclude($1, yylloc);
|
|
std::free($1);
|
|
}
|
|
| impl_include
|
|
{
|
|
context->HandleImplInclude($1, yylloc);
|
|
std::free($1);
|
|
}
|
|
| angle_impl_include
|
|
{
|
|
context->HandleAngleImplInclude($1, yylloc);
|
|
std::free($1);
|
|
}
|
|
| class
|
|
{
|
|
context->HandleClass(*$1, yylloc);
|
|
delete $1;
|
|
}
|
|
| validator
|
|
{
|
|
context->HandleValidator(*$1, yylloc);
|
|
delete $1;
|
|
}
|
|
| namespace
|
|
| code
|
|
{
|
|
context->HandleCode($1, yylloc);
|
|
std::free($1);
|
|
}
|
|
| library
|
|
;
|
|
|
|
include: T_INCLUDE T_STRING
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
angle_include: T_INCLUDE T_ANGLE_STRING
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
impl_include: T_IMPL_INCLUDE T_STRING
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
angle_impl_include: T_IMPL_INCLUDE T_ANGLE_STRING
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
namespace: T_NAMESPACE identifier '{'
|
|
{
|
|
context->HandleNamespaceBegin($2, yylloc);
|
|
std::free($2);
|
|
}
|
|
statements '}'
|
|
{
|
|
context->HandleNamespaceEnd(yylloc);
|
|
}
|
|
;
|
|
|
|
code: T_CODE T_STRING
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
library: T_LIBRARY T_IDENTIFIER ';'
|
|
{
|
|
context->HandleLibrary($2, yylloc);
|
|
free($2);
|
|
}
|
|
;
|
|
|
|
class: class_attribute_list T_CLASS T_IDENTIFIER inherits_specifier type_base_specifier '{' class_fields '}' ';'
|
|
{
|
|
$$ = new Klass();
|
|
|
|
$$->Name = $3;
|
|
std::free($3);
|
|
|
|
if ($4) {
|
|
$$->Parent = $4;
|
|
std::free($4);
|
|
}
|
|
|
|
if ($5) {
|
|
$$->TypeBase = $5;
|
|
std::free($5);
|
|
}
|
|
|
|
$$->Attributes = $1;
|
|
|
|
for (const Field& field : *$7) {
|
|
if (field.Attributes & FALoadDependency) {
|
|
$$->LoadDependencies.push_back(field.Name);
|
|
} else
|
|
$$->Fields.push_back(field);
|
|
}
|
|
|
|
delete $7;
|
|
|
|
ClassCompiler::OptimizeStructLayout($$->Fields);
|
|
}
|
|
;
|
|
|
|
class_attribute_list: /* empty */
|
|
{
|
|
$$ = 0;
|
|
}
|
|
| T_CLASS_ATTRIBUTE
|
|
{
|
|
$$ = $1;
|
|
}
|
|
| class_attribute_list T_CLASS_ATTRIBUTE
|
|
{
|
|
$$ = $1 | $2;
|
|
}
|
|
|
|
inherits_specifier: /* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| ':' identifier
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
type_base_specifier: /* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| '<' identifier
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
class_fields: /* empty */
|
|
{
|
|
$$ = new std::vector<Field>();
|
|
}
|
|
| class_fields class_field
|
|
{
|
|
$$->push_back(*$2);
|
|
delete $2;
|
|
}
|
|
;
|
|
|
|
field_type: identifier
|
|
{
|
|
$$ = new FieldType();
|
|
$$->IsName = false;
|
|
$$->TypeName = $1;
|
|
free($1);
|
|
}
|
|
| T_NAME '(' identifier ')'
|
|
{
|
|
$$ = new FieldType();
|
|
$$->IsName = true;
|
|
$$->TypeName = $3;
|
|
$$->ArrayRank = 0;
|
|
free($3);
|
|
}
|
|
| T_ARRAY '(' field_type ')'
|
|
{
|
|
$$ = $3;
|
|
$$->ArrayRank++;
|
|
}
|
|
;
|
|
|
|
class_field: field_attribute_list field_type identifier alternative_name_specifier field_accessor_list ';'
|
|
{
|
|
Field *field = $1;
|
|
|
|
if ((field->Attributes & (FAConfig | FAState)) == 0)
|
|
field->Attributes |= FAEphemeral;
|
|
|
|
field->Type = *$2;
|
|
delete $2;
|
|
|
|
field->Name = $3;
|
|
std::free($3);
|
|
|
|
if ($4) {
|
|
field->AlternativeName = $4;
|
|
std::free($4);
|
|
}
|
|
|
|
std::vector<FieldAccessor>::const_iterator it;
|
|
for (it = $5->begin(); it != $5->end(); it++) {
|
|
switch (it->Type) {
|
|
case FTGet:
|
|
field->GetAccessor = it->Accessor;
|
|
field->PureGetAccessor = it->Pure;
|
|
break;
|
|
case FTSet:
|
|
field->SetAccessor = it->Accessor;
|
|
field->PureSetAccessor = it->Pure;
|
|
break;
|
|
case FTDefault:
|
|
field->DefaultAccessor = it->Accessor;
|
|
break;
|
|
case FTTrack:
|
|
field->TrackAccessor = it->Accessor;
|
|
break;
|
|
case FTNavigate:
|
|
field->NavigateAccessor = it->Accessor;
|
|
field->PureNavigateAccessor = it->Pure;
|
|
break;
|
|
}
|
|
}
|
|
|
|
delete $5;
|
|
|
|
$$ = field;
|
|
}
|
|
| T_LOAD_AFTER identifier ';'
|
|
{
|
|
Field *field = new Field();
|
|
field->Attributes = FALoadDependency;
|
|
field->Name = $2;
|
|
std::free($2);
|
|
$$ = field;
|
|
}
|
|
;
|
|
|
|
alternative_name_specifier: /* empty */
|
|
{
|
|
$$ = NULL;
|
|
}
|
|
| '(' identifier ')'
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
field_attribute_list: /* empty */
|
|
{
|
|
$$ = new Field();
|
|
}
|
|
| '[' field_attributes ']'
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
field_attribute: T_FIELD_ATTRIBUTE
|
|
{
|
|
$$ = new Field();
|
|
$$->Attributes = $1;
|
|
}
|
|
| T_REQUIRED
|
|
{
|
|
$$ = new Field();
|
|
$$->Attributes = FARequired;
|
|
}
|
|
| T_NAVIGATION '(' identifier ')'
|
|
{
|
|
$$ = new Field();
|
|
$$->Attributes = FANavigation;
|
|
$$->NavigationName = $3;
|
|
std::free($3);
|
|
}
|
|
| T_NAVIGATION
|
|
{
|
|
$$ = new Field();
|
|
$$->Attributes = FANavigation;
|
|
}
|
|
;
|
|
|
|
field_attributes: /* empty */
|
|
{
|
|
$$ = new Field();
|
|
}
|
|
| field_attributes ',' field_attribute
|
|
{
|
|
$$ = $1;
|
|
$$->Attributes |= $3->Attributes;
|
|
if (!$3->NavigationName.empty())
|
|
$$->NavigationName = $3->NavigationName;
|
|
delete $3;
|
|
}
|
|
| field_attribute
|
|
{
|
|
$$ = $1;
|
|
}
|
|
;
|
|
|
|
field_accessor_list: /* empty */
|
|
{
|
|
$$ = new std::vector<FieldAccessor>();
|
|
}
|
|
| '{' field_accessors '}'
|
|
{
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
field_accessors: /* empty */
|
|
{
|
|
$$ = new std::vector<FieldAccessor>();
|
|
}
|
|
| field_accessors field_accessor
|
|
{
|
|
$$ = $1;
|
|
$$->push_back(*$2);
|
|
delete $2;
|
|
}
|
|
;
|
|
|
|
field_accessor: T_FIELD_ACCESSOR_TYPE T_STRING
|
|
{
|
|
$$ = new FieldAccessor(static_cast<FieldAccessorType>($1), $2, false);
|
|
std::free($2);
|
|
}
|
|
| T_FIELD_ACCESSOR_TYPE ';'
|
|
{
|
|
$$ = new FieldAccessor(static_cast<FieldAccessorType>($1), "", true);
|
|
}
|
|
;
|
|
|
|
validator_rules: /* empty */
|
|
{
|
|
$$ = new std::vector<Rule>();
|
|
}
|
|
| validator_rules validator_rule
|
|
{
|
|
$$->push_back(*$2);
|
|
delete $2;
|
|
}
|
|
;
|
|
|
|
validator_rule: T_NAME '(' T_IDENTIFIER ')' identifier ';'
|
|
{
|
|
$$ = new Rule();
|
|
$$->Attributes = 0;
|
|
$$->IsName = true;
|
|
$$->Type = $3;
|
|
std::free($3);
|
|
$$->Pattern = $5;
|
|
std::free($5);
|
|
}
|
|
| T_IDENTIFIER identifier ';'
|
|
{
|
|
$$ = new Rule();
|
|
$$->Attributes = 0;
|
|
$$->IsName = false;
|
|
$$->Type = $1;
|
|
std::free($1);
|
|
$$->Pattern = $2;
|
|
std::free($2);
|
|
}
|
|
| T_NAME '(' T_IDENTIFIER ')' identifier '{' validator_rules '}' ';'
|
|
{
|
|
$$ = new Rule();
|
|
$$->Attributes = 0;
|
|
$$->IsName = true;
|
|
$$->Type = $3;
|
|
std::free($3);
|
|
$$->Pattern = $5;
|
|
std::free($5);
|
|
$$->Rules = *$7;
|
|
delete $7;
|
|
}
|
|
| T_IDENTIFIER identifier '{' validator_rules '}' ';'
|
|
{
|
|
$$ = new Rule();
|
|
$$->Attributes = 0;
|
|
$$->IsName = false;
|
|
$$->Type = $1;
|
|
std::free($1);
|
|
$$->Pattern = $2;
|
|
std::free($2);
|
|
$$->Rules = *$4;
|
|
delete $4;
|
|
}
|
|
| T_REQUIRED identifier ';'
|
|
{
|
|
$$ = new Rule();
|
|
$$->Attributes = RARequired;
|
|
$$->IsName = false;
|
|
$$->Type = "";
|
|
$$->Pattern = $2;
|
|
std::free($2);
|
|
}
|
|
;
|
|
|
|
validator: T_VALIDATOR T_IDENTIFIER '{' validator_rules '}' ';'
|
|
{
|
|
$$ = new Validator();
|
|
|
|
$$->Name = $2;
|
|
std::free($2);
|
|
|
|
$$->Rules = *$4;
|
|
delete $4;
|
|
}
|
|
;
|
|
|
|
identifier: T_IDENTIFIER
|
|
| T_STRING
|
|
{
|
|
$$ = $1;
|
|
}
|
|
;
|