@ -12,25 +12,21 @@ open! Utils
module F = Format
module F = Format
module L = Logging
module L = Logging
module PPrawpath = PrettyPrintable . MakePPSet ( struct
type t = AccessPath . raw
let compare = AccessPath . raw_compare
let pp_element = AccessPath . pp_raw
end )
module LocksDomain = AbstractDomain . FiniteSet ( StringPPSet )
module PathDomain = AbstractDomain . FiniteSet ( PPrawpath )
module Summary = Summary . Make ( struct
type summary = ThreadSafetyDomain . astate
module ReadWriteDomain = AbstractDomain . Pair ( PathDomain ) ( PathDomain )
let update_payload astate payload =
{ payload with Specs . threadsafety = Some astate }
module CombinedDomain = AbstractDomain . Pair ( LocksDomain ) ( ReadWriteDomain )
let read_from_payload payload =
(* a typical element is ( ) {locked}, {vars and fields} ) *)
payload . Specs . threadsafety
end )
module TransferFunctions ( CFG : ProcCfg . S ) = struct
module TransferFunctions ( CFG : ProcCfg . S ) = struct
module CFG = CFG
module CFG = CFG
module Domain = Combined Domain
module Domain = ThreadSafety Domain
type extras = ProcData . no_extras
type extras = ProcData . no_extras
let is_lock_procedure pn = Procname . equal pn BuiltinDecl . __set_locked_attribute
let is_lock_procedure pn = Procname . equal pn BuiltinDecl . __set_locked_attribute
@ -39,22 +35,24 @@ module TransferFunctions (CFG : ProcCfg.S) = struct
let add_path_to_state exp typ pathdomainstate =
let add_path_to_state exp typ pathdomainstate =
IList . fold_left
IList . fold_left
( fun pathdomainstate_acc rawpath -> PathDomain . add rawpath pathdomainstate_acc )
( fun pathdomainstate_acc rawpath -> ThreadSafetyDomain . PathDomain . add
rawpath pathdomainstate_acc )
pathdomainstate
pathdomainstate
( AccessPath . of_exp exp typ ~ f_resolve_id : ( fun _ -> None ) )
( AccessPath . of_exp exp typ ~ f_resolve_id : ( fun _ -> None ) )
let exec_instr ( ( lockstate , ( readstate , writestate ) ) as astate ) { ProcData . pdesc ; } _ =
let exec_instr ( ( lockstate , ( readstate , writestate ) ) as astate ) { ProcData . pdesc ; } _ =
let is_unprotected lockstate =
let is_unprotected lockstate =
( not ( Procdesc . is_java_synchronized pdesc ) ) && ( LocksDomain . is_empty lockstate )
( not ( Procdesc . is_java_synchronized pdesc ) ) &&
( ThreadSafetyDomain . LocksDomain . is_empty lockstate )
in
in
function
function
| Sil . Call ( _ , Const ( Cfun pn ) , _ , _ , _ ) ->
| Sil . Call ( _ , Const ( Cfun pn ) , _ , _ , _ ) ->
if is_lock_procedure pn
if is_lock_procedure pn
then
then
( ( LocksDomain. add " locked " lockstate ) , ( readstate , writestate ) )
( ( ThreadSafetyDomain. LocksDomain. add " locked " lockstate ) , ( readstate , writestate ) )
else if is_unlock_procedure pn
else if is_unlock_procedure pn
then
then
( ( LocksDomain. remove " locked " lockstate ) , ( readstate , writestate ) )
( ( ThreadSafetyDomain. LocksDomain. remove " locked " lockstate ) , ( readstate , writestate ) )
else
else
astate
astate
@ -83,19 +81,25 @@ module Analyzer =
( Scheduler . ReversePostorder )
( Scheduler . ReversePostorder )
( TransferFunctions )
( TransferFunctions )
let method_analysis { Callbacks . proc_desc ; tenv ; } =
module Interprocedural = Analyzer . Interprocedural ( Summary )
match Analyzer . compute_post ( ProcData . make_default proc_desc tenv ) with
(* This is a "checker" *)
let method_analysis callback =
let proc_desc = callback . Callbacks . proc_desc in
let opost = Interprocedural . checker callback ProcData . empty_extras in
match opost with
| Some post -> (* I am printing to commandline and out to cater to javac and buck *)
| Some post -> (* I am printing to commandline and out to cater to javac and buck *)
( L . stdout " \n Procedure: %s@ "
( L . stdout " \n Procedure: %s@ "
( Procname . to_string ( Procdesc . get_proc_name proc_desc ) )
( Procname . to_string ( Procdesc . get_proc_name proc_desc ) )
) ;
) ;
L . stdout " \n POST: %a \n " CombinedDomain . pp post ;
L . stdout " \n POST: %a \n " ThreadSafety Domain. pp post ;
( L . out " \n Procedure: %s@ "
( L . out " \n Procedure: %s@ "
( Procname . to_string ( Procdesc . get_proc_name proc_desc ) )
( Procname . to_string ( Procdesc . get_proc_name proc_desc ) )
) ;
) ;
L . out " \n POST: %a \n " Combined Domain. pp post
L . out " \n POST: %a \n " ThreadSafety Domain. pp post
| None -> ()
| None -> ()
(* a results table is a Map where a key is an a procedure environment,
(* a results table is a Map where a key is an a procedure environment,
i . e . , something of type Idenv . t * Tenv . t * Procname . t * Procdesc . t
i . e . , something of type Idenv . t * Tenv . t * Procname . t * Procdesc . t
* )
* )
@ -111,8 +115,9 @@ let should_analyze_proc (_,tenv,proc_name,proc_desc) =
not ( Procname . is_class_initializer proc_name ) &&
not ( Procname . is_class_initializer proc_name ) &&
Procdesc . get_access proc_desc < > PredSymb . Private
Procdesc . get_access proc_desc < > PredSymb . Private
(* creates a map from proc_envs to postconditions *)
(* creates a map from proc_envs to postconditions *)
let make_results_table file_env =
let make_results_table get_proc_desc file_env =
let procs_to_analyze = IList . filter should_analyze_proc file_env
let procs_to_analyze = IList . filter should_analyze_proc file_env
in
in
(* make a Map sending each element e of list l to ( f e ) *)
(* make a Map sending each element e of list l to ( f e ) *)
@ -121,10 +126,13 @@ let make_results_table file_env =
) ResultsTableType . empty l
) ResultsTableType . empty l
in
in
let compute_post_for_procedure = (* takes proc_env as arg *)
let compute_post_for_procedure = (* takes proc_env as arg *)
fun ( _ , tenv , _ , proc_desc ) ->
fun ( idenv , tenv , proc_name , proc_desc ) ->
match Analyzer . compute_post ( ProcData . make_default proc_desc tenv ) with
let callback_arg =
{ Callbacks . get_proc_desc ; get_procs_in_file = ( fun _ -> [] ) ;
idenv ; tenv ; proc_name ; proc_desc } in
match Interprocedural . checker callback_arg ProcData . empty_extras with
| Some post -> post
| Some post -> post
| None -> CombinedDomain . initial
| None -> ThreadSafety Domain. initial
in
in
map_post_computation_over_procs compute_post_for_procedure procs_to_analyze
map_post_computation_over_procs compute_post_for_procedure procs_to_analyze
@ -165,7 +173,7 @@ let report_thread_safety_errors ( _, tenv, pname, pdesc) writestate =
( Procdesc . get_loc pdesc )
( Procdesc . get_loc pdesc )
description
description
in
in
IList . iter report_one_error ( IList . map snd ( PathDomain. elements writestate ) )
IList . iter report_one_error ( IList . map snd ( ThreadSafetyDomain. PathDomain. elements writestate ) )
(* For now, just checks if there is one active element amongst the posts of the analyzed methods.
(* For now, just checks if there is one active element amongst the posts of the analyzed methods.
@ -188,13 +196,14 @@ let should_analyze_file file_env =
in
in
IList . exists current_class_or_super_marked_threadsafe file_env
IList . exists current_class_or_super_marked_threadsafe file_env
(* This is a "cluster checker" *)
(* Gathers results by analyzing all the methods in a file, then post-processes
(* Gathers results by analyzing all the methods in a file, then post-processes
the results to check ( approximation of ) thread safety * )
the results to check ( approximation of ) thread safety * )
(* file_env: ( Idenv.t * Tenv.t * Procname.t * Procdesc.t ) list *)
(* file_env: ( Idenv.t * Tenv.t * Procname.t * Procdesc.t ) list *)
let file_analysis _ _ _ file_env =
let file_analysis _ _ get_procdesc file_env =
if should_analyze_file file_env then
if should_analyze_file file_env then
process_results_table
process_results_table
( make_results_table file_env)
( make_results_table get_procdesc file_env)
(*
(*
Todo :
Todo :