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.

75 lines
3.1 KiB

(*
* 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 L = Logging
module rec CTransImpl : CModule_type.CTranslation = CTrans.CTrans_funct (CFrontend_declImpl)
and CFrontend_declImpl : CModule_type.CFrontend = CFrontend_decl.CFrontend_decl_funct (CTransImpl)
(* Translates a file by translating the ast into a cfg. *)
let compute_icfg trans_unit_ctx tenv ast =
match ast with
| Clang_ast_t.TranslationUnitDecl (_, decl_list, _, _) ->
CFrontend_config.global_translation_unit_decls := decl_list ;
L.(debug Capture Verbose) "@\n Start creating icfg@\n" ;
let cg = Cg.create trans_unit_ctx.CFrontend_config.source_file in
let cfg = Cfg.create_cfg () in
List.iter
~f:(CFrontend_declImpl.translate_one_declaration trans_unit_ctx tenv cg cfg `DeclTraversal)
decl_list ;
L.(debug Capture Verbose) "@\n Finished creating icfg@\n" ;
(cg, cfg)
| _ ->
assert false
(* NOTE: Assumes that an AST always starts with a TranslationUnitDecl *)
let init_global_state_capture () =
Ident.NameGenerator.reset () ;
CFrontend_config.global_translation_unit_decls := [] ;
CProcname.reset_block_counter ()
let do_source_file translation_unit_context ast =
let tenv = Tenv.create () in
CType_decl.add_predefined_types tenv ;
init_global_state_capture () ;
let source_file = translation_unit_context.CFrontend_config.source_file in
L.(debug Capture Verbose)
"@\n Start building call/cfg graph for '%a'....@\n" SourceFile.pp source_file ;
let call_graph, cfg = compute_icfg translation_unit_context tenv ast in
L.(debug Capture Verbose)
"@\n End building call/cfg graph for '%a'.@\n" SourceFile.pp source_file ;
(* This part below is a boilerplate in every frontends. *)
(* This could be moved in the cfg_infer module *)
let source_dir = DB.source_dir_from_source_file source_file in
let tenv_file = DB.source_dir_get_internal_file source_dir ".tenv" in
let cg_file = DB.source_dir_get_internal_file source_dir ".cg" in
NullabilityPreanalysis.analysis cfg tenv ;
Cg.store_to_file cg_file call_graph ;
Cfg.store source_file cfg ;
Tenv.sort_fields_tenv tenv ;
Tenv.store_to_file tenv_file tenv ;
if Config.debug_mode then Cfg.check_cfg_connectedness cfg ;
if Config.debug_mode || Config.testing_mode || Config.frontend_tests
|| Option.is_some Config.icfg_dotty_outfile
then (
Dotty.print_icfg_dotty source_file cfg ;
Cg.save_call_graph_dotty source_file call_graph ) ;
L.(debug Capture Verbose) "%a" Cfg.pp_proc_signatures cfg ;
L.(debug Capture Verbose)
"# Procedures started: %d@\n# Procedures completed: %d@\n@\n"
!CFrontend_config.procedures_attempted
(!CFrontend_config.procedures_attempted - !CFrontend_config.procedures_failed) ;
(* NOTE: nothing should be written to source_dir after this *)
DB.mark_file_updated (DB.source_dir_to_string source_dir)