-
Wilke Pierre authoredWilke Pierre authored
riscv.ml 6.43 KiB
open Batteries
open BatList
open Elang
open Rtl
open Ltl
open Ltl_print
open Utils
open Prog
open Options
open Archi
(* This file performs the translation from LTL programs to RISC-V assembly
programs. The languages are basically the same, so the only thing to do here
is to print LTL instructions in RISC-V assembly syntax.
Another thing to do is to deal with command-line arguments. These are given
as strings and need to be converted into strings. That is the purpose of the
[riscv_load_args] function in this file.
Finally, [riscv_prelude] gives an entry point in the program : a global
"_start" symbol is created, which points to startup code, whose job is the
following:
- initializes the global pointer (for "heap" allocation),
- sets up the stack,
- retrieves arguments,
- calls the "main" function of our program,
- prints the return value to the screen,
- and finally exits the program by issuing the appropriate system call.
*)
let riscv_of_cmp (cmp: rtl_cmp) =
match cmp with
| Rtl.Rclt -> "blt"
| Rtl.Rcle -> "ble"
| Rtl.Rcgt -> "bgt"
| Rtl.Rcge -> "bge"
| Rtl.Rceq -> "beq"
| Rtl.Rcne -> "bne"
let print_binop (b: binop) =
match b with
| Elang.Eadd -> "add"
| Elang.Emul -> "mul"
| Elang.Emod -> "remu"
| Elang.Exor -> "xor"
| Elang.Ediv -> "divu"
| Elang.Esub -> "sub"
| _ -> failwith "Unexpected binop"
let print_unop (u: unop) =
match u with
| Elang.Eneg -> "neg"
let instrsuffix_of_size sz =
match sz with
| MAS1 -> 'b'
| MAS4 -> 'w'
| MAS8 -> 'd'
let dump_riscv_instr oc (i: ltl_instr) =
match i with
| LAddi(rd, rs, i) ->
Format.fprintf oc "addi %s, %s, %d\n" (print_reg rd) (print_reg rs) i
| LSubi(rd, rs, i) ->
Format.fprintf oc "addi %s, %s, %d\n" (print_reg rd) (print_reg rs) (-i)
| LBinop(b, rd, rs1, rs2) ->
begin match b with
| Elang.Eclt ->
Format.fprintf oc "slt %s, %s, %s\n"
(print_reg rd) (print_reg rs1) (print_reg rs2)
| Elang.Ecgt ->
Format.fprintf oc "slt %s, %s, %s\n"
(print_reg rd) (print_reg rs2) (print_reg rs1)
| Elang.Ecle ->
(* 'rd <- rs1 <= rs2' == 'rd <- rs2 < rs1; rd <- seqz rd' *)
Format.fprintf oc "slt %s, %s, %s\n"
(print_reg rd) (print_reg rs2) (print_reg rs1);
Format.fprintf oc "seqz %s, %s\n"
(print_reg rd) (print_reg rd)
| Elang.Ecge ->
Format.fprintf oc "slt %s, %s, %s\n"
(print_reg rd) (print_reg rs1) (print_reg rs2);
Format.fprintf oc "seqz %s, %s\n"
(print_reg rd) (print_reg rd)
| Elang.Eceq ->
Format.fprintf oc "sub %s, %s, %s\n"
(print_reg rd) (print_reg rs1) (print_reg rs2);
Format.fprintf oc "seqz %s, %s\n"
(print_reg rd) (print_reg rd)
| Elang.Ecne ->
Format.fprintf oc "sub %s, %s, %s\n"
(print_reg rd) (print_reg rs1) (print_reg rs2);
Format.fprintf oc "snez %s, %s\n"
(print_reg rd) (print_reg rd)
| _ -> Format.fprintf oc "%s %s, %s, %s\n"
(print_binop b) (print_reg rd) (print_reg rs1) (print_reg rs2)
end
| LUnop(u, rd, rs) ->
Format.fprintf oc "%s %s, %s\n"
(print_unop u) (print_reg rd) (print_reg rs)
| LStore(rt, i, rs, sz) ->
let sz = instrsuffix_of_size sz in
Format.fprintf oc "s%c %s, %d(%s)\n"
sz (print_reg rs) i (print_reg rt)
| LLoad(rd, rt, i, sz) ->
let sz = (instrsuffix_of_size sz) in
Format.fprintf oc "l%c %s, %d(%s)\n"
sz (print_reg rd) i (print_reg rt)
| LMov(rd, rs) ->
Format.fprintf oc "mv %s, %s\n" (print_reg rd) (print_reg rs)
| LLabel l ->
Format.fprintf oc "%s:\n" l
| LJmp l -> Format.fprintf oc "j %s\n" l
| LJmpr r -> Format.fprintf oc "jr %s\n" (print_reg r)
| LConst (rd, i) -> Format.fprintf oc "li %s, %d\n\n" (print_reg rd) i
| LComment l -> Format.fprintf oc "# %s\n" l
| LBranch(cmp, rs1, rs2, s) ->
Format.fprintf oc "%s %s, %s, %s\n"
(riscv_of_cmp cmp) (print_reg rs1) (print_reg rs2) s
| LCall fname ->
Format.fprintf oc "jal ra, %s\n" fname
| LHalt -> Format.fprintf oc "halt\n"
let dump_riscv_fun oc (fname , lf) =
Format.fprintf oc "%s:\n" fname;
List.iter (dump_riscv_instr oc) lf.ltlfunbody
let riscv_load_args target oc : unit =
(match target with
| Linux -> LLoad(reg_s1, reg_sp, 0, archi_mas ()) :: (* s1 <- argc *)
LAddi(reg_s2, reg_sp, (Archi.wordsize ())) :: []
| Xv6 -> LMov(reg_s1, reg_a0) ::
LMov(reg_s2, reg_a1) :: []) @
LConst(reg_s3, 1) ::
LSubi(reg_sp, reg_sp, 72) ::
LLabel "Lloop" ::
LBranch(Rceq, reg_s3, reg_s1, "Lendargs") ::
LMov(reg_a0, reg_t4) ::
LAddi(reg_s4, reg_s3, 0) ::
LConst(reg_t1, (Archi.wordsize ())) ::
LBinop(Emul, reg_s4, reg_s4, reg_t1) ::
LBinop(Eadd, reg_t3, reg_s4, reg_s2) ::
LLoad(reg_a0, reg_t3, 0, archi_mas ()) ::
LCall "atoi" ::
LBinop(Esub, reg_s4, reg_fp, reg_s4) ::
LStore(reg_s4, 0, reg_a0, archi_mas ()) ::
LAddi(reg_s3, reg_s3, 1) ::
LJmp "Lloop" ::
LLabel "Lendargs" ::
LLoad(reg_a0, reg_fp, -8, archi_mas ()) ::
LLoad(reg_a1, reg_fp, -16, archi_mas ()) ::
LLoad(reg_a2, reg_fp, -24, archi_mas ()) ::
LLoad(reg_a3, reg_fp, -32, archi_mas ()) ::
LLoad(reg_a4, reg_fp, -40, archi_mas ()) ::
LLoad(reg_a5, reg_fp, -48, archi_mas ()) ::
LLoad(reg_a6, reg_fp, -56, archi_mas ()) ::
LLoad(reg_a7, reg_fp, -64, archi_mas ()) ::
[] |>
List.iter (dump_riscv_instr oc)
let rv_store () =
Format.sprintf "s%c" (Archi.instrsuffix ())
let rv_load () =
Format.sprintf "l%c" (Archi.instrsuffix ())
let riscv_prelude target oc =
Format.fprintf oc ".include \"syscall_numbers.s\"\n";
Format.fprintf oc ".globl _start\n";
Format.fprintf oc "_start:\n";
Format.fprintf oc " lui gp, %%hi(_heap_start)\n";
Format.fprintf oc " addi gp, gp, %%lo(_heap_start)\n";
Format.fprintf oc " addi t0, gp, 8\n";
Format.fprintf oc " %s t0, 0(gp)\n" (rv_store ());
Format.fprintf oc " mv s0, sp\n";
riscv_load_args target oc ;
Format.fprintf oc "jal ra, main\n";
Format.fprintf oc "mv s0, a0\n";
Format.fprintf oc "jal ra, println\n";
Format.fprintf oc "mv a0, s0\n";
Format.fprintf oc "jal ra, print_int\n";
Format.fprintf oc "jal ra, println\n";
Format.fprintf oc "addi a7, zero, SYSCALL_EXIT\n";
Format.fprintf oc "ecall\n"
let dump_riscv_prog target oc lp : unit =
(if !nostart then () else riscv_prelude target oc);
Format.fprintf oc ".global main\n";
List.iter (function
(fname, Gfun f) -> dump_riscv_fun oc (fname,f)
) lp