(* * 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 alsways 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 (* Naming scheme of .cfg file matters for OndemandCapture module. If it changes here, it should be changed there as well*) let cfg_file = DB.source_dir_get_internal_file source_dir ".cfg" 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_cfg_to_file ~source_file cfg_file cfg ; Tenv.sort_fields_tenv tenv ; Tenv.store_to_file tenv_file tenv ; if Config.stats_mode then Cfg.check_cfg_connectedness cfg ; if Config.stats_mode || 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 ; (* NOTE: nothing should be written to source_dir after this *) DB.mark_file_updated (DB.source_dir_to_string source_dir)