(* * Copyright (c) 2016 - 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! Utils (** Top-level driver that orchestrates build system integration, frontends, and backend *) let set_env_for_clang_wrapper () = (match Config.clang_include_to_override with | Some dir -> Unix.putenv "FCP_CLANG_INCLUDE_TO_REPLACE" dir | None -> () ); if Config.cxx_experimental then Unix.putenv "FCP_INFER_CXX_MODELS" "1" ; if Config.debug_mode || Config.frontend_stats then Unix.putenv "FCP_DEBUG_MODE" "1" ; if not Config.failures_allowed then Unix.putenv "FCP_REPORT_FRONTEND_FAILURE" "1" ; if Config.llvm then Unix.putenv "LLVM_MODE" "1" (** as the Config.fail_on_bug flag mandates, exit with error when an issue is reported *) let fail_on_issue_epilogue () = let issues_json = DB.Results_dir.(path_to_filename Abs_root ["report.json"]) in match read_file (DB.filename_to_string issues_json) with | Some lines -> let issues = Jsonbug_j.report_of_string @@ String.concat "" lines in if issues <> [] then exit Config.fail_on_issue_exit_code | None -> () let () = set_env_for_clang_wrapper () ; (* The infer executable in the bin directory is a symbolic link to the real binary in the lib directory, so that the python script in the lib directory can be found relative to it. Packaging may also create longer symlink chains to the real executable, hence the recursion. *) let real_exe = let rec real_path path = match Unix.readlink path with | link when Filename.is_relative link -> (* [path] is a relative symbolic link *) real_path ((Filename.dirname path) // link) | link -> (* [path] is an absolute symbolic link *) real_path link | exception Unix.Unix_error(Unix.EINVAL, _, _) -> (* [path] is not a symbolic link *) path in real_path Sys.executable_name in let infer_py = (Filename.dirname real_exe) // "python" // "infer.py" in let build_cmd = IList.rev Config.rest in let in_buck_mode = match build_cmd with "buck" :: _ -> true | _ -> false in let args_py = Array.of_list ( infer_py :: Config.anon_args @ (if not Config.absolute_paths then [] else ["--absolute-paths"]) @ (match Config.analyzer with None -> [] | Some a -> ["--analyzer"; IList.assoc (=) a (IList.map (fun (n,a) -> (a,n)) Config.string_to_analyzer)]) @ (match Config.blacklist with | Some s when in_buck_mode -> ["--blacklist-regex"; s] | _ -> []) @ (if not Config.create_harness then [] else ["--android-harness"]) @ (if not Config.buck then [] else ["--buck"]) @ (match IList.rev Config.buck_build_args with | args when in_buck_mode -> IList.map (fun arg -> ["--Xbuck"; "'" ^ arg ^ "'"]) args |> IList.flatten | _ -> []) @ (if not Config.continue_capture then [] else ["--continue"]) @ (if not Config.debug_mode then [] else ["--debug"]) @ (if not Config.debug_exceptions then [] else ["--debug-exceptions"]) @ (if Config.filtering then [] else ["--no-filtering"]) @ (if not Config.frontend_debug then [] else ["--frontend-debug"]) @ (if not Config.frontend_stats then [] else ["--frontend-stats"]) @ (if not Config.flavors || not in_buck_mode then [] else ["--use-flavors"]) @ (if Option.is_none Config.use_compilation_database || not in_buck_mode then [] else ["--use-compilation-database"]) @ (match Config.infer_cache with None -> [] | Some s -> ["--infer_cache"; s]) @ "-j" :: (string_of_int Config.jobs) :: (match Config.load_average with None -> [] | Some f -> ["-l"; string_of_float f]) @ (if not Config.pmd_xml then [] else ["--pmd-xml"]) @ (if not Config.reactive_mode then [] else ["--reactive"]) @ "--out" :: Config.results_dir :: (match Config.project_root with None -> [] | Some pr -> ["--project_root"; pr]) @ (match Config.xcode_developer_dir with None -> [] | Some d -> ["--xcode-developer-dir"; d]) @ (if Config.rest = [] then [] else ("--" :: build_cmd)) ) in let pid = Unix.create_process args_py.(0) args_py Unix.stdin Unix.stdout Unix.stderr in let _, status = Unix.waitpid [] pid in let exit_code = match status with | Unix.WEXITED i -> i | _ -> 1 in if exit_code = Config.infer_py_argparse_error_exit_code then (* swallow infer.py argument parsing error *) Config.print_usage_exit (); if exit_code <> 0 then ( prerr_endline ("Failed to execute: " ^ (String.concat " " (Array.to_list args_py))) ; exit exit_code ); if Config.is_originator then ( if Config.analyzer = Some Config.Crashcontext then Crashcontext.crashcontext_epilogue ~in_buck_mode; if Config.fail_on_bug then fail_on_issue_epilogue (); )