Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
F
frama-c
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Deploy
Releases
Package Registry
Container Registry
Model registry
Operate
Terraform modules
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
Charles Southerland
frama-c
Commits
f5b131fe
Commit
f5b131fe
authored
4 years ago
by
Julien Signoles
Browse files
Options
Downloads
Patches
Plain Diff
[e-acsl] remove the main visitor in Prepare_ast in order to propagate property statuses soundly
[e-acsl] simplify and document Prepare_ast a bit more
parent
9acdf608
No related branches found
Branches containing commit
No related tags found
Tags containing commit
No related merge requests found
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
src/plugins/e-acsl/src/project_initializer/prepare_ast.ml
+274
-321
274 additions, 321 deletions
src/plugins/e-acsl/src/project_initializer/prepare_ast.ml
with
274 additions
and
321 deletions
src/plugins/e-acsl/src/project_initializer/prepare_ast.ml
+
274
−
321
View file @
f5b131fe
...
@@ -21,6 +21,7 @@
...
@@ -21,6 +21,7 @@
(**************************************************************************)
(**************************************************************************)
open
Cil_types
open
Cil_types
open
Cil_datatype
let
dkey
=
Options
.
dkey_prepare
let
dkey
=
Options
.
dkey_prepare
...
@@ -28,81 +29,69 @@ let dkey = Options.dkey_prepare
...
@@ -28,81 +29,69 @@ let dkey = Options.dkey_prepare
(* Environment *)
(* Environment *)
(* ********************************************************************** *)
(* ********************************************************************** *)
let
fct_tbl
:
unit
Kernel_function
.
Hashtbl
.
t
=
Kernel_function
.
Hashtbl
.
create
7
module
Dup_functions
:
sig
val
generate_vi
:
varinfo
->
varinfo
(* return the new varinfo *)
(* The purpose of [actions] is similar to the Frama-C visitor's
val
mem
:
varinfo
->
bool
[get_filling_actions] but we need to fill it outside the visitor. So it is
val
find
:
varinfo
->
varinfo
our own version. *)
let
actions
=
Queue
.
create
()
(* global table for ensuring that logic info are not shared between a function
definition and its duplicate *)
module
Global
:
sig
val
add_logic_info
:
logic_info
->
unit
val
mem_logic_info
:
logic_info
->
bool
val
reset
:
unit
->
unit
val
reset
:
unit
->
unit
end
=
struct
end
=
struct
let
tbl
=
Cil_datatype
.
Logic_info
.
Hashtbl
.
create
7
let
tbl
:
varinfo
Varinfo
.
Hashtbl
.
t
=
Varinfo
.
Hashtbl
.
create
7
let
add_logic_info
x
=
Cil_datatype
.
Logic_info
.
Hashtbl
.
add
tbl
x
()
let
mem_logic_info
x
=
Cil_datatype
.
Logic_info
.
Hashtbl
.
mem
tbl
x
(* assume [vi] is not already in [tbl] *)
let
reset
()
=
Cil_datatype
.
Logic_info
.
Hashtbl
.
clear
tbl
let
generate_vi
vi
=
let
new_name
=
Functions
.
RTL
.
mk_gen_name
vi
.
vname
in
let
new_vi
=
Cil
.
makeGlobalVar
~
referenced
:
vi
.
vreferenced
~
loc
:
vi
.
vdecl
new_name
vi
.
vtype
in
Varinfo
.
Hashtbl
.
add
tbl
vi
new_vi
;
new_vi
end
let
mem
=
Varinfo
.
Hashtbl
.
mem
tbl
let
find
=
Varinfo
.
Hashtbl
.
find
tbl
let
reset
()
=
Varinfo
.
Hashtbl
.
clear
tbl
let
reset
()
=
end
Kernel_function
.
Hashtbl
.
clear
fct_tbl
;
Global
.
reset
()
;
Queue
.
clear
actions
(* ********************************************************************** *)
(* ********************************************************************** *)
(* Duplicating a function *)
(* Duplicating a function *)
(* ********************************************************************** *)
(* ********************************************************************** *)
(* [tbl] associates the old formals to the new ones *)
(* [formals] associates the old formals to the new ones in order to
let
dup_funspec
tbl
bhv
spec
=
substitute them in [spec]. *)
let
dup_funspec
formals
spec
=
(* Options.feedback "DUP SPEC %a" Cil.d_funspec spec;*)
(* Options.feedback "DUP SPEC %a" Cil.d_funspec spec;*)
let
o
=
object
let
o
=
object
inherit
Cil
.
genericCilVisitor
bhv
inherit
Visitor
.
frama_c_refresh
(
Project
.
current
()
)
val
already_visited
=
Cil_datatype
.
Logic_var
.
Hashtbl
.
create
7
method
!
vlogic_info_use
li
=
if
Global
.
mem_logic_info
li
then
Cil
.
ChangeDoChildrenPost
({
li
with
l_var_info
=
li
.
l_var_info
}
(* force a copy *)
,
Visitor_behavior
.
Get
.
logic_info
bhv
)
else
Cil
.
JustCopy
method
!
vterm_offset
_
=
val
already_visited
=
Logic_var
.
Hashtbl
.
create
7
Cil
.
DoChildrenPost
(
function
(* no way to directly visit fieldinfo and model_info uses *)
|
TField
(
fi
,
off
)
->
TField
(
Visitor_behavior
.
Get
.
fieldinfo
bhv
fi
,
off
)
|
TModel
(
mi
,
off
)
->
TModel
(
Visitor_behavior
.
Get
.
model_info
bhv
mi
,
off
)
|
off
->
off
)
(* just substituting in [!vvrbl] (when using a varinfo) does not work
because varinfo's occurrences are shared in logic_vars, so modifying the
varinfo would modify any logic_var based on it, even if it is not part of
this [spec] (e.g., if it is in another annotation of the same
function) *)
method
!
vlogic_var_use
orig_lvi
=
method
!
vlogic_var_use
orig_lvi
=
match
orig_lvi
.
lv_origin
with
match
orig_lvi
.
lv_origin
with
|
None
->
|
None
->
Cil
.
JustCopy
Cil
.
JustCopy
|
Some
vi
->
|
Some
vi
->
try
try
let
new_lvi
=
let
new_lvi
=
Logic_var
.
Hashtbl
.
find
already_visited
orig_lvi
in
Cil_datatype
.
Logic_var
.
Hashtbl
.
find
already_visited
orig_lvi
(* recreate sharing of the logic_var in this [spec] *)
in
Cil
.
ChangeTo
new_lvi
Cil
.
ChangeTo
new_lvi
with
Not_found
->
with
Not_found
->
(* first time visiting this logic var *)
Cil
.
ChangeDoChildrenPost
Cil
.
ChangeDoChildrenPost
({
orig_lvi
with
lv_id
=
orig_lvi
.
lv_id
}
(* force a copy *)
,
({
orig_lvi
with
lv_id
=
orig_lvi
.
lv_id
}
(* force a copy *)
,
fun
lvi
->
fun
lvi
->
try
try
let
new_vi
=
Cil_datatype
.
Varinfo
.
Hashtbl
.
find
tbl
vi
in
let
new_vi
=
Varinfo
.
Hashtbl
.
find
formals
vi
in
Cil_datatype
.
Logic_var
.
Hashtbl
.
add
Logic_var
.
Hashtbl
.
add
already_visited
orig_lvi
lvi
;
already_visited
orig_lvi
lvi
;
(* [lvi] is the logic counterpart of a formal varinfo that has
(* [lvi] is the logic counterpart of a formal varinfo that has
been replaced by a new one: propagate this change *)
been replaced by a new one: propagate this change *)
lvi
.
lv_id
<-
new_vi
.
vid
;
lvi
.
lv_id
<-
new_vi
.
vid
;
...
@@ -112,20 +101,12 @@ let dup_funspec tbl bhv spec =
...
@@ -112,20 +101,12 @@ let dup_funspec tbl bhv spec =
lvi
lvi
with
Not_found
->
with
Not_found
->
assert
vi
.
vglob
;
assert
vi
.
vglob
;
(* using [Visitor_behavior.Get.logic_var bhv lvi] is correct
lvi
)
only because the lv_id used to compare the lvi does not
change between the original one and this copy *)
Visitor_behavior
.
Get
.
logic_var
bhv
lvi
)
method
!
videntified_term
_
=
Cil
.
DoChildrenPost
Logic_const
.
refresh_identified_term
method
!
videntified_predicate
_
=
Cil
.
DoChildrenPost
Logic_const
.
refresh_predicate
end
in
end
in
Cil
.
visitCilFunspec
o
spec
Cil
.
visitCilFunspec
(
o
:>
Cil
.
cilVisitor
)
spec
let
dup_fundec
loc
spec
bhv
sound_verdict_vi
kf
vi
new_vi
=
let
dup_fundec
loc
spec
sound_verdict_vi
kf
vi
new_vi
=
new_vi
.
vdefined
<-
true
;
new_vi
.
vdefined
<-
true
;
let
formals
=
Kernel_function
.
get_formals
kf
in
let
formals
=
Kernel_function
.
get_formals
kf
in
let
mk_formal
vi
=
let
mk_formal
vi
=
...
@@ -174,9 +155,9 @@ let dup_fundec loc spec bhv sound_verdict_vi kf vi new_vi =
...
@@ -174,9 +155,9 @@ let dup_fundec loc spec bhv sound_verdict_vi kf vi new_vi =
let
locals
=
match
res
with
None
->
[]
|
Some
r
->
[
r
]
in
let
locals
=
match
res
with
None
->
[]
|
Some
r
->
[
r
]
in
let
body
=
Cil
.
mkBlock
stmts
in
let
body
=
Cil
.
mkBlock
stmts
in
body
.
blocals
<-
locals
;
body
.
blocals
<-
locals
;
let
tbl
=
Cil_datatype
.
Varinfo
.
Hashtbl
.
create
7
in
let
tbl
=
Varinfo
.
Hashtbl
.
create
7
in
List
.
iter2
(
Cil_datatype
.
Varinfo
.
Hashtbl
.
add
tbl
)
formals
new_formals
;
List
.
iter2
(
Varinfo
.
Hashtbl
.
add
tbl
)
formals
new_formals
;
let
new_spec
=
dup_funspec
tbl
bhv
spec
in
let
new_spec
=
dup_funspec
tbl
spec
in
let
fundec
=
let
fundec
=
{
svar
=
new_vi
;
{
svar
=
new_vi
;
sformals
=
new_formals
;
sformals
=
new_formals
;
...
@@ -192,54 +173,69 @@ let dup_fundec loc spec bhv sound_verdict_vi kf vi new_vi =
...
@@ -192,54 +173,69 @@ let dup_fundec loc spec bhv sound_verdict_vi kf vi new_vi =
Cfg
.
cfgFun
fundec
;
Cfg
.
cfgFun
fundec
;
fundec
fundec
let
dup_global
loc
actions
spec
bhv
sound_verdict_vi
kf
vi
new_vi
=
let
dup_global
loc
spec
sound_verdict_vi
kf
vi
new_vi
=
let
name
=
vi
.
vname
in
let
name
=
vi
.
vname
in
Options
.
feedback
~
dkey
~
level
:
2
"entering in function %s"
name
;
Options
.
feedback
~
dkey
~
level
:
2
"entering in function %s"
name
;
let
fundec
=
dup_fundec
loc
spec
bhv
sound_verdict_vi
kf
vi
new_vi
in
let
new_fundec
=
dup_fundec
loc
spec
sound_verdict_vi
kf
vi
new_vi
in
let
fct
=
Definition
(
fundec
,
loc
)
in
let
new_fct
=
Definition
(
new_fundec
,
loc
)
in
let
new_spec
=
fundec
.
sspec
in
let
new_spec
=
new_fundec
.
sspec
in
let
new_kf
=
{
fundec
=
fct
;
spec
=
new_spec
}
in
let
new_kf
=
{
fundec
=
new_fct
;
spec
=
new_spec
}
in
Queue
.
add
Globals
.
Functions
.
register
new_kf
;
(
fun
()
->
Globals
.
Functions
.
replace_by_definition
new_spec
new_fundec
loc
;
Kernel_function
.
Hashtbl
.
add
fct_tbl
new_kf
()
;
Annotations
.
register_funspec
new_kf
;
Globals
.
Functions
.
register
new_kf
;
if
Datatype
.
String
.
equal
new_vi
.
vname
"main"
then
begin
Globals
.
Functions
.
replace_by_definition
new_spec
fundec
loc
;
(* recompute kernel's info about the old main since its name has changed;
Annotations
.
register_funspec
new_kf
;
see the corresponding part in the main visitor *)
if
new_vi
.
vname
=
"main"
then
begin
Globals
.
Functions
.
remove
vi
;
(* recompute the info about the old main since its name has changed;
Globals
.
Functions
.
register
kf
see the corresponding part in the main visitor *)
end
;
Globals
.
Functions
.
remove
vi
;
(* copy property statuses to the new spec *)
Globals
.
Functions
.
register
kf
let
copy
old_ip
new_ip
=
end
)
let
open
Property_status
in
actions
;
let
cp
status
ep
=
(* remove the specs attached to the previous kf iff it is a definition:
let
e
=
Emitter
.
Usable_emitter
.
get
ep
.
emitter
in
it is necessary to keep stable the number of annotations in order to get
if
ep
.
logical_consequence
[Keep_status] working fine. *)
then
logical_consequence
e
new_ip
ep
.
properties
let
kf
=
Visitor_behavior
.
Get
.
kernel_function
bhv
kf
in
else
emit
e
new_ip
~
hyps
:
ep
.
properties
status
in
match
get
old_ip
with
|
Never_tried
->
()
|
Best
(
s
,
epl
)
->
List
.
iter
(
cp
s
)
epl
|
Inconsistent
icst
->
List
.
iter
(
cp
True
)
icst
.
valid
;
(* either the program is reachable and [False_and_reachable] is
fine, or the program point is not reachable and it does not
matter for E-ACSL that checks it at runtime. *)
List
.
iter
(
cp
False_and_reachable
)
icst
.
invalid
in
let
ips
kf
s
=
Property
.
ip_of_spec
kf
Kglobal
~
active
:
[]
s
in
List
.
iter2
copy
(
ips
kf
spec
)
(
ips
new_kf
new_spec
);
(* remove annotations from the old spec. Yet keep them in functions only
declared since, otherwise, the kernel would complain about functions
with neither contract nor body *)
if
Kernel_function
.
is_definition
kf
then
begin
if
Kernel_function
.
is_definition
kf
then
begin
Queue
.
add
let
old_bhvs
=
(
fun
()
->
Annotations
.
fold_behaviors
(
fun
e
b
acc
->
(
e
,
b
)
::
acc
)
kf
[]
let
bhvs
=
in
Annotations
.
fold_behaviors
(
fun
e
b
acc
->
(
e
,
b
)
::
acc
)
kf
[]
List
.
iter
in
(
fun
(
e
,
b
)
->
Annotations
.
remove_behavior
~
force
:
true
e
kf
b
)
List
.
iter
old_bhvs
;
(
fun
(
e
,
b
)
->
Annotations
.
remove_behavior
~
force
:
true
e
kf
b
)
Annotations
.
iter_decreases
bhvs
;
(
fun
e
_
->
Annotations
.
remove_decreases
e
kf
)
Annotations
.
iter_decreases
kf
;
(
fun
e
_
->
Annotations
.
remove_decreases
e
kf
)
Annotations
.
iter_terminates
kf
;
(
fun
e
_
->
Annotations
.
remove_terminates
e
kf
)
Annotations
.
iter_terminates
kf
;
(
fun
e
_
->
Annotations
.
remove_terminates
e
kf
)
Annotations
.
iter_complete
kf
;
(
fun
e
l
->
Annotations
.
remove_complete
e
kf
l
)
Annotations
.
iter_complete
kf
;
(
fun
e
l
->
Annotations
.
remove_complete
e
kf
l
)
Annotations
.
iter_disjoint
kf
;
(
fun
e
l
->
Annotations
.
remove_disjoint
e
kf
l
)
Annotations
.
iter_disjoint
kf
(
fun
e
l
->
Annotations
.
remove_disjoint
e
kf
l
)
kf
)
actions
end
;
end
;
GFun
(
fundec
,
loc
)
,
GFunDecl
(
new_spec
,
new_vi
,
loc
)
GFun
(
new_
fundec
,
loc
)
,
GFunDecl
(
new_spec
,
new_vi
,
loc
)
(* ********************************************************************** *)
(* ********************************************************************** *)
(* Alignment *)
(* Alignment *)
...
@@ -293,230 +289,187 @@ let require_alignment vi algn =
...
@@ -293,230 +289,187 @@ let require_alignment vi algn =
Cil
.
bitsSizeOf
vi
.
vtype
<
algn
*
8
&&
not
(
sufficiently_aligned
vi
algn
)
Cil
.
bitsSizeOf
vi
.
vtype
<
algn
*
8
&&
not
(
sufficiently_aligned
vi
algn
)
(* ********************************************************************** *)
(* ********************************************************************** *)
(* Visit
or
*)
(* Visit
ing the globals
*)
(* ********************************************************************** *)
(* ********************************************************************** *)
class
prepare_visitor
=
object
(
self
)
(* perform a few modifications in the [kf]'s code and annotations that are
inherit
Visitor
.
frama_c_inplace
required by E-ACSL's analyses and transformations *)
let
prepare_fundec
kf
=
val
mutable
has_new_stmt_in_fundec
=
false
let
o
=
object
inherit
Visitor
.
frama_c_inplace
(* ---------------------------------------------------------------------- *)
(* visitor's local variable *)
(* table for ensuring absence of term sharing *)
(* ---------------------------------------------------------------------- *)
val
terms
=
Misc
.
Id_term
.
Hashtbl
.
create
7
val
terms
=
Misc
.
Id_term
.
Hashtbl
.
create
7
(* substitute function's varinfos by their duplicate in function calls *)
(* table for ensuring absence of term sharing *)
method
!
vvrbl
vi
=
try
Cil
.
ChangeTo
(
Dup_functions
.
find
vi
)
with
Not_found
->
Cil
.
SkipChildren
(* temporal analysis requires that attributes are aligned to local
variables *)
method
!
vblock
_
=
if
Options
.
Temporal_validity
.
get
()
then
Cil
.
DoChildrenPost
(
fun
blk
->
List
.
iter
(
fun
vi
->
(* 4 bytes alignment is required to allow sufficient space for
storage of 32-bit timestamps in a 1:1 shadow. *)
if
require_alignment
vi
4
then
vi
.
vattr
<-
Attr
(
"aligned"
,
[
AInt
Integer
.
four
])
::
vi
.
vattr
)
blk
.
blocals
;
blk
)
else
Cil
.
DoChildren
val
unduplicable_functions
=
(* remove term sharing introduced by the Frama-C kernel. For instance, an
let
white_list
=
annotation such as [a <= t <= b] is replaced by [a <= t && t <= b] with
[
"__builtin_va_arg"
;
both occurrences of [t] being shared. Yet, the E-ACSL type system may
"__builtin_va_end"
;
require to assign them different types. *)
"__builtin_va_start"
;
method
!
vterm
_
=
"__builtin_va_copy"
]
in
List
.
fold_left
(
fun
acc
s
->
Datatype
.
String
.
Set
.
add
s
acc
)
Datatype
.
String
.
Set
.
empty
white_list
val
fct_tbl
=
Cil_datatype
.
Varinfo
.
Hashtbl
.
create
7
val
mutable
new_definitions
:
global
list
=
[]
(* new definitions of the annotated functions which will contain the
translation of the E-ACSL contract *)
(* the variable [sound_verdict] belongs to the E-ACSL's RTL and indicates
whether the verdict emitted by E-ACSL is sound. It needs to be visible at
that point because it is set in all function duplicates
(see [dup_fundec]). *)
val
mutable
sound_verdict_vi
=
let
name
=
Functions
.
RTL
.
mk_api_name
"sound_verdict"
in
let
vi
=
Cil
.
makeGlobalVar
name
Cil
.
intType
in
vi
.
vstorage
<-
Extern
;
vi
.
vreferenced
<-
true
;
vi
(* ---------------------------------------------------------------------- *)
(* visitor's private methods *)
(* ---------------------------------------------------------------------- *)
method
private
is_variadic_function
vi
=
match
Cil
.
unrollType
vi
.
vtype
with
|
TFun
(
_
,
_
,
variadic
,
_
)
->
variadic
|
_
->
true
(* ---------------------------------------------------------------------- *)
(* visitor's method overloading *)
(* ---------------------------------------------------------------------- *)
method
!
vlogic_info_decl
li
=
Global
.
add_logic_info
li
;
Cil
.
SkipChildren
method
!
vvrbl
vi
=
try
let
new_vi
=
Cil_datatype
.
Varinfo
.
Hashtbl
.
find
fct_tbl
vi
in
(* replace functions at callsite by its duplicated version *)
Cil
.
ChangeTo
new_vi
with
Not_found
->
Cil
.
SkipChildren
method
!
vterm
_t
=
Cil
.
DoChildrenPost
(
fun
t
->
if
Misc
.
Id_term
.
Hashtbl
.
mem
terms
t
then
(* remove term sharing for soundness of E-ACSL typing
(see typing.ml) *)
{
t
with
term_node
=
t
.
term_node
}
else
begin
Misc
.
Id_term
.
Hashtbl
.
add
terms
t
()
;
t
end
)
(* Add align attributes to local variables (required by temporal analysis) *)
method
!
vblock
_
=
if
Options
.
Temporal_validity
.
get
()
then
Cil
.
DoChildrenPost
Cil
.
DoChildrenPost
(
fun
blk
->
(
fun
t
->
List
.
iter
if
Misc
.
Id_term
.
Hashtbl
.
mem
terms
t
then
(
fun
vi
->
{
t
with
term_node
=
t
.
term_node
}
(* 4 bytes alignment is required to allow sufficient space for
else
begin
storage of 32-bit timestamps in a 1:1 shadow. *)
Misc
.
Id_term
.
Hashtbl
.
add
terms
t
()
;
if
require_alignment
vi
4
then
t
vi
.
vattr
<-
end
)
Attr
(
"aligned"
,
[
AInt
Integer
.
four
])
::
vi
.
vattr
)
blk
.
blocals
;
blk
)
else
Cil
.
DoChildren
method
!
vfunc
fundec
=
method
!
videntified_predicate
_
=
Cil
.
DoChildrenPost
(* when entering a new annotation, clear the context to keep it small:
(
fun
_
->
anyway, no sharing is possible between two identified predicates *)
if
has_new_stmt_in_fundec
then
begin
Misc
.
Id_term
.
Hashtbl
.
clear
terms
;
has_new_stmt_in_fundec
<-
false
;
(* recompute the CFG *)
Cfg
.
clearCFGinfo
~
clear_id
:
false
fundec
;
Cfg
.
cfgFun
fundec
;
end
;
fundec
)
method
!
vglob_aux
=
function
|
GFunDecl
(
_
,
vi
,
loc
)
|
GFun
({
svar
=
vi
}
,
loc
)
when
(* duplicate a function iff: *)
(* it is not already duplicated *)
not
(
Cil_datatype
.
Varinfo
.
Hashtbl
.
mem
fct_tbl
vi
)
&&
(* it is duplicable *)
not
(
Datatype
.
String
.
Set
.
mem
vi
.
vname
unduplicable_functions
)
&&
(* it is not a variadic function *)
not
(
self
#
is_variadic_function
vi
)
&&
(* it is not a built-in *)
not
(
Misc
.
is_fc_or_compiler_builtin
vi
)
&&
(
let
kf
=
try
Globals
.
Functions
.
get
vi
with
Not_found
->
assert
false
in
(* either explicitely listed as to be not instrumented *)
not
(
Functions
.
instrument
kf
)
||
(* or: *)
((
*
it
has
a
function
contract
*
)
not
(
Cil
.
is_empty_funspec
(
Annotations
.
funspec
~
populate
:
false
(
Extlib
.
the
self
#
current_kf
)))
&&
(* its annotations must be monitored *)
Functions
.
check
kf
))
&&
(* it is neither malloc nor free *)
(* [TODO:] this special case preserves the former behavior.
Should be generalized latter by considering only preconditions of
libc functions *)
vi
.
vname
<>
"malloc"
&&
vi
.
vname
<>
"calloc"
&&
vi
.
vname
<>
"realloc"
&&
vi
.
vname
<>
"free"
->
let
name
=
Functions
.
RTL
.
mk_gen_name
vi
.
vname
in
let
new_vi
=
Cil
.
makeGlobalVar
name
vi
.
vtype
in
Cil_datatype
.
Varinfo
.
Hashtbl
.
add
fct_tbl
vi
new_vi
;
Cil
.
DoChildrenPost
(
fun
l
->
match
l
with
|
[
GFunDecl
(
_
,
vi
,
_
)
|
GFun
({
svar
=
vi
}
,
_
)
as
g
]
->
let
kf
=
Extlib
.
the
self
#
current_kf
in
(
match
g
with
|
GFunDecl
_
->
if
not
(
Kernel_function
.
is_definition
kf
)
then
Options
.
warning
"@[annotating undefined function `%a':@ \
the generated program may miss memory instrumentation@ \
if there are memory-related annotations.@]"
Printer
.
pp_varinfo
vi
|
GFun
_
->
()
|
_
->
assert
false
);
let
tmp
=
vi
.
vname
in
if
tmp
=
"main"
then
begin
(* the new function becomes the new main: *)
(* 1. swap the name of both functions *)
vi
.
vname
<-
new_vi
.
vname
;
new_vi
.
vname
<-
tmp
;
(* 2. force recomputation of the entry point if necessary *)
if
Kernel
.
MainFunction
.
get
()
=
tmp
then
begin
let
selection
=
State_selection
.
with_dependencies
Kernel
.
MainFunction
.
self
in
Project
.
clear
~
selection
()
end
(* 3. recompute what is necessary in [Globals.Functions]:
done in [dup_global] *)
end
;
let
new_g
,
new_decl
=
dup_global
loc
self
#
get_filling_actions
(
Annotations
.
funspec
~
populate
:
false
kf
)
self
#
behavior
sound_verdict_vi
kf
vi
new_vi
in
(* postpone the introduction of the new function definition to the
end *)
new_definitions
<-
new_g
::
new_definitions
;
(* put the declaration before the original function in order to
solve issue with recursive functions *)
[
new_decl
;
g
]
|
_
->
assert
false
)
|
_
->
Cil
.
DoChildren
Cil
.
DoChildren
method
!
vfile
f
=
end
in
Cil
.
DoChildrenPost
ignore
(
Visitor
.
visitFramacKf
o
kf
)
(
fun
_
->
match
new_definitions
with
(* the variable [sound_verdict] belongs to the E-ACSL's RTL and indicates
|
[]
->
f
whether the verdict emitted by E-ACSL is sound. It needs to be visible at
|
_
::
_
->
that point because it is set in all function duplicates (see
(* add the generated definitions of libc at the end of
[dup_fundec]). *)
[new_definitions]. This way, we are sure that they have access to
let
sound_verdict_vi
=
all of it (in particular, the memory model, GMP and the soundness
lazy
variable). Also add the [__e_acsl_sound_verdict] variable at the
(
let
name
=
Functions
.
RTL
.
mk_api_name
"sound_verdict"
in
beginning *)
let
vi
=
Cil
.
makeGlobalVar
name
Cil
.
intType
in
let
new_globals
=
vi
.
vstorage
<-
Extern
;
GVarDecl
(
sound_verdict_vi
,
Cil_datatype
.
Location
.
unknown
)
vi
.
vreferenced
<-
true
;
::
f
.
globals
vi
)
@
new_definitions
in
let
is_variadic_function
vi
=
match
Cil
.
unrollType
vi
.
vtype
with
f
.
globals
<-
new_globals
;
|
TFun
(
_
,
_
,
variadic
,
_
)
->
variadic
f
)
|
_
->
false
initializer
(* set of functions that must never be duplicated *)
reset
()
let
unduplicable_functions
=
let
white_list
=
end
[
"__builtin_va_arg"
;
"__builtin_va_end"
;
"__builtin_va_start"
;
"__builtin_va_copy"
;
(* *alloc and free should not be duplicated. *)
(* [TODO:] it preserves the former behavior. Should be modified latter by
checking only preconditions for any libc functions *)
"malloc"
;
"calloc"
;
"realloc"
;
"free"
]
in
List
.
fold_left
(
fun
acc
s
->
Datatype
.
String
.
Set
.
add
s
acc
)
Datatype
.
String
.
Set
.
empty
white_list
let
must_duplicate
kf
vi
=
(* it is not already duplicated *)
not
(
Dup_functions
.
mem
vi
)
&&
(* it is duplicable *)
not
(
Datatype
.
String
.
Set
.
mem
vi
.
vname
unduplicable_functions
)
&&
(* it is not a variadic function *)
not
(
is_variadic_function
vi
)
&&
(* it is not a built-in *)
not
(
Misc
.
is_fc_or_compiler_builtin
vi
)
&&
((
*
either
explicitely
listed
as
to
be
not
instrumented
*
)
not
(
Functions
.
instrument
kf
)
||
(* or: *)
((
*
it
has
a
function
contract
*
)
not
(
Cil
.
is_empty_funspec
(
Annotations
.
funspec
~
populate
:
false
kf
))
&&
(* its annotations must be monitored *)
Functions
.
check
kf
))
let
prepare_global
(
globals
,
new_defs
)
=
function
|
GFunDecl
(
_
,
vi
,
loc
)
|
GFun
({
svar
=
vi
}
,
loc
)
as
g
->
let
kf
=
try
Globals
.
Functions
.
get
vi
with
Not_found
->
assert
false
in
if
must_duplicate
kf
vi
then
begin
let
new_vi
=
Dup_functions
.
generate_vi
vi
in
if
Kernel_function
.
is_definition
kf
then
prepare_fundec
kf
else
(* TODO: this warning could be more precise if emitted during code
generation; see also E-ACSL issue #85 about partial verdicts *)
Options
.
warning
"@[annotating undefined function `%a':@ \
the generated program may miss memory instrumentation@ \
if there are memory-related annotations.@]"
Printer
.
pp_varinfo
vi
;
let
tmp
=
vi
.
vname
in
if
Datatype
.
String
.
equal
tmp
"main"
then
begin
(* the new function becomes the new main: *)
(* 1. swap the name of both functions *)
vi
.
vname
<-
new_vi
.
vname
;
new_vi
.
vname
<-
tmp
;
(* 2. force recomputation of the entry point if necessary *)
if
Kernel
.
MainFunction
.
get
()
=
tmp
then
begin
let
selection
=
State_selection
.
with_dependencies
Kernel
.
MainFunction
.
self
in
Project
.
clear
~
selection
()
end
end
;
let
new_def
,
new_decl
=
dup_global
loc
(
Annotations
.
funspec
~
populate
:
false
kf
)
(
Lazy
.
force
sound_verdict_vi
)
kf
vi
new_vi
in
(* postpone the introduction of the new function definition *)
let
new_defs
=
new_def
::
new_defs
in
(* put the declaration before the original function in order to solve
issues with recursive functions *)
g
::
new_decl
::
globals
,
new_defs
end
else
begin
(* not duplicated *)
prepare_fundec
kf
;
g
::
globals
,
new_defs
end
|
g
->
(* nothing to do for globals that are not functions *)
g
::
globals
,
new_defs
let
prepare_file
file
=
Dup_functions
.
reset
()
;
let
rev_globals
,
new_defs
=
List
.
fold_left
prepare_global
([]
,
[]
)
file
.
globals
in
match
new_defs
with
|
[]
->
()
|
_
::
_
->
(* insert the new_definitions at the end and reverse back the globals *)
let
globals
=
List
.
fold_left
(
fun
acc
g
->
g
::
acc
)
new_defs
rev_globals
in
(* insert [__e_acsl_sound_verdict] at the beginning *)
let
sg
=
GVarDecl
(
Lazy
.
force
sound_verdict_vi
,
Location
.
unknown
)
in
file
.
globals
<-
sg
::
globals
let
prepare
()
=
let
prepare
()
=
Options
.
feedback
~
level
:
2
"prepare AST for E-ACSL transformations"
;
Options
.
feedback
~
level
:
2
"prepare AST for E-ACSL transformations"
;
Visitor
.
visitFramacFile
(
new
prepare_visitor
)
(
Ast
.
get
()
);
prepare_file
(
Ast
.
get
()
);
Queue
.
iter
(
fun
f
->
f
()
)
actions
;
Ast
.
mark_as_grown
()
Ast
.
mark_as_grown
()
(*
(*
...
...
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment