File: x86-gen-fn.sml

package info (click to toggle)
mlton 20210117%2Bdfsg-3
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 58,464 kB
  • sloc: ansic: 27,682; sh: 4,455; asm: 3,569; lisp: 2,879; makefile: 2,347; perl: 1,169; python: 191; pascal: 68; javascript: 7
file content (79 lines) | stat: -rw-r--r-- 2,282 bytes parent folder | download | duplicates (5)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
(* x86-gen-fn.sml
 *
 * X86-specific portion of the interpreter.
 *)

functor X86GenFn (
	structure Gen : C_CALL_GEN
	structure MLRISCGen : MLRISC_GEN
          where T = Gen.T
	val push : MLRISCGen.T.rexp -> MLRISCGen.T.stm
	val leave : MLRISCGen.T.stm
  ) = struct

    val defTy = 32

    structure T = MLRISCGen.T
    structure C = X86Cells
    structure CB = CellsBasis
    structure Consts = VarargConstants

    fun gpr r = T.GPR(T.REG(32, r))
    val calleeSaveRegs = List.map gpr [C.ebx, C.esi, C.edi]

    structure InterpGen = GenFn (
	structure T = Gen.T
	val gprs = []
	val fprs = []
	val gprWidths = [8, 16, 32]
	val fprWidths = [32, 64]
	val spReg = T.REG(32, C.esp)
	val defaultWidth = 32
	val callerSaves = [C.eax, C.ecx, C.edx]
	val callerSavesF = []
	structure CCallGen = Gen
	structure SA = Gen.SA)

    fun lit i = T.LI (T.I.fromInt (defTy, i))

  (* get the ith argument in the calling sequence *)
    fun getArg i = 
	    T.LOAD(defTy, T.ADD(defTy, T.REG(defTy, C.ebp), lit (4*i+8)), T.Region.memory)

  (* MLRISC code for the x86 vararg interpreter *)
    fun gen () = let
	   val largsReg = C.newReg()
         (* we align the frame to a 16-bytes to support Mac OS. *)
	   val frameSzB = 1024*4-2*4
	   val interpFunPtr = getArg 0
	   val endOfLargs = getArg 2
           in
	       List.concat [
	         (* preserve callee-save registers *)
	           [T.LIVE calleeSaveRegs],
		   [push (T.REG(defTy, C.ebp))],
		   [T.COPY (defTy, [C.ebp], [C.esp])],
		   [T.MV(defTy, largsReg, getArg 1)],
		 (* allocate stack space for the arguments *)
		   [T.MV(defTy, C.esp, T.SUB(defTy, T.REG(defTy, C.esp), lit frameSzB))],
	           InterpGen.gen {interpFunPtr=interpFunPtr, largsReg=largsReg, endOfLargs=endOfLargs},
		   [leave],
	           [T.LIVE calleeSaveRegs],
		   [T.RET []]
		   ]
	   end

    fun main (cmd, args) = let
	   val _ = Label.reset()
	   val lab = Label.global Consts.varargInterpreter
	   val stms = gen()
	   val asmOutStrm = TextIO.openOut "vararg-interp-x86-linux.s"
	   val _ = TextIO.output(asmOutStrm, Consts.header^"\n")
	   fun doit () = MLRISCGen.gen(lab, stms, [T.GPR (T.REG (defTy, C.eax))])
	   val _ = AsmStream.withStream asmOutStrm doit ()
	   val _ = TextIO.closeOut asmOutStrm
           in
	      0
	   end

  end