formula opt_dest
%type transition
%type transitions
%type ident_list
%destructor { delete $$; }
%destructor { $$->first->destroy(); delete $$->second; delete $$; }
%destructor {
for (std::list::iterator i = $$->begin();
i != $$->end(); ++i)
{
i->first->destroy();
delete i->second;
}
delete $$;
}
%printer {
if ($$)
debug_stream() << *$$;
else
debug_stream() << "\"\""; }
%%
neverclaim:
"never" '{' states '}'
states:
/* empty */
| state
| states ';' state
| states ';'
ident_list:
IDENT ':'
{
$$ = $1;
}
| ident_list IDENT ':'
{
result->add_state_alias(*$2, *$1);
// Keep any identifier that start with accept.
if (strncmp("accept", $1->c_str(), 6))
{
delete $1;
$$ = $2;
}
else
{
delete $2;
$$ = $1;
}
}
state:
ident_list "skip"
{
spot::state_explicit_string::transition* t = result->create_transition(*$1, *$1);
bool acc = !strncmp("accept", $1->c_str(), 6);
if (acc)
result->add_acceptance_condition(t,
spot::ltl::constant::true_instance());
delete $1;
}
| ident_list { delete $1; }
| ident_list "false" { delete $1; }
| ident_list "if" transitions "fi"
{
std::list::iterator it;
bool acc = !strncmp("accept", $1->c_str(), 6);
for (it = $3->begin(); it != $3->end(); ++it)
{
spot::state_explicit_string::transition* t =
result->create_transition(*$1, *it->second);
result->add_condition(t, it->first);
if (acc)
result
->add_acceptance_condition(t, spot::ltl::constant::true_instance());
}
// Free the list
delete $1;
for (std::list::iterator it = $3->begin();
it != $3->end(); ++it)
delete it->second;
delete $3;
}
transitions:
/* empty */ { $$ = new std::list; }
| transitions transition
{
if ($2)
{
$1->push_back(*$2);
delete $2;
}
$$ = $1;
}
formula: FORMULA | "false" { $$ = new std::string("0"); }
opt_dest:
/* empty */
{
$$ = 0;
}
| "->" "goto" IDENT
{
$$ = $3;
}
transition:
':' ':' formula opt_dest
{
// If there is no destination, do ignore the transition.
// This happens for instance with
// if
// :: false
// fi
if (!$4)
{
delete $3;
$$ = 0;
}
else
{
spot::ltl::parse_error_list pel;
const spot::ltl::formula* f = spot::ltl::parse(*$3, pel,
parse_environment,
false, true);
delete $3;
for(spot::ltl::parse_error_list::const_iterator i = pel.begin();
i != pel.end(); ++i)
{
// Adjust the diagnostic to the current position.
location here = @3;
here.end.line = here.begin.line + i->first.end.line - 1;
here.end.column = here.begin.column + i->first.end.column -1;
here.begin.line += i->first.begin.line - 1;
here.begin.column += i->first.begin.column - 1;
error(here, i->second);
}
$$ = new pair(f, $4);
}
}
%%
void
neverclaimyy::parser::error(const location_type& location,
const std::string& message)
{
error_list.push_back(spot::neverclaim_parse_error(location, message));
}
namespace spot
{
tgba_explicit_string*
neverclaim_parse(const std::string& name,
neverclaim_parse_error_list& error_list,
bdd_dict* dict,
environment& env,
bool debug)
{
if (neverclaimyyopen(name))
{
error_list.push_back
(neverclaim_parse_error(neverclaimyy::location(),
std::string("Cannot open file ") + name));
return 0;
}
tgba_explicit_string* result = new tgba_explicit_string(dict);
result->declare_acceptance_condition(spot::ltl::constant::true_instance());
neverclaimyy::parser parser(error_list, env, result);
parser.set_debug_level(debug);
parser.parse();
neverclaimyyclose();
return result;
}
}
// Local Variables:
// mode: c++
// End: