105 lines
3.2 KiB
OCaml
105 lines
3.2 KiB
OCaml
(***********************************************************************)
|
|
(* *)
|
|
(* 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
|