parseaut: accept Alias: before AP:

Fixes #345.

* spot/parseaut/parseaut.yy: Deal with this inconvenient order.
* tests/core/parseaut.test: Test it.
* NEWS: Mention the bug fix.
This commit is contained in:
Alexandre Duret-Lutz 2018-05-21 10:24:25 +02:00
parent 7b580e006a
commit eca96cdf80
3 changed files with 135 additions and 7 deletions

3
NEWS
View file

@ -96,6 +96,9 @@ New in spot 2.5.3.dev (not yet released)
- print_dot() will correctly escape strings containing \n in HTML - print_dot() will correctly escape strings containing \n in HTML
mode. mode.
- The HOA parser will now accept Alias: declarations that occur
before AP:.
New in spot 2.5.3 (2018-04-20) New in spot 2.5.3 (2018-04-20)
Bugs fixed: Bugs fixed:

View file

@ -95,6 +95,18 @@ extern "C" int strverscmp(const char *s1, const char *s2);
std::vector<int> ap; std::vector<int> ap;
std::vector<bdd> guards; std::vector<bdd> guards;
std::vector<bdd>::const_iterator cur_guard; std::vector<bdd>::const_iterator cur_guard;
// If "Alias: ..." occurs before "AP: ..." in the HOA format we
// are in trouble because the parser would like to turn each
// alias into a BDD, yet the atomic proposition have not been
// declared yet. We solve that by using arbitrary BDD variables
// numbers (in fact we use the same number given in the Alias:
// definition) and keeping track of the highest variable number
// we have seen (unknown_ap_max). Once AP: is encountered,
// we can remap everything. If AP: is never encountered an
// unknown_ap_max is non-negative, then we can signal an error.
int unknown_ap_max = -1;
spot::location unknown_ap_max_location;
bool in_alias = false;
map_t dest_map; map_t dest_map;
std::vector<state_info> info_states; // States declared and used. std::vector<state_info> info_states; // States declared and used.
std::vector<std::pair<spot::location, std::vector<std::pair<spot::location,
@ -372,6 +384,13 @@ header: format-version header-items
error(@$, "missing 'Acceptance:' header"); error(@$, "missing 'Acceptance:' header");
res.ignore_acc = true; res.ignore_acc = true;
} }
if (res.unknown_ap_max >= 0 && !res.ignore_more_ap)
{
error(res.unknown_ap_max_location,
"atomic proposition used in Alias without AP declaration");
for (auto& p: res.alias)
p.second = bddtrue;
}
// Process properties. // Process properties.
{ {
auto explicit_labels = res.prop_is_true("explicit-labels"); auto explicit_labels = res.prop_is_true("explicit-labels");
@ -678,6 +697,29 @@ aps: "AP:" INT
error(@$, out.str()); error(@$, out.str());
} }
res.ignore_more_ap = true; res.ignore_more_ap = true;
// If we have seen Alias: before AP: we have some variable
// renaming to perform.
if (res.unknown_ap_max >= 0)
{
int apsize = res.ap.size();
if (apsize <= res.unknown_ap_max)
{
error(res.unknown_ap_max_location,
"AP number is larger than the number of APs...");
error(@1, "... declared here");
}
bddPair* pair = bdd_newpair();
int max = std::min(res.unknown_ap_max, apsize - 1);
for (int i = 0; i <= max; ++i)
if (i != res.ap[i])
bdd_setbddpair(pair, i, res.ap[i]);
bdd extra = bddtrue;
for (unsigned i = apsize; i <= res.unknown_ap_max; ++i)
extra &= bdd_ithvar(i);
for (auto& p: res.alias)
p.second = bdd_restrict(bdd_replace(p.second, pair), extra);
bdd_freepair(pair);
}
} }
} }
@ -710,16 +752,17 @@ header-item: "States:" INT
res.start.emplace_back(@$, std::vector<unsigned>{$2}); res.start.emplace_back(@$, std::vector<unsigned>{$2});
} }
| aps | aps
| "Alias:" ANAME label-expr | "Alias:" ANAME { res.in_alias=true; } label-expr
{ {
if (!res.alias.emplace(*$2, bdd_from_int($3)).second) res.in_alias = false;
if (!res.alias.emplace(*$2, bdd_from_int($4)).second)
{ {
std::ostringstream o; std::ostringstream o;
o << "ignoring redefinition of alias @" << *$2; o << "ignoring redefinition of alias @" << *$2;
error(@$, o.str()); error(@$, o.str());
} }
delete $2; delete $2;
bdd_delref($3); bdd_delref($4);
} }
| "Acceptance:" INT | "Acceptance:" INT
{ {
@ -891,7 +934,7 @@ state-conj-2: checked-state-num '&' checked-state-num
} }
// Same as state-conj-2 except we cannot check the state numbers // Same as state-conj-2 except we cannot check the state numbers
// against a number of state that may not have been declared yet. // against a number of states that may not have been declared yet.
init-state-conj-2: state-num '&' state-num init-state-conj-2: state-num '&' state-num
{ {
$$ = new std::vector<unsigned>{$1, $3}; $$ = new std::vector<unsigned>{$1, $3};
@ -912,7 +955,22 @@ label-expr: 't'
} }
| INT | INT
{ {
if ($1 >= res.ap.size()) if (res.in_alias && !res.ignore_more_ap)
{
// We are reading Alias: before AP: has been given.
// Use $1 as temporary variable number. We will relabel
// everything once AP: is known.
if (res.unknown_ap_max < (int)$1)
{
res.unknown_ap_max = $1;
res.unknown_ap_max_location = @1;
int missing_vars = 1 + bdd_varnum() - $1;
if (missing_vars > 0)
bdd_extvarnum(missing_vars);
}
$$ = bdd_ithvar($1).id();
}
else if ($1 >= res.ap.size())
{ {
error(@1, "AP number is larger than the number of APs..."); error(@1, "AP number is larger than the number of APs...");
error(res.ap_loc, "... declared here"); error(res.ap_loc, "... declared here");

View file

@ -1,6 +1,6 @@
#!/bin/sh #!/bin/sh
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
# Copyright (C) 2014-2017 Laboratoire de Recherche et Développement de # Copyright (C) 2014-2018 Laboratoire de Recherche et Développement de
# l'Epita (LRDE). # l'Epita (LRDE).
# #
# This file is part of Spot, a model checking library. # This file is part of Spot, a model checking library.
@ -504,6 +504,37 @@ cat >input <<EOF
[!@a & @bc] 0 {1} [!@a & @bc] 0 {1}
[@a & @bc] 0 {0 1} [@a & @bc] 0 {0 1}
--END-- --END--
HOA: v1
name: "GFa & GF(b & c)"
States: 1
Start: 0
acc-name: generalized-Buchi 2
Acceptance: 2 (Inf(0) & Inf(1))
Alias: @a 0
Alias: @bc 1 & 2
--BODY--
State: 0
[!@a & !@bc] 0
[@a & !@bc] 0 {0}
[!@a & @bc] 0 {1}
[@a & @bc] 0 {0 1}
--END--
HOA: v1
name: "GFa & GF(b & c)"
States: 1
Start: 0
acc-name: generalized-Buchi 2
Acceptance: 2 (Inf(0) & Inf(1))
Alias: @bc 1 & 2
AP: 1 "a"
Alias: @a 0
--BODY--
State: 0
[!@a & !@bc] 0
[@a & !@bc] 0 {0}
[!@a & @bc] 0 {1}
[@a & @bc] 0 {0 1}
--END--
EOF EOF
expecterr input <<EOF expecterr input <<EOF
@ -516,6 +547,9 @@ input:15.20-22: trans-based acceptance used despite...
input:11.17-25: ... declaration of state-based acceptance. input:11.17-25: ... declaration of state-based acceptance.
input:16.12-14: unknown alias @bc input:16.12-14: unknown alias @bc
input:17.11-13: unknown alias @bc input:17.11-13: unknown alias @bc
input:26.18: atomic proposition used in Alias without AP declaration
input:40.16: AP number is larger than the number of APs...
input:41.1-3: ... declared here
EOF EOF
cat >input <<EOF cat >input <<EOF
@ -535,6 +569,22 @@ cat >input <<EOF
[!@a & @b.c] 0 {1} [!@a & @b.c] 0 {1}
[@a & @b.c] 0 {0 1} [@a & @b.c] 0 {0 1}
--END-- --END--
HOA: v1.1
Alias: @b.c 1 & 2
name: "GFa & GF(b & c)"
States: 1
Start: 0
acc-name: who cares
Acceptance: 2 (Inf(0) & Inf(1))
AP: 3 "a" "b" "c"
Alias: @a 0
--BODY--
State: 0
[!@a & !@b.c] 0
[@a & !@b.c] 0 {0}
[!@a & @b.c] 0 {1}
[@a & @b.c] 0 {0 1}
--END--
/* Some comment */ /* Some comment */
HOA: v1 HOA: v1
States: 2 States: 2
@ -565,6 +615,22 @@ State: 0
[0&1&2] 0 {0 1} [0&1&2] 0 {0 1}
--END-- --END--
HOA: v1 HOA: v1
name: "GFa & GF(b & c)"
States: 1
Start: 0
AP: 3 "a" "b" "c"
acc-name: generalized-Buchi 2
Acceptance: 2 Inf(0)&Inf(1)
properties: trans-labels explicit-labels trans-acc complete
properties: deterministic
--BODY--
State: 0
[!0&!1 | !0&!2] 0
[0&!1 | 0&!2] 0 {0}
[!0&1&2] 0 {1}
[0&1&2] 0 {0 1}
--END--
HOA: v1
States: 2 States: 2
Start: 0 Start: 0
AP: 1 "a" AP: 1 "a"
@ -581,7 +647,8 @@ EOF
expectok input --stats='%F:%L' <<EOF expectok input --stats='%F:%L' <<EOF
input:1.5-16.11 input:1.5-16.11
input:18.1-26.7 input:17.3-32.9
input:34.1-42.7
EOF EOF
cat >input <<EOF cat >input <<EOF