ocaml/asmcomp/arm/selection.ml

332 lines
13 KiB
OCaml

(**************************************************************************)
(* *)
(* OCaml *)
(* *)
(* Benedikt Meurer, University of Siegen *)
(* *)
(* Copyright 1998 Institut National de Recherche en Informatique et *)
(* en Automatique. *)
(* Copyright 2012 Benedikt Meurer. *)
(* *)
(* All rights reserved. This file is distributed under the terms of *)
(* the GNU Lesser General Public License version 2.1, with the *)
(* special exception on linking described in the file LICENSE. *)
(* *)
(**************************************************************************)
(* Instruction selection for the ARM processor *)
open Arch
open Proc
open Cmm
open Mach
let is_offset chunk n =
match chunk with
(* VFPv{2,3} load/store have -1020 to 1020. Offset must be multiple of 4 *)
| Single | Double | Double_u
when !fpu >= VFPv2 ->
n >= -1020 && n <= 1020 && n mod 4 = 0
(* ARM load/store byte/word have -4095 to 4095 *)
| Byte_unsigned | Byte_signed
| Thirtytwo_unsigned | Thirtytwo_signed
| Word_int | Word_val | Single
when not !thumb ->
n >= -4095 && n <= 4095
(* Thumb-2 load/store have -255 to 4095 *)
| _ when !arch > ARMv6 && !thumb ->
n >= -255 && n <= 4095
(* Everything else has -255 to 255 *)
| _ ->
n >= -255 && n <= 255
let select_shiftop = function
Clsl -> Ishiftlogicalleft
| Clsr -> Ishiftlogicalright
| Casr -> Ishiftarithmeticright
| __-> assert false
(* Special constraints on operand and result registers *)
exception Use_default
let r1 = phys_reg 1
let r6 = phys_reg 6
let r7 = phys_reg 7
let pseudoregs_for_operation op arg res =
match op with
(* For mul rd,rm,rs and mla rd,rm,rs,ra (pre-ARMv6) the registers rm
and rd must be different. We deal with this by pretending that rm
is also a result of the mul / mla operation. *)
Iintop Imul | Ispecific Imuladd when !arch < ARMv6 ->
(arg, [| res.(0); arg.(0) |])
(* For smull rdlo,rdhi,rn,rm (pre-ARMv6) the registers rdlo, rdhi and rn
must be different. Also, rdlo (whose contents we discard) is always
forced to be r12 in proc.ml, which means that neither rdhi and rn can
be r12. To keep things simple, we force both of those two to specific
hard regs: rdhi in r6 and rn in r7. *)
| Iintop Imulh when !arch < ARMv6 ->
([| r7; arg.(1) |], [| r6 |])
(* Soft-float Iabsf and Inegf: arg.(0) and res.(0) must be the same *)
| Iabsf | Inegf when !fpu = Soft ->
([|res.(0); arg.(1)|], res)
(* VFPv{2,3} Imuladdf...Inegmulsubf: arg.(0) and res.(0) must be the same *)
| Ispecific(Imuladdf | Inegmuladdf | Imulsubf | Inegmulsubf) ->
let arg' = Array.copy arg in
arg'.(0) <- res.(0);
(arg', res)
(* We use __aeabi_idivmod for Cmodi only, and hence we care only
for the remainder in r1, so fix up the destination register. *)
| Iextcall { func = "__aeabi_idivmod"; _ } ->
(arg, [|r1|])
(* Other instructions are regular *)
| _ -> raise Use_default
(* Instruction selection *)
class selector = object(self)
inherit Selectgen.selector_generic as super
method! regs_for tyv =
Reg.createv (if !fpu = Soft then begin
(* Expand floats into pairs of integer registers *)
(* CR mshinwell: we need to check this in conjunction with
the unboxed external functionality *)
let rec expand = function
[] -> []
| Float :: tyl -> Int :: Int :: expand tyl
| ty :: tyl -> ty :: expand tyl in
Array.of_list (expand (Array.to_list tyv))
end else begin
tyv
end)
method! is_immediate op n =
match op with
| Iadd | Isub | Iand | Ior | Ixor | Icomp _ | Icheckbound ->
Arch.is_immediate (Int32.of_int n)
| _ ->
super#is_immediate op n
method is_immediate_test _op n =
Arch.is_immediate (Int32.of_int n)
method! is_simple_expr = function
(* inlined floating-point ops are simple if their arguments are *)
| Cop(Cextcall("sqrt", _, _, _), args, _) when !fpu >= VFPv2 ->
List.for_all self#is_simple_expr args
(* inlined byte-swap ops are simple if their arguments are *)
| Cop(Cextcall("caml_bswap16_direct", _, _, _), args, _)
when !arch >= ARMv6T2 ->
List.for_all self#is_simple_expr args
| Cop(Cextcall("caml_int32_direct_bswap", _, _, _), args, _)
when !arch >= ARMv6 ->
List.for_all self#is_simple_expr args
| e -> super#is_simple_expr e
method! effects_of e =
match e with
| Cop(Cextcall("sqrt", _, _, _), args, _) when !fpu >= VFPv2 ->
Selectgen.Effect_and_coeffect.join_list_map args self#effects_of
| Cop(Cextcall("caml_bswap16_direct", _, _, _), args, _)
when !arch >= ARMv6T2 ->
Selectgen.Effect_and_coeffect.join_list_map args self#effects_of
| Cop(Cextcall("caml_int32_direct_bswap",_ ,_ , _), args, _)
when !arch >= ARMv6 ->
Selectgen.Effect_and_coeffect.join_list_map args self#effects_of
| e -> super#effects_of e
method select_addressing chunk = function
| Cop((Cadda | Caddv), [arg; Cconst_int (n, _)], _)
when is_offset chunk n ->
(Iindexed n, arg)
| Cop((Cadda | Caddv as op),
[arg1; Cop(Caddi, [arg2; Cconst_int (n, _)], _)], dbg)
when is_offset chunk n ->
(Iindexed n, Cop(op, [arg1; arg2], dbg))
| arg ->
(Iindexed 0, arg)
method select_shift_arith op dbg arithop arithrevop args =
match args with
[arg1; Cop(Clsl | Clsr | Casr as op, [arg2; Cconst_int (n, _)], _)]
when n > 0 && n < 32 ->
(Ispecific(Ishiftarith(arithop, select_shiftop op, n)), [arg1; arg2])
| [Cop(Clsl | Clsr | Casr as op, [arg1; Cconst_int (n, _)], _); arg2]
when n > 0 && n < 32 ->
(Ispecific(Ishiftarith(arithrevop, select_shiftop op, n)), [arg2; arg1])
| args ->
begin match super#select_operation op args dbg with
(* Recognize multiply high and add *)
(Iintop Iadd, [Cop(Cmulhi, args, _); arg3])
| (Iintop Iadd, [arg3; Cop(Cmulhi, args, _)]) as op_args
when !arch >= ARMv6 ->
begin match self#select_operation Cmulhi args dbg with
(Iintop Imulh, [arg1; arg2]) ->
(Ispecific Imulhadd, [arg1; arg2; arg3])
| _ -> op_args
end
(* Recognize multiply and add *)
| (Iintop Iadd, [Cop(Cmuli, args, _); arg3])
| (Iintop Iadd, [arg3; Cop(Cmuli, args, _)]) as op_args ->
begin match self#select_operation Cmuli args dbg with
(Iintop Imul, [arg1; arg2]) ->
(Ispecific Imuladd, [arg1; arg2; arg3])
| _ -> op_args
end
(* Recognize multiply and subtract *)
| (Iintop Isub, [arg3; Cop(Cmuli, args, _)]) as op_args
when !arch > ARMv6 ->
begin match self#select_operation Cmuli args dbg with
(Iintop Imul, [arg1; arg2]) ->
(Ispecific Imulsub, [arg1; arg2; arg3])
| _ -> op_args
end
| op_args -> op_args
end
method private iextcall func ty_res ty_args =
Iextcall { func; ty_res; ty_args; alloc = false; }
method! select_operation op args dbg =
match (op, args) with
(* Recognize special forms of add immediate / sub immediate *)
| ((Caddv | Cadda | Caddi), [arg; Cconst_int (n, _)])
when n < 0 && Arch.is_immediate (Int32.of_int (-n)) ->
(Iintop_imm(Isub, -n), [arg])
| (Csubi, [arg; Cconst_int (n, _)])
when n < 0 && Arch.is_immediate (Int32.of_int (-n)) ->
(Iintop_imm(Iadd, -n), [arg])
| (Csubi, [Cconst_int (n, _); arg])
when Arch.is_immediate (Int32.of_int n) ->
(Ispecific(Irevsubimm n), [arg])
(* Recognize special shift arithmetic *)
| ((Caddv | Cadda | Caddi as op), args) ->
self#select_shift_arith op dbg Ishiftadd Ishiftadd args
| (Csubi as op, args) ->
self#select_shift_arith op dbg Ishiftsub Ishiftsubrev args
| (Cand as op, args) ->
self#select_shift_arith op dbg Ishiftand Ishiftand args
| (Cor as op, args) ->
self#select_shift_arith op dbg Ishiftor Ishiftor args
| (Cxor as op, args) ->
self#select_shift_arith op dbg Ishiftxor Ishiftxor args
| (Ccheckbound,
[Cop(Clsl | Clsr | Casr as op, [arg1; Cconst_int (n, _)], _); arg2])
when n > 0 && n < 32 ->
(Ispecific(Ishiftcheckbound(select_shiftop op, n)), [arg1; arg2])
(* Turn integer division/modulus into runtime ABI calls *)
| (Cdivi, args) ->
(self#iextcall "__aeabi_idiv" typ_int [], args)
| (Cmodi, args) ->
(* See above for fix up of return register *)
(self#iextcall "__aeabi_idivmod" typ_int [], args)
(* Recognize 16-bit bswap instruction (ARMv6T2 because we need movt) *)
| (Cextcall("caml_bswap16_direct", _, _, _), args) when !arch >= ARMv6T2 ->
(Ispecific(Ibswap 16), args)
(* Recognize 32-bit bswap instructions (ARMv6 and above) *)
| (Cextcall("caml_int32_direct_bswap", _, _, _), args)
when !arch >= ARMv6 ->
(Ispecific(Ibswap 32), args)
(* Turn floating-point operations into runtime ABI calls for softfp *)
| (op, args) when !fpu = Soft -> self#select_operation_softfp op args dbg
(* Select operations for VFPv{2,3} *)
| (op, args) -> self#select_operation_vfpv3 op args dbg
method private select_operation_softfp op args dbg =
match (op, args) with
(* Turn floating-point operations into runtime ABI calls *)
| (Caddf, args) ->
(self#iextcall "__aeabi_dadd" typ_float [XFloat;XFloat], args)
| (Csubf, args) ->
(self#iextcall "__aeabi_dsub" typ_float [XFloat;XFloat], args)
| (Cmulf, args) ->
(self#iextcall "__aeabi_dmul" typ_float [XFloat;XFloat], args)
| (Cdivf, args) ->
(self#iextcall "__aeabi_ddiv" typ_float [XFloat;XFloat], args)
| (Cfloatofint, args) ->
(self#iextcall "__aeabi_i2d" typ_float [XInt], args)
| (Cintoffloat, args) ->
(self#iextcall "__aeabi_d2iz" typ_int [XFloat], args)
| (Ccmpf comp, args) ->
let comp, func =
match comp with
| CFeq -> Cne, "__aeabi_dcmpeq"
| CFneq -> Ceq, "__aeabi_dcmpeq"
| CFlt -> Cne, "__aeabi_dcmplt"
| CFnlt -> Ceq, "__aeabi_dcmplt"
| CFle -> Cne, "__aeabi_dcmple"
| CFnle -> Ceq, "__aeabi_dcmple"
| CFgt -> Cne, "__aeabi_dcmpgt"
| CFngt -> Ceq, "__aeabi_dcmpgt"
| CFge -> Cne, "__aeabi_dcmpge"
| CFnge -> Ceq, "__aeabi_dcmpge"
in
(Iintop_imm(Icomp(Iunsigned comp), 0),
[Cop(Cextcall(func, typ_int, [XFloat;XFloat], false),
args, dbg)])
(* Add coercions around loads and stores of 32-bit floats *)
| (Cload (Single, mut), args) ->
(self#iextcall "__aeabi_f2d" typ_float [XInt],
[Cop(Cload (Word_int, mut), args, dbg)])
| (Cstore (Single, init), [arg1; arg2]) ->
let arg2' =
Cop(Cextcall("__aeabi_d2f", typ_int, [XFloat], false),
[arg2], dbg) in
self#select_operation (Cstore (Word_int, init)) [arg1; arg2'] dbg
(* Other operations are regular *)
| (op, args) -> super#select_operation op args dbg
method private select_operation_vfpv3 op args dbg =
match (op, args) with
(* Recognize floating-point negate and multiply *)
(Cnegf, [Cop(Cmulf, args, _)]) ->
(Ispecific Inegmulf, args)
(* Recognize floating-point multiply and add *)
| (Caddf, [arg; Cop(Cmulf, args, _)])
| (Caddf, [Cop(Cmulf, args, _); arg]) ->
(Ispecific Imuladdf, arg :: args)
(* Recognize floating-point negate, multiply and subtract *)
| (Csubf, [Cop(Cnegf, [arg], _); Cop(Cmulf, args, _)])
| (Csubf, [Cop(Cnegf, [Cop(Cmulf, args, _)], _); arg]) ->
(Ispecific Inegmulsubf, arg :: args)
(* Recognize floating-point negate, multiply and add *)
| (Csubf, [arg; Cop(Cmulf, args, _)]) ->
(Ispecific Inegmuladdf, arg :: args)
(* Recognize multiply and subtract *)
| (Csubf, [Cop(Cmulf, args, _); arg]) ->
(Ispecific Imulsubf, arg :: args)
(* Recognize floating-point square root *)
| (Cextcall("sqrt", _, _, false), args) ->
(Ispecific Isqrtf, args)
(* Other operations are regular *)
| (op, args) -> super#select_operation op args dbg
method! select_condition = function
(* Turn floating-point comparisons into runtime ABI calls *)
Cop(Ccmpf _ as op, args, dbg) when !fpu = Soft ->
begin match self#select_operation_softfp op args dbg with
(Iintop_imm(Icomp(Iunsigned Ceq), 0), [arg]) -> (Ifalsetest, arg)
| (Iintop_imm(Icomp(Iunsigned Cne), 0), [arg]) -> (Itruetest, arg)
| _ -> assert false
end
| expr ->
super#select_condition expr
(* Deal with some register constraints *)
method! insert_op_debug env op dbg rs rd =
try
let (rsrc, rdst) = pseudoregs_for_operation op rs rd in
self#insert_moves env rs rsrc;
self#insert_debug env (Iop op) dbg rsrc rdst;
self#insert_moves env rdst rd;
rd
with Use_default ->
super#insert_op_debug env op dbg rs rd
end
let fundecl f = (new selector)#emit_fundecl f