Skip to content
Snippets Groups Projects
Commit c7e28528 authored by Virgile Robles's avatar Virgile Robles
Browse files

Add conditional invariant context

parent b970ffdf
No related branches found
No related tags found
No related merge requests found
...@@ -76,6 +76,12 @@ function defined in `main.c`. ...@@ -76,6 +76,12 @@ function defined in `main.c`.
The context is the situation in a target function in which the property must The context is the situation in a target function in which the property must
hold. It can be one of the following : hold. It can be one of the following :
- `\conditional_invariant` : the property `P` is guarantted to be verified at
the end of the target function *if* it is verified at the begining.
- `\precond` : the property `P` is a precondition of the target function and is
must hold at each of its callsites.
- `\postcond` : the property `P` hold at the end of the target function for
every initial state satisfying the preconditions of the function.
- `\weak_invariant` : the property `P` is guaranteed to be verified at the - `\weak_invariant` : the property `P` is guaranteed to be verified at the
beginning and end of the target function. beginning and end of the target function.
- `\strong_invariant` : the property `P` must hold at every point of the target - `\strong_invariant` : the property `P` must hold at every point of the target
......
...@@ -186,22 +186,34 @@ class weak_inv_visitor flags full_table table (pre, post) = object (self) ...@@ -186,22 +186,34 @@ class weak_inv_visitor flags full_table table (pre, post) = object (self)
let pred_name = (Emitter.get_name ump.ump_emitter) :: pred'.pred_name in let pred_name = (Emitter.get_name ump.ump_emitter) :: pred'.pred_name in
let pred'' = {pred' with pred_name} in let pred'' = {pred' with pred_name} in
let only_check = not ump.ump_assert in let only_check = not ump.ump_assert in
let ip_requires = Logic_const.new_predicate ~only_check pred'' in
let ip_ensures = Logic_const.refresh_predicate ip_requires in
let deps = ref [] in let deps = ref [] in
if pre then ( if pre then (
let ip_requires = Logic_const.new_predicate ~only_check pred'' in
Annotations.add_requires ump.ump_emitter kf [ip_requires]; Annotations.add_requires ump.ump_emitter kf [ip_requires];
let defb = Extlib.the (Cil.find_default_behavior (Annotations.funspec kf)) in let defb = Extlib.the (Cil.find_default_behavior (Annotations.funspec kf)) in
let rips = Property.ip_of_requires kf Kglobal defb ip_requires :: !deps in let rips = Property.ip_of_requires kf Kglobal defb ip_requires :: !deps in
let callsites_ip = get_callsites_ips kf rips in let callsites_ip = get_callsites_ips kf rips in
deps := rips @ callsites_ip @ !deps deps := rips @ callsites_ip @ !deps
); );
if post then ( let add_post ip =
Annotations.add_ensures ump.ump_emitter kf [(Normal, ip_ensures)]; Annotations.add_ensures ump.ump_emitter kf [(Normal, ip)];
let defb = Extlib.the (Cil.find_default_behavior (Annotations.funspec kf)) in let defb = Extlib.the (Cil.find_default_behavior (Annotations.funspec kf)) in
let eip = Property.ip_of_ensures kf Kglobal defb (Normal, ip_ensures) in let eip = Property.ip_of_ensures kf Kglobal defb (Normal, ip) in
deps := eip :: !deps deps := eip :: !deps
); in
begin match post with
| `None -> ()
| `Strict ->
let ip_ensures = Logic_const.new_predicate ~only_check pred'' in
add_post ip_ensures
| `Conditional ->
let pred' = Logic_const.pimplies (pred, pred) in
let pred'' = Meta_dispatch.name_mp_pred flags pred' ump.ump_counter in
let pred_name = (Emitter.get_name ump.ump_emitter) :: pred''.pred_name in
let pred''' = {pred'' with pred_name} in
let ip_ensures = Logic_const.new_predicate ~only_check pred''' in
add_post ip_ensures
end;
Meta_dispatch.add_dependency ump !deps Meta_dispatch.add_dependency ump !deps
in in
(* Delay actual instanciation for correct numbering *) (* Delay actual instanciation for correct numbering *)
...@@ -506,7 +518,7 @@ let lvals_of_predicate pred = ...@@ -506,7 +518,7 @@ let lvals_of_predicate pred =
* maintain the invariant at the end * maintain the invariant at the end
*) *)
class strong_inv_visitor flags all_mp table = object (self) class strong_inv_visitor flags all_mp table = object (self)
inherit weak_inv_visitor flags all_mp table (true, true) inherit weak_inv_visitor flags all_mp table (true, `Strict)
val mutable lenient = false val mutable lenient = false
(* Given a lval which is to be modified, check if its modification (* Given a lval which is to be modified, check if its modification
...@@ -598,7 +610,7 @@ let report all_mp tables = ...@@ -598,7 +610,7 @@ let report all_mp tables =
let annotate flags all_mp by_context = let annotate flags all_mp by_context =
let get_vis table = function let get_vis table = function
| Weak_invariant -> | Weak_invariant ->
(new weak_inv_visitor flags all_mp table (true, true) :> Visitor.frama_c_visitor) (new weak_inv_visitor flags all_mp table (true, `Strict) :> Visitor.frama_c_visitor)
| Calling -> | Calling ->
(new calling_visitor flags all_mp table :> Visitor.frama_c_visitor) (new calling_visitor flags all_mp table :> Visitor.frama_c_visitor)
| Writing -> | Writing ->
...@@ -608,9 +620,11 @@ let annotate flags all_mp by_context = ...@@ -608,9 +620,11 @@ let annotate flags all_mp by_context =
| Strong_invariant -> | Strong_invariant ->
(new strong_inv_visitor flags all_mp table :> Visitor.frama_c_visitor) (new strong_inv_visitor flags all_mp table :> Visitor.frama_c_visitor)
| Precond -> | Precond ->
(new weak_inv_visitor flags all_mp table (true, false) :> Visitor.frama_c_visitor) (new weak_inv_visitor flags all_mp table (true, `None) :> Visitor.frama_c_visitor)
| Postcond -> | Postcond ->
(new weak_inv_visitor flags all_mp table (false, true) :> Visitor.frama_c_visitor) (new weak_inv_visitor flags all_mp table (false, `Strict) :> Visitor.frama_c_visitor)
| Conditional_invariant ->
(new weak_inv_visitor flags all_mp table (false, `Conditional) :> Visitor.frama_c_visitor)
in in
List.iter (fun (ctx, table) -> List.iter (fun (ctx, table) ->
let vis = get_vis table ctx in let vis = get_vis table ctx in
......
...@@ -217,6 +217,7 @@ let generate_mp prefix preds globals fmt (mp, tset) = ...@@ -217,6 +217,7 @@ let generate_mp prefix preds globals fmt (mp, tset) =
(match mp.mp_context with (match mp.mp_context with
| Weak_invariant -> "Weak invariant" | Weak_invariant -> "Weak invariant"
| Strong_invariant -> "Strong invariant" | Strong_invariant -> "Strong invariant"
| Conditional_invariant -> "Conditional invariant"
| Postcond -> "Postcond" | Postcond -> "Postcond"
| Precond -> "Precond" | Precond -> "Precond"
| Writing -> "Writing" | Writing -> "Writing"
......
...@@ -102,7 +102,7 @@ let unpack_mp flags mp admit = ...@@ -102,7 +102,7 @@ let unpack_mp flags mp admit =
let dispatch flags mps = let dispatch flags mps =
let all_mp = Hashtbl.create 10 in let all_mp = Hashtbl.create 10 in
let ctxts = [Weak_invariant; Strong_invariant; Writing; Reading; Calling; let ctxts = [Weak_invariant; Strong_invariant; Writing; Reading; Calling;
Precond; Postcond] in Precond; Postcond; Conditional_invariant] in
let tables = List.map (fun ctx -> ctx, Str_Hashtbl.create 5) ctxts in let tables = List.map (fun ctx -> ctx, Str_Hashtbl.create 5) ctxts in
List.iter (fun mp -> List.iter (fun mp ->
let table = List.assoc mp.mp_context tables in let table = List.assoc mp.mp_context tables in
......
...@@ -34,6 +34,7 @@ type context = ...@@ -34,6 +34,7 @@ type context =
| Reading | Reading
| Postcond | Postcond
| Precond | Precond
| Conditional_invariant
type target = type target =
| TgAll | TgAll
...@@ -76,7 +77,8 @@ let mp_contexts = [ ...@@ -76,7 +77,8 @@ let mp_contexts = [
"\\calling"; "\\calling";
"\\reading"; "\\reading";
"\\postcond"; "\\postcond";
"\\precond" "\\precond";
"\\conditional_invariant"
] ]
let mp_metavariables = [ let mp_metavariables = [
"\\written"; "\\written";
...@@ -387,6 +389,7 @@ let process_property tc loc = function ...@@ -387,6 +389,7 @@ let process_property tc loc = function
let mp_context = match econtext.lexpr_node with let mp_context = match econtext.lexpr_node with
| PLvar "\\weak_invariant" -> Weak_invariant | PLvar "\\weak_invariant" -> Weak_invariant
| PLvar "\\strong_invariant" -> Strong_invariant | PLvar "\\strong_invariant" -> Strong_invariant
| PLvar "\\conditional_invariant" -> Conditional_invariant
| PLvar "\\writing" -> Writing | PLvar "\\writing" -> Writing
| PLvar "\\reading" -> Reading | PLvar "\\reading" -> Reading
| PLvar "\\calling" -> Calling | PLvar "\\calling" -> Calling
......
...@@ -30,6 +30,7 @@ type context = ...@@ -30,6 +30,7 @@ type context =
| Reading | Reading
| Postcond | Postcond
| Precond | Precond
| Conditional_invariant
type target = type target =
| TgAll | TgAll
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
#include <assert.h> #include <assert.h>
unsigned A = 2; unsigned A = 2, B;
//@ assigns \nothing; //@ assigns \nothing;
void requiresInv() { void requiresInv() {
...@@ -14,9 +14,11 @@ void requiresInv() { ...@@ -14,9 +14,11 @@ void requiresInv() {
void breaksInv() { void breaksInv() {
A += 1; A += 1;
} }
//@ assigns \nothing; //@ assigns \nothing;
void useless() { void useless() {
} }
//@ assigns A; //@ assigns A;
void maintainsInv() { void maintainsInv() {
if(A < 42) if(A < 42)
...@@ -24,8 +26,17 @@ void maintainsInv() { ...@@ -24,8 +26,17 @@ void maintainsInv() {
else A -= 4; else A -= 4;
useless(); useless();
} }
//@ assigns A;
//@ assigns B;
void cond() {
if(B != 42){
B *= 2;
}
}
//@ assigns A, B;
void main() { void main() {
cond();
maintainsInv(); maintainsInv();
requiresInv(); requiresInv();
...@@ -50,7 +61,13 @@ void main() { ...@@ -50,7 +61,13 @@ void main() {
meta \prop, meta \prop,
\name("invariant_2"), \name("invariant_2"),
\targets(\diff(\ALL, {main, useless})), \targets(\diff(\ALL, {main, useless, cond})),
\context(\weak_invariant), \context(\weak_invariant),
A % 2 == 0; A % 2 == 0;
meta \prop,
\name(conditional),
\targets(cond),
\context(\conditional_invariant),
A == 42;
*/ */
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment