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:
parent
7b580e006a
commit
eca96cdf80
3 changed files with 135 additions and 7 deletions
3
NEWS
3
NEWS
|
|
@ -96,6 +96,9 @@ New in spot 2.5.3.dev (not yet released)
|
|||
- print_dot() will correctly escape strings containing \n in HTML
|
||||
mode.
|
||||
|
||||
- The HOA parser will now accept Alias: declarations that occur
|
||||
before AP:.
|
||||
|
||||
New in spot 2.5.3 (2018-04-20)
|
||||
|
||||
Bugs fixed:
|
||||
|
|
|
|||
|
|
@ -95,6 +95,18 @@ extern "C" int strverscmp(const char *s1, const char *s2);
|
|||
std::vector<int> ap;
|
||||
std::vector<bdd> guards;
|
||||
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;
|
||||
std::vector<state_info> info_states; // States declared and used.
|
||||
std::vector<std::pair<spot::location,
|
||||
|
|
@ -372,6 +384,13 @@ header: format-version header-items
|
|||
error(@$, "missing 'Acceptance:' header");
|
||||
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.
|
||||
{
|
||||
auto explicit_labels = res.prop_is_true("explicit-labels");
|
||||
|
|
@ -678,6 +697,29 @@ aps: "AP:" INT
|
|||
error(@$, out.str());
|
||||
}
|
||||
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});
|
||||
}
|
||||
| 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;
|
||||
o << "ignoring redefinition of alias @" << *$2;
|
||||
error(@$, o.str());
|
||||
}
|
||||
delete $2;
|
||||
bdd_delref($3);
|
||||
bdd_delref($4);
|
||||
}
|
||||
| "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
|
||||
// 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
|
||||
{
|
||||
$$ = new std::vector<unsigned>{$1, $3};
|
||||
|
|
@ -912,7 +955,22 @@ label-expr: 't'
|
|||
}
|
||||
| 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(res.ap_loc, "... declared here");
|
||||
|
|
|
|||
|
|
@ -1,6 +1,6 @@
|
|||
#!/bin/sh
|
||||
# -*- 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).
|
||||
#
|
||||
# This file is part of Spot, a model checking library.
|
||||
|
|
@ -504,6 +504,37 @@ cat >input <<EOF
|
|||
[!@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: @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
|
||||
|
||||
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:16.12-14: 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
|
||||
|
||||
cat >input <<EOF
|
||||
|
|
@ -535,6 +569,22 @@ cat >input <<EOF
|
|||
[!@a & @b.c] 0 {1}
|
||||
[@a & @b.c] 0 {0 1}
|
||||
--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 */
|
||||
HOA: v1
|
||||
States: 2
|
||||
|
|
@ -565,6 +615,22 @@ State: 0
|
|||
[0&1&2] 0 {0 1}
|
||||
--END--
|
||||
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
|
||||
Start: 0
|
||||
AP: 1 "a"
|
||||
|
|
@ -581,7 +647,8 @@ EOF
|
|||
|
||||
expectok input --stats='%F:%L' <<EOF
|
||||
input:1.5-16.11
|
||||
input:18.1-26.7
|
||||
input:17.3-32.9
|
||||
input:34.1-42.7
|
||||
EOF
|
||||
|
||||
cat >input <<EOF
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue