package b0
 sectionYPositions = computeSectionYPositions($el), 10)"
  x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
  >
  
  
  Software construction and deployment kit
Install
    
    dune-project
 Dependency
Authors
Maintainers
Sources
  
    
      b0-0.0.6.tbz
    
    
        
    
  
  
  
    
  
        sha512=e9aa779e66c08fc763019f16d4706f465d16c05d6400b58fbd0313317ef33ddea51952e2b058db28e65f7ddb7012f328c8bf02d8f1da17bb543348541a2587f0
    
    
  doc/src/b0.file/b0_driver.ml.html
Source file b0_driver.ml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433(*--------------------------------------------------------------------------- Copyright (c) 2020 The b0 programmers. All rights reserved. SPDX-License-Identifier: ISC ---------------------------------------------------------------------------*) open B0_std open Result.Syntax module Exit = struct open Cmdliner let deploy_error = Os.Exit.Code 118 let build_error = Os.Exit.Code 119 let b0_file_error = Os.Exit.Code 120 let no_b0_file = Os.Exit.Code 121 let e c doc = Cmd.Exit.info (Os.Exit.get_code c) ~doc let info_deploy_error = e deploy_error "on deploy error." let info_build_error = e build_error "on build error." let info_b0_file_error = e b0_file_error "on b0 file error." let info_no_b0_file = e no_b0_file "no b0 file found." let infos = info_deploy_error :: info_build_error :: info_b0_file_error :: info_no_b0_file :: B0_std_cli.Exit.infos end module Env = struct (* FIXME is this module useful ? *) let b0_dir = Cmdliner.Cmd.Env.info_var B0_cli.b0_dir_var let b0_file = "B0_FILE" let cache_dir = Cmdliner.Cmd.Env.info_var B0_memo_cli.File_cache.dir_var let code = "B0_DRIVER_CODE" let hash_fun = Cmdliner.Cmd.Env.info_var B0_memo_cli.Hash.hash_fun_var let jobs = Cmdliner.Cmd.Env.info_var B0_memo_cli.jobs_var end module Conf = struct let b0_file_name = "B0.ml" let drivers_dir_name = ".drivers" type t = { b0_dir : Fpath.t; b0_file : Fpath.t option; cache_dir : Fpath.t; cwd : Fpath.t; env : Os.Env.t; code : B0_ocaml.Code.t option; hash_fun : (module B0_hash.T); jobs : int; no_pager : bool; memo : (B0_memo.t, string) result Lazy.t } let memo ~hash_fun ~cwd ~env ~cache_dir ~trash_dir ~jobs = let feedback = let op_howto ppf o = Fmt.pf ppf "b0 file log --id %d" (B0_zero.Op.id o) in let output_op_level = Log.Info and output_ui_level = Log.Error in let level = Log.level () in B0_memo_cli.pp_leveled_feedback ~op_howto ~output_op_level ~output_ui_level ~level Fmt.stderr in B0_memo.make ~hash_fun ~cwd ~env ~cache_dir ~trash_dir ~jobs ~feedback () let make ~b0_dir ~b0_file ~cache_dir ~cwd ~code ~env ~hash_fun ~jobs ~no_pager () = let trash_dir = Fpath.(b0_dir / B0_memo_cli.trash_dirname) in let memo = lazy (memo ~hash_fun ~cwd ~env ~cache_dir ~trash_dir ~jobs) in { b0_dir; b0_file; cache_dir; cwd; code; env; hash_fun; jobs; memo; no_pager; } let b0_dir c = c.b0_dir let b0_file c = c.b0_file let cache_dir c = c.cache_dir let cwd c = c.cwd let code c = c.code let env c = c.env let hash_fun c = c.hash_fun let jobs c = c.jobs let memo c = Lazy.force c.memo let no_pager c = c.no_pager let get_b0_file c = match c.b0_file with | Some file -> Ok file | None -> let path = Fmt.(code' Fpath.pp_unquoted) in Fmt.error "@[<v>No %a file found in %a@,\ or upwards. Use option %a to specify one or %a for help.@]" Fmt.code "B0.ml" path c.cwd Fmt.code "--b0-file" Fmt.code "--help" (* Setup *) let find_b0_file ~cwd ~b0_file = match b0_file with | Some b0_file -> Some (Fpath.(cwd // b0_file)) | None -> let rec loop dir = let file = Fpath.(dir / b0_file_name) in match Os.File.exists file with | Ok true -> Some file | _ -> if not (Fpath.is_root dir) then loop (Fpath.parent dir) else None in loop cwd let set_no_color_and_log_level ~no_color ~log_level = if no_color then Fmt.set_styler Plain; Log.set_level log_level let setup_with_cli ~b0_dir ~b0_file ~cache_dir ~code ~hash_fun ~jobs ~no_color ~log_level ~no_pager () = set_no_color_and_log_level ~no_color ~log_level; let* cwd = Os.Dir.cwd () in let* env = Os.Env.current () in let b0_file = find_b0_file ~cwd ~b0_file in let root = match b0_file with Some f -> Fpath.parent f | None -> cwd in let b0_dir = B0_cli.get_b0_dir ~cwd ~root ~b0_dir in let cache_dir = B0_cli.get_cache_dir ~cwd ~b0_dir ~cache_dir in let hash_fun = B0_memo_cli.Hash.get_hash_fun ~hash_fun in let jobs = B0_memo_cli.get_jobs ~jobs in Ok (make ~b0_dir ~b0_file ~cache_dir ~cwd ~code ~env ~hash_fun ~jobs ~no_pager ()) end module Cli = struct open Cmdliner open Cmdliner.Term.Syntax let docs = Manpage.s_common_options let b0_dir = B0_cli.b0_dir () let b0_file = let env = Cmd.Env.info Env.b0_file in let doc = "Use $(docv) as the b0 file." and docv = "PATH" in let absent = "$(b,B0.ml) file in cwd or first upwards" in Arg.(value & opt (Arg.some B0_std_cli.filepath) None & info ["b0-file"] ~absent ~doc ~docv ~docs ~env) let cache_dir = B0_memo_cli.File_cache.dir () let code = let env = Cmd.Env.info Env.code in let code_enum = ["byte", Some B0_ocaml.Code.Byte; "native", Some Native; "auto", None]in let code = Arg.enum code_enum in let docv = "CODE" in let doc = "Compile driver to $(docv). $(docv) must be $(b,byte), $(b,native) or \ $(b,auto). If $(b,auto) favors native code if available." in Arg.(value & opt code None & info ["driver-code"] ~doc ~docv ~docs ~env) let no_color = B0_std_cli.no_color () let log_level = B0_std_cli.log_level () let conf = Term.term_result' @@ let+ b0_dir and+ b0_file and+ cache_dir and+ code and+ hash_fun = B0_memo_cli.Hash.hash_fun () and+ jobs = B0_memo_cli.jobs () and+ no_color and+ log_level and+ no_pager = B0_cli.no_pager in Conf.setup_with_cli ~b0_dir ~b0_file ~cache_dir ~code ~hash_fun ~jobs ~no_color ~log_level ~no_pager () let set_no_color_and_log_level = let+ no_color and+ log_level in Conf.set_no_color_and_log_level ~no_color ~log_level end (* Drivers *) type main = unit -> (Os.Exit.t Cmdliner.Cmd.eval_ok, Cmdliner.Cmd.eval_error) result type t = { name : string; version : string; libs : B0_ocaml.Libname.t list } let make ~name ~version ~libs = { name; version; libs } let name d = d.name let version d = d.version let libs d = d.libs let has_b0_file = ref false (* set by run *) let driver = ref None let set ~driver:d ~main = driver := Some (d, main) let timing b0_file d _ m = let b0_file = if b0_file then "with B0.ml" else "no B0.ml" in m "total time %s %s %s" d.name d.version b0_file let run ~has_b0_file:b0_file = Os.Exit.exit @@ match !driver with | None -> invalid_arg "No driver set via B0_driver.set" | Some (d, main) -> has_b0_file := b0_file; Log.time begin fun _ m -> let b0_file = if b0_file then "with B0.ml" else "no B0.ml" in m "total time %s %s %s" d.name d.version b0_file end @@ fun () -> B0_std_cli.Exit.of_eval_result @@ try main () with B0_scope.After_seal e -> (* FIXME I suspect we may never see this it will be catched by memo protection. At least make a good error msg. *) let bt = Printexc.get_raw_backtrace () in Log.err (fun m -> m ~header:"" "@[<v>%a@,@[%s@]@]" Fmt.backtrace bt e); Ok (`Ok Exit.b0_file_error) let has_b0_file () = !has_b0_file module Compile = struct open B0_std.Fut.Syntax let build_dir c ~driver = Fpath.(Conf.b0_dir c / Conf.drivers_dir_name / name driver) let build_log c ~driver = Fpath.(Conf.b0_dir c / Conf.drivers_dir_name / name driver / "log") let exe c ~driver = Fpath.(Conf.b0_dir c / Conf.drivers_dir_name / name driver / "exe") let write_src m c expanded_src ~file_api_stamp ~src_file = let src_reads = B0_file.expanded_file_manifest expanded_src in let reads = List.rev_append file_api_stamp src_reads in B0_memo.ready_files m src_reads; B0_memo.write m ~reads src_file @@ fun () -> Ok (B0_file.expanded_src expanded_src) let base_ext_libs = [ B0_ocaml.Libname.v "cmdliner"; B0_ocaml.Libname.v "unix"; ] let b0_file_lib = B0_ocaml.Libname.v "b0.file" let base_libs = [ B0_ocaml.Libname.v "b0.std"; B0_ocaml.Libname.v "b0.memo"; b0_file_lib; B0_ocaml.Libname.v "b0.kit"; ] let find_libs m r libs = Fut.of_list @@ List.map (B0_ocaml.Libresolver.get m r) libs let find_boot_libs m ~clib_ext ~env libs r = match Os.Env.var ~empty_is_none:true "B0_BOOTSTRAP" with | None -> find_libs m r libs | Some bdir -> let bdir = Fpath.v bdir in let boot_lib libname = let dir = Fpath.(bdir / B0_ocaml.Libname.undot ~rep:'-' libname) in let archive = Some (B0_ocaml.Libname.to_archive_name libname) in let lib = B0_ocaml.Lib.of_dir m ~clib_ext ~libname ~requires:[] ~exports:[] ~archive ~dir ~js_stubs:[] ~warning:None in match lib with | Error _ as e -> B0_memo.fail_if_error m e | Ok lib -> Fut.return lib in Fut.of_list (List.map boot_lib libs) let find_libs m ocaml_conf ~build_dir ~driver ~requires = let cache_dir = Fpath.(build_dir / B0_ocaml.Libresolver.Scope.cache_dir_name) in (* let ocamlpath = B0_ocaml.Libresolver.Scope.ocamlpath ~cache_dir in *) let ocamlfind = B0_ocaml.Libresolver.Scope.ocamlfind ~cache_dir in let scopes = [(*ocamlpath;*) ocamlfind] in let r = B0_ocaml.Libresolver.make m ocaml_conf scopes in let requires = List.map fst requires in let* requires = find_libs m r requires in (* FIXME we are loosing locations here would be nice to have them to report errors. *) (* FIXME we likely also want a notion of ext lib for drivers *) let clib_ext = B0_ocaml.Conf.lib_ext ocaml_conf in let* driver_libs = find_boot_libs m ~clib_ext ~env:"B0_DRIVER_BOOTSTRAP" (libs driver) r in let* base_ext_libs = find_libs m r base_ext_libs in let* base_libs = find_boot_libs m ~clib_ext ~env:"B0_BOOTSTRAP" base_libs r in let b0_file_lib = let is_b0_file_lib l = B0_ocaml.Libname.equal (B0_ocaml.Lib.libname l) b0_file_lib in List.find_opt is_b0_file_lib base_libs |> Option.get in let all_libs = base_ext_libs @ base_libs @ driver_libs @ requires in let seen_libs = base_ext_libs @ base_libs @ requires in Fut.return (b0_file_lib, all_libs, seen_libs) let find_compiler c m = match Conf.code c with | Some (Byte as c) -> Fut.return (B0_ocaml.Tool.ocamlc, c) | Some (Native as c) -> Fut.return (B0_ocaml.Tool.ocamlopt, c) | Some Wasm -> assert false | None -> let* ocamlopt = B0_memo.tool_opt m B0_ocaml.Tool.ocamlopt in match ocamlopt with | None -> Fut.return (B0_ocaml.Tool.ocamlc, B0_ocaml.Code.Byte) | Some comp -> Fut.return (B0_ocaml.Tool.ocamlopt, B0_ocaml.Code.Native) let compile_src m c ~driver ~build_dir src ~exe = let ocaml_conf = Fpath.(build_dir / "ocaml.conf") in let* comp, code = find_compiler c m in B0_ocaml.Conf.write m ~comp ~o:ocaml_conf; let* ocaml_conf = B0_ocaml.Conf.read m ocaml_conf in let obj_ext = B0_ocaml.Conf.obj_ext ocaml_conf in let comp = B0_memo.tool m comp in let expanded_src = B0_memo.fail_if_error m (B0_file.expand src) in let requires = B0_file.expanded_requires expanded_src in let* b0_file_lib, all_libs, seen_libs = find_libs m ocaml_conf ~build_dir ~driver ~requires in let src_file = Fpath.(build_dir / "src.ml") in let file_api_stamp = match code with (* archive changes when API does *) | B0_ocaml.Code.Byte -> Option.to_list (B0_ocaml.Lib.cma b0_file_lib) | B0_ocaml.Code.Native -> Option.to_list (B0_ocaml.Lib.cmxa b0_file_lib) | B0_ocaml.Code.Wasm -> assert false in write_src m c expanded_src ~file_api_stamp ~src_file; let writes = let base = Fpath.strip_ext ~multi:false src_file in let base ext = Fpath.(base + ext) in match code with | B0_ocaml.Code.Byte -> [base ".cmo"; exe ] | B0_ocaml.Code.Native -> [base ".cmx"; base obj_ext; exe ] | B0_ocaml.Code.Wasm -> assert false in let dirs = List.map B0_ocaml.Lib.dir seen_libs in let incs = Cmd.unstamp @@ Cmd.paths ~slip:"-I" dirs in let archives = let ar = match code with | B0_ocaml.Code.Native -> B0_ocaml.Lib.cmxa | B0_ocaml.Code.Byte -> B0_ocaml.Lib.cma | B0_ocaml.Code.Wasm -> assert false in List.filter_map ar all_libs in let c_archives = List.filter_map B0_ocaml.Lib.c_archive all_libs in let ars = List.rev_append archives c_archives in (* FIXME this should be done b the resolver *) B0_memo.ready_files m ars; let reads = src_file :: ars in B0_memo.spawn m ~reads ~writes @@ comp Cmd.(arg "-linkall" % "-g" % "-o" %% unstamp (path exe) % "-opaque" %% incs %% (unstamp @@ (paths archives %% path src_file))); Fut.return () let write_log_file ~log_file m = Log.if_error ~use:() @@ B0_memo_log.(write log_file (of_memo m)) let compile c ~driver ~feedback src = Result.bind (Conf.memo c) @@ fun m -> let m = if feedback then m else B0_memo.with_feedback m ignore in let build_dir = build_dir c ~driver in let log_file = build_log c ~driver in let exe = exe c ~driver in (* That shit should be streamlined: brzo, odig, b0caml all have similar setup/log/reporting bits. *) Os.Exit.on_sigint ~hook:(fun () -> write_log_file ~log_file m) @@ fun () -> B0_memo.run_proc m begin fun () -> let* () = B0_memo.delete m build_dir in let* () = B0_memo.mkdir m build_dir in compile_src m c ~driver ~build_dir src ~exe end; B0_memo.stir ~block:true m; write_log_file ~log_file m; match B0_memo.status m with | Ok () -> Ok exe | Error e -> let name = name driver in let dopt = if name = "b0" then "" else Fmt.str " --driver %s" name in let read_howto ppf _ = Fmt.pf ppf "b0 file log%s -r " dopt in let write_howto ppf _ = Fmt.pf ppf "b0 file log%s -w " dopt in if feedback then B0_zero_conv.Op.pp_aggregate_error ~read_howto ~write_howto () Fmt.stderr e; Fmt.error "Could not compile b0 file %a" Fmt.(code' Fpath.pp) (B0_file.file src) end let compile_b0_file conf ~driver ~feedback b0_file = let* src = Os.File.read b0_file in let* src = B0_file.of_string ~file:b0_file src in Compile.compile conf ~driver ~feedback src let with_b0_file ~driver cmd = let run conf cmd = if has_b0_file () then cmd conf else Log.if_error ~use:Exit.no_b0_file @@ let* b0_file = Conf.get_b0_file conf in Log.if_error' ~use:Exit.b0_file_error @@ let* exe = compile_b0_file conf ~driver ~feedback:true b0_file in let exe = Fpath.to_string exe in let cmd = match Array.to_list Sys.argv with | [] -> Cmd.arg exe | _ :: args -> Cmd.list (exe :: args) in Ok (Os.Exit.execv ~argv0:"b0" cmd) in Cmdliner.Term.(const run $ Cli.conf $ cmd) let has_failed_b0_file = ref false let with_b0_file_if_any ~driver cmd = let run conf cmd = if has_b0_file () then cmd conf else match Conf.b0_file conf with | None -> cmd conf | Some b0_file -> match compile_b0_file conf ~driver ~feedback:false b0_file with | Error e -> (Log.warn @@ fun m -> m "%s. See %a." e Fmt.code "b0 file log -e"); has_failed_b0_file := true; cmd conf | Ok exe -> let exe = Fpath.to_string exe in let cmd = match Array.to_list Sys.argv with | [] -> Cmd.arg exe | _ :: args -> Cmd.list (exe :: args) in Os.Exit.execv ~argv0:"b0" cmd in Cmdliner.Term.(const run $ Cli.conf $ cmd) let has_failed_b0_file () = !has_failed_b0_file
 sectionYPositions = computeSectionYPositions($el), 10)"
  x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
  >