You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

85 lines
3.6 KiB

(*
* Copyright (c) 2016-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*)
open! IStd
module TestInterpreter =
AnalyzerTester.Make (ProcCfg.Exceptional) (BoundedCallTree.TransferFunctions)
let tests =
let open OUnit2 in
let open AnalyzerTester.StructuredSil in
let initial = BoundedCallTree.Domain.empty in
let f_proc_name = Typ.Procname.from_string_c_fun "f" in
let g_proc_name = Typ.Procname.from_string_c_fun "g" in
let g_args = [(Exp.Const (Const.Cint IntLit.one), Typ.mk (Tint IInt))] in
let g_return = (ident_of_str "r", Typ.mk (Tint IInt)) in
let class_name = "com.example.SomeClass" in
let file_name = "SomeClass.java" in
let trace =
Stacktrace.make "java.lang.NullPointerException"
[ Stacktrace.make_frame class_name "foo" file_name (Some 16)
; Stacktrace.make_frame class_name "bar" file_name (Some 20) ]
in
let extras = {BoundedCallTree.stacktraces= [trace]} in
let multi_trace_1 =
Stacktrace.make "java.lang.NullPointerException"
[Stacktrace.make_frame class_name "foo" file_name (Some 16)]
in
let multi_trace_2 =
Stacktrace.make "java.lang.NullPointerException"
[Stacktrace.make_frame class_name "bar" file_name (Some 20)]
in
let multi_trace_extras = {BoundedCallTree.stacktraces= [multi_trace_1; multi_trace_2]} in
let caller_foo_name = Typ.Procname.from_string_c_fun "foo" in
let caller_bar_name = Typ.Procname.from_string_c_fun "bar" in
let caller_baz_name = Typ.Procname.from_string_c_fun "baz" in
let test_list_from_foo =
[ ( "on_call_add_proc_name"
, [make_call ~procname:f_proc_name []; (* means f() *) invariant "{ f }"] )
; ( "on_call_add_proc_name_w_args"
, [ make_call ~procname:g_proc_name ~return:g_return g_args
; (* means r = a.g(1) *)
invariant "{ g }" ] )
; ( "handle_two_proc_calls"
, [ make_call ~procname:f_proc_name []
; invariant "{ f }"
; make_call ~procname:g_proc_name ~return:g_return g_args
; invariant "{ f, g }" ] )
; ( "dont_record_procs_twice"
, [ make_call ~procname:f_proc_name []
; invariant "{ f }"
; make_call ~procname:f_proc_name []
; invariant "{ f }" ] ) ]
|> TestInterpreter.create_tests ~test_pname:caller_foo_name
~initial:BoundedCallTree.Domain.empty extras
in
let test_list_from_bar =
[ ( "on_call_anywhere_on_stack_add_proc_name"
, [make_call ~procname:f_proc_name []; (* means f() *) invariant "{ f }"] ) ]
|> TestInterpreter.create_tests ~test_pname:caller_bar_name extras ~initial
in
let test_list_from_baz =
[ ( "ignore_procs_unrelated_to_trace"
, [make_call ~procname:f_proc_name []; (* means f() *) invariant "{ }"] ) ]
|> TestInterpreter.create_tests ~test_pname:caller_baz_name extras ~initial
in
let test_list_multiple_traces_from_foo =
[ ( "on_call_add_proc_name_in_any_stack_1"
, [make_call ~procname:f_proc_name []; (* means f() *) invariant "{ f }"] ) ]
|> TestInterpreter.create_tests ~test_pname:caller_foo_name multi_trace_extras ~initial
in
let test_list_multiple_traces_from_bar =
[ ( "on_call_add_proc_name_in_any_stack_2"
, [make_call ~procname:f_proc_name []; (* means f() *) invariant "{ f }"] ) ]
|> TestInterpreter.create_tests ~test_pname:caller_bar_name multi_trace_extras ~initial
in
let test_list =
test_list_from_foo @ test_list_from_bar @ test_list_from_baz
@ test_list_multiple_traces_from_foo @ test_list_multiple_traces_from_bar
in
"bounded_calltree_test_suite" >::: test_list