File ‹Tools/cakeml_compiler.ML›

signature CAKEML_COMPILER = sig
  datatype mode =
    Literal (* concrete syntax *) |
    Prog (* sexp syntax *)

  val compiler: unit -> Path.T
  val compile_ml: mode -> string -> Path.T
  val compile_ml_file: mode -> Path.T -> Path.T
  val compile_c_file: Path.T -> Path.T
  val compile_ffi: unit -> Path.T
  val link: Path.T list -> Path.T

  val eval: mode -> string -> string
  val eval_source: mode -> Input.source -> string

  val string_of_prog: Proof.context -> term -> string

  val cakeml_cmd: Proof.context -> mode -> Input.source -> unit
end

structure CakeML_Compiler : CAKEML_COMPILER = struct

datatype mode = Literal | Prog

fun compiler () =
  let
    val platform = getenv_strict "ISABELLE_PLATFORM64"
    val paths = [getenv_strict "ISABELLE_CAKEML_HOME", "bin", platform, "cake"]
    val file = Path.appends (map Path.explode paths)
  in
    if File.exists file then
      file
    else
      error "CakeML: unsupported platform"
  end

fun basis_ffi () =
  Path.append (Path.explode (getenv_strict "ISABELLE_CAKEML_HOME")) (Path.basic "basis_ffi.c")

fun compile_ml_file mode source =
  let
    val id = serial_string ()
    val output = File.tmp_path (Path.basic ("cakeml_out" ^ id ^ ".S"))

    val sexp = if mode = Literal then "false" else "true"

    val bash_cake = File.bash_path (compiler ())
    val bash_source = File.bash_path source
    val res =
      Isabelle_System.bash_process
        (Bash.script (bash_cake ^ " --sexp=" ^ sexp ^ " < " ^ bash_source))
    val err = Process_Result.err res
    val out = Process_Result.out res
    val _ = if err <> "" then warning err else ()
  in
    if not (Process_Result.ok res) orelse err <> "" then
      error "CakeML: ML compilation failed"
    else
      (File.write output out; output)
  end

fun compile_ml mode source =
  let
    val id = serial_string ()
    val output = File.tmp_path (Path.basic ("cakeml_in" ^ id))
    val _ = File.write output source
  in compile_ml_file mode output end

fun compile_c_file source =
  let
    val id = serial_string ()
    val output = File.tmp_path (Path.basic ("c_out" ^ id ^ ".o"))

    val bash_cc = File.bash_path (Path.explode (getenv_strict "ISABELLE_CC"))
    val bash_source = File.bash_path source
    val bash_output = File.bash_path output
    val res =
      Isabelle_System.bash_process
        (Bash.script (bash_cc ^ " -no-pie -c -o " ^ bash_output ^ " " ^ bash_source))
    val err = Process_Result.err res
    val out = Process_Result.out res
    val _ = if err <> "" then warning err else ()
    val _ = writeln out
  in
    if not (Process_Result.ok res) orelse err <> "" then
      error "CakeML: C compilation failed"
    else
      output
  end

val compile_ffi =
  compile_c_file o basis_ffi

fun link sources =
  let
    val id = serial_string ()
    val output = File.tmp_path (Path.basic ("bin" ^ id ^ ".out"))

    val bash_cc = File.bash_path (Path.explode (getenv_strict "ISABELLE_CC"))
    val bash_sources = Bash.strings (map File.standard_path sources)
    val bash_output = File.bash_path output
    val res =
      Isabelle_System.bash_process
        (Bash.script (bash_cc ^ " -no-pie -o " ^ bash_output ^ " " ^ bash_sources))
    val err = Process_Result.err res
    val out = Process_Result.out res
    val _ = if err <> "" then warning err else ()
    val _ = writeln out
  in
    if not (Process_Result.ok res) orelse err <> "" then
      error "CakeML: linking failed"
    else
      output
  end

fun eval mode source =
  let
    val cake = compile_ml mode source
    val ffi = compile_ffi ()
    val bin = link [cake, ffi]
    val res = Isabelle_System.bash_process (Bash.script (File.bash_path bin))
    val err = Process_Result.err res
    val out = Process_Result.out res
    val _ = if err <> "" then warning err else ()
  in
    if not (Process_Result.ok res) orelse err <> "" then
      error "CakeML: evaluation failed"
    else
      out
  end

fun eval_source mode =
  eval mode o #1 o Input.source_content

fun string_of_prog ctxt t =
  let
    val (_, raw_prog) =
      Thm.cterm_of ctxt t
      |> Code_Simp.dynamic_conv ctxt
      |> Thm.prop_of
      |> Logic.dest_equals
  in CakeML_Sexp.print_prog raw_prog end

val parse_mode =
  Args.parens (Parse.reserved "literal" >> K Literal ||
               Parse.reserved "prog" >> K Prog)

fun cakeml_cmd ctxt mode source =
  let
    val source =
      case mode of
        Literal => #1 (Input.source_content source)
      | Prog =>
          Syntax.implode_input source
          |> Syntax.parse_term ctxt
          |> Type.constraint @{typ Ast.prog}
          |> Syntax.check_term ctxt
          |> string_of_prog ctxt
    val _ = tracing ("Evaluating: " ^ source)
    val output = eval mode source
  in writeln output end

val _ =
  Outer_Syntax.command command_keywordcakeml "evalute CakeML source"
    (parse_mode -- Parse.embedded_input >> (fn (mode, src) =>
      Toplevel.keep (fn state => cakeml_cmd (Toplevel.context_of state) mode src)))

end