(***********************************************************************) (* *) (* Caml Special Light *) (* *) (* Xavier Leroy, projet Cristal, INRIA Rocquencourt *) (* *) (* Copyright 1995 Institut National de Recherche en Informatique et *) (* Automatique. Distributed only by permission. *) (* *) (***********************************************************************) (* $Id$ *) (* From lambda to assembly code *) open Format open Clflags open Misc open Cmm type error = Assembler_error of string exception Error of error let liveness phrase = Liveness.fundecl phrase; phrase let dump_if flag message phrase = if !flag then Printmach.phase message phrase; phrase let dump_linear_if flag message phrase = if !flag then begin print_string "*** "; print_string message; print_newline(); Printlinear.fundecl phrase; print_newline() end; phrase let rec regalloc fd = dump_if dump_live "Liveness analysis" fd; Interf.build_graph fd; if !dump_interf then Printmach.interferences(); if !dump_prefer then Printmach.preferences(); Coloring.allocate_registers(); dump_if dump_regalloc "After register allocation" fd; let (newfd, redo_regalloc) = Reload.fundecl fd in dump_if dump_reload "After insertion of reloading code" newfd; if redo_regalloc then begin Reg.reinit(); Liveness.fundecl newfd; regalloc newfd end else newfd let (++) x f = f x let compile_fundecl fd_cmm = Reg.reset(); fd_cmm ++ Selection.fundecl ++ dump_if dump_selection "After instruction selection" ++ liveness ++ dump_if dump_live "Liveness analysis" ++ Spill.fundecl ++ liveness ++ dump_if dump_spill "After spilling" ++ Split.fundecl ++ dump_if dump_split "After live range splitting" ++ liveness ++ regalloc ++ Linearize.fundecl ++ dump_linear_if dump_linear "Linearized code" ++ Scheduling.fundecl ++ dump_linear_if dump_scheduling "After instruction scheduling" ++ Emit.fundecl let compile_phrase p = if !dump_cmm then begin Printcmm.phrase p; print_newline() end; match p with Cfunction fd -> compile_fundecl fd | Cdata dl -> Emit.data dl let compile_implementation prefixname lam = let asmfile = if !keep_asm_file then prefixname ^ ".s" else temp_file "camlasm" ".s" in let oc = open_out asmfile in begin try Emitaux.output_channel := oc; Emit.begin_assembly(); List.iter compile_phrase (Cmmgen.compunit (Closure.intro lam)); Emit.end_assembly(); close_out oc with x -> close_out oc; if !keep_asm_file then () else remove_file asmfile; raise x end; if Proc.assemble_file asmfile (prefixname ^ ".o") <> 0 then raise(Error(Assembler_error asmfile)); if !keep_asm_file then () else remove_file asmfile (* Error report *) let report_error = function Assembler_error file -> print_string "Assembler error, input left in file "; print_string file