|
|
|
(*
|
|
|
|
* Copyright (c) 2013 - present Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
*)
|
|
|
|
|
|
|
|
open! IStd
|
|
|
|
module F = Format
|
|
|
|
|
|
|
|
(** Module that contains constants and global state used in the frontend *)
|
|
|
|
|
|
|
|
type clang_lang = C | CPP | ObjC | ObjCPP [@@deriving compare]
|
|
|
|
|
|
|
|
let string_of_clang_lang (lang: clang_lang) : string =
|
|
|
|
match lang with C -> "C" | CPP -> "CPP" | ObjC -> "ObjC" | ObjCPP -> "ObjCPP"
|
|
|
|
|
|
|
|
|
|
|
|
let equal_clang_lang = [%compare.equal : clang_lang]
|
|
|
|
|
|
|
|
type exception_details =
|
|
|
|
{ msg: string
|
|
|
|
; position: string * int * int * int
|
|
|
|
; source_range: Clang_ast_t.source_range
|
|
|
|
; ast_node: string option }
|
|
|
|
|
|
|
|
exception Unimplemented of exception_details
|
|
|
|
|
|
|
|
let unimplemented position source_range ?ast_node fmt =
|
|
|
|
F.kasprintf (fun msg -> raise (Unimplemented {msg; position; source_range; ast_node})) fmt
|
|
|
|
|
|
|
|
|
|
|
|
exception IncorrectAssumption of exception_details
|
|
|
|
|
|
|
|
let incorrect_assumption position source_range ?ast_node fmt =
|
|
|
|
F.kasprintf (fun msg -> raise (IncorrectAssumption {msg; position; source_range; ast_node})) fmt
|
|
|
|
|
|
|
|
|
|
|
|
type translation_unit_context = {lang: clang_lang; source_file: SourceFile.t}
|
|
|
|
|
|
|
|
(** Constants *)
|
|
|
|
|
|
|
|
let alloc = "alloc"
|
|
|
|
|
|
|
|
let array_with_objects_count_m = "arrayWithObjects:count:"
|
|
|
|
|
|
|
|
let assert_fail = "__assert_fail"
|
|
|
|
|
|
|
|
let assert_rtn = "__assert_rtn"
|
|
|
|
|
|
|
|
let biniou_buffer_size =
|
|
|
|
(* the C++ standard suggests that implementation should support string literals up to this length *)
|
|
|
|
65535
|
|
|
|
|
|
|
|
|
|
|
|
let builtin_expect = "__builtin_expect"
|
|
|
|
|
|
|
|
let builtin_memset_chk = "__builtin___memset_chk"
|
|
|
|
|
|
|
|
let builtin_object_size = "__builtin_object_size"
|
|
|
|
|
|
|
|
let ckcomponent_cl = "CKComponent"
|
|
|
|
|
|
|
|
let ckcomponentcontroller_cl = "CKComponentController"
|
|
|
|
|
|
|
|
(** script to run our own clang *)
|
|
|
|
let clang_bin xx =
|
|
|
|
Config.bin_dir ^/ Filename.parent_dir_name ^/ Filename.parent_dir_name
|
|
|
|
^/ "facebook-clang-plugins" ^/ "clang" ^/ "install" ^/ "bin" ^/ "clang" ^ xx
|
|
|
|
|
|
|
|
|
|
|
|
let class_method = "class"
|
|
|
|
|
|
|
|
let fbAssertWithSignalAndLogFunctionHelper = "FBAssertWithSignalAndLogFunctionHelper"
|
|
|
|
|
|
|
|
let google_LogMessageFatal = "google::LogMessageFatal_LogMessageFatal"
|
|
|
|
|
|
|
|
let google_MakeCheckOpString = "google::MakeCheckOpString"
|
|
|
|
|
|
|
|
let handleFailureInFunction = "handleFailureInFunction:file:lineNumber:description:"
|
|
|
|
|
|
|
|
let handleFailureInMethod = "handleFailureInMethod:object:file:lineNumber:description:"
|
|
|
|
|
|
|
|
let id_cl = "id"
|
|
|
|
|
|
|
|
let infer = "infer"
|
|
|
|
|
|
|
|
let infer_skip_fun = "__infer_skip_function"
|
|
|
|
|
|
|
|
let infer_skip_gcc_asm_stmt = "__infer_skip_gcc_asm_stmt"
|
|
|
|
|
|
|
|
let init = "init"
|
|
|
|
|
|
|
|
let is_kind_of_class = "isKindOfClass:"
|
|
|
|
|
|
|
|
let malloc = "malloc"
|
|
|
|
|
|
|
|
let new_str = "new"
|
|
|
|
|
|
|
|
let next_object = "nextObject"
|
|
|
|
|
|
|
|
let nsproxy_cl = "NSProxy"
|
|
|
|
|
|
|
|
let nsobject_cl = "NSObject"
|
|
|
|
|
|
|
|
let nsstring_cl = "NSString"
|
|
|
|
|
|
|
|
let objc_class = "objc_class"
|
|
|
|
|
|
|
|
let objc_object = "objc_object"
|
|
|
|
|
|
|
|
let return_param = "__return_param"
|
|
|
|
|
|
|
|
let self = "self"
|
|
|
|
|
|
|
|
let std_addressof = QualifiedCppName.Match.of_fuzzy_qual_names ["std::addressof"]
|
|
|
|
|
|
|
|
let string_with_utf8_m = "stringWithUTF8String:"
|
|
|
|
|
|
|
|
let this = "this"
|
|
|
|
|
|
|
|
let replace_with_deref_first_arg_attr = "__infer_replace_with_deref_first_arg"
|
|
|
|
|
|
|
|
let modeled_function_attributes = [replace_with_deref_first_arg_attr]
|
|
|
|
|
|
|
|
(** Global state *)
|
|
|
|
|
|
|
|
let enum_map = ref ClangPointers.Map.empty
|
|
|
|
|
|
|
|
let global_translation_unit_decls : Clang_ast_t.decl list ref = ref []
|
|
|
|
|
|
|
|
let log_out = ref Format.std_formatter
|
|
|
|
|
|
|
|
let sil_types_map = ref Clang_ast_extend.TypePointerMap.empty
|
|
|
|
|
|
|
|
let procedures_attempted = ref 0
|
|
|
|
|
|
|
|
let procedures_failed = ref 0
|
|
|
|
|
|
|
|
let reset_global_state () =
|
|
|
|
enum_map := ClangPointers.Map.empty ;
|
|
|
|
global_translation_unit_decls := [] ;
|
|
|
|
log_out := Format.std_formatter ;
|
|
|
|
sil_types_map := Clang_ast_extend.TypePointerMap.empty ;
|
|
|
|
procedures_attempted := 0 ;
|
|
|
|
procedures_failed := 0
|
|
|
|
|
|
|
|
|
|
|
|
let tableaux_evaluation = false
|