diff options
author | Tristan Gingold | 2015-11-14 21:54:50 +0100 |
---|---|---|
committer | Tristan Gingold | 2015-11-14 21:54:50 +0100 |
commit | 20ea3682027c0725a02797665b58dc4adb382851 (patch) | |
tree | 76250347bbc66272781e9d7e34efeaa88ac39696 /src/ortho/mcode | |
parent | 826d00295979c7b5e88e1150c191ebc2f9302f52 (diff) | |
download | ghdl-20ea3682027c0725a02797665b58dc4adb382851.tar.gz ghdl-20ea3682027c0725a02797665b58dc4adb382851.tar.bz2 ghdl-20ea3682027c0725a02797665b58dc4adb382851.zip |
mcode x86: fix regressions.
Diffstat (limited to 'src/ortho/mcode')
-rw-r--r-- | src/ortho/mcode/ortho_code-debug.adb | 4 | ||||
-rw-r--r-- | src/ortho/mcode/ortho_code-debug.ads | 18 | ||||
-rw-r--r-- | src/ortho/mcode/ortho_code-exprs.adb | 9 | ||||
-rw-r--r-- | src/ortho/mcode/ortho_code-exprs.ads | 1 | ||||
-rw-r--r-- | src/ortho/mcode/ortho_code-x86-abi.adb | 9 | ||||
-rw-r--r-- | src/ortho/mcode/ortho_code-x86-abi.ads | 1 | ||||
-rw-r--r-- | src/ortho/mcode/ortho_code-x86-insns.adb | 174 |
7 files changed, 143 insertions, 73 deletions
diff --git a/src/ortho/mcode/ortho_code-debug.adb b/src/ortho/mcode/ortho_code-debug.adb index 0f3e01a..1caa6c0 100644 --- a/src/ortho/mcode/ortho_code-debug.adb +++ b/src/ortho/mcode/ortho_code-debug.adb @@ -65,10 +65,8 @@ package body Ortho_Code.Debug is case C is when 'a' => Flag_Debug_Asm := True; - when 'b' => - Flag_Debug_Body := True; when 'B' => - Flag_Debug_Body2 := True; + Flag_Debug_Body := True; when 'c' => Flag_Debug_Code := True; when 'C' => diff --git a/src/ortho/mcode/ortho_code-debug.ads b/src/ortho/mcode/ortho_code-debug.ads index 03f550a..1eb3652 100644 --- a/src/ortho/mcode/ortho_code-debug.ads +++ b/src/ortho/mcode/ortho_code-debug.ads @@ -22,14 +22,15 @@ package Ortho_Code.Debug is procedure Disp_Mode (M : Mode_Type); - -- Set a debug flag. + -- Set a debug flag (--be-debug=X). procedure Set_Debug_Be_Flag (C : Character); -- any '--be-XXX=YY' option. procedure Set_Be_Flag (Str : String); - -- c: tree created, before any back-end. + -- --be-disp=c: tree created, before any back-end. Flag_Disp_Code : Boolean := False; + -- --be-dump=c: tree created, before any back-end. Flag_Dump_Code : Boolean := False; -- a: disp assembly code. @@ -38,16 +39,14 @@ package Ortho_Code.Debug is -- A: do internal checks (assertions). Flag_Debug_Assert : Boolean := True; - -- b: disp top-level subprogram body before code generation. + -- B: dump generated insns (at the end of insn generation). Flag_Debug_Body : Boolean := False; - -- B: disp top-level subprogram body after code generation. - Flag_Debug_Body2 : Boolean := False; - - -- c: display generated code. + -- c: display generated insns (at the end of insn generation). + -- This is a log dump. Flag_Debug_Code : Boolean := False; - -- C: display generated code just before asm. + -- C: display insns when generating code. Useful to debug code generation. Flag_Debug_Code2 : Boolean := False; -- h: disp bytes generated (in hexa). @@ -60,6 +59,9 @@ package Ortho_Code.Debug is Flag_Debug_Dump : Boolean := False; -- i: disp insns, when generated. + -- The output may be misleading as a spill inserted later is not displayed. + -- Useful only when debugging insn generation. Use --be-debug=c to view + -- the correct output. Flag_Debug_Insn : Boolean := False; -- s: disp stats (number of nodes). diff --git a/src/ortho/mcode/ortho_code-exprs.adb b/src/ortho/mcode/ortho_code-exprs.adb index a45c9f8..17a47f4 100644 --- a/src/ortho/mcode/ortho_code-exprs.adb +++ b/src/ortho/mcode/ortho_code-exprs.adb @@ -185,6 +185,11 @@ package body Ortho_Code.Exprs is return Int32 (Enodes.Table (Enode).Arg1); end Get_Stack_Adjust; + procedure Set_Stack_Adjust (Enode : O_Enode; Off : Int32) is + begin + Enodes.Table (Enode).Arg1 := O_Enode (Off); + end Set_Stack_Adjust; + function Get_Arg_Link (Enode : O_Enode) return O_Enode is begin return Enodes.Table (Enode).Arg2; @@ -582,9 +587,7 @@ package body Ortho_Code.Exprs is -- Disp declarations. if Cur_Subprg.Parent = null then - if Ortho_Code.Debug.Flag_Debug_Body - or Ortho_Code.Debug.Flag_Debug_Code - then + if Ortho_Code.Debug.Flag_Debug_Code then while Last_Decl <= D_Body loop case Get_Decl_Kind (Last_Decl) is when OD_Block => diff --git a/src/ortho/mcode/ortho_code-exprs.ads b/src/ortho/mcode/ortho_code-exprs.ads index f8ee88a..971c57a 100644 --- a/src/ortho/mcode/ortho_code-exprs.ads +++ b/src/ortho/mcode/ortho_code-exprs.ads @@ -339,6 +339,7 @@ package Ortho_Code.Exprs is -- stack pointer stays aligned. For negtive values, this is the amount of -- bytes to release on the stack. function Get_Stack_Adjust (Enode : O_Enode) return Int32; + procedure Set_Stack_Adjust (Enode : O_Enode; Off : Int32); -- Get the subprogram called by ENODE. function Get_Call_Subprg (Enode : O_Enode) return O_Dnode; diff --git a/src/ortho/mcode/ortho_code-x86-abi.adb b/src/ortho/mcode/ortho_code-x86-abi.adb index 8af6a57..2be10fe 100644 --- a/src/ortho/mcode/ortho_code-x86-abi.adb +++ b/src/ortho/mcode/ortho_code-x86-abi.adb @@ -104,7 +104,7 @@ package body Ortho_Code.X86.Abi is Insns.Gen_Subprg_Insns (Subprg); - if Ortho_Code.Debug.Flag_Debug_Body2 then + if Ortho_Code.Debug.Flag_Debug_Body then Disp_Subprg_Body (1, Subprg.E_Entry); end if; @@ -353,8 +353,8 @@ package body Ortho_Code.X86.Abi is is use Ada.Text_IO; use Debug.Int32_IO; - Kind : OE_Kind; - Mode : Mode_Type; + Kind : constant OE_Kind := Get_Expr_Kind (Stmt); + Mode : constant Mode_Type := Get_Expr_Mode (Stmt); procedure Disp_Op_Name (Name : String) is begin @@ -373,9 +373,6 @@ package body Ortho_Code.X86.Abi is end Disp_Reg_Op_Name; begin - Kind := Get_Expr_Kind (Stmt); - Mode := Get_Expr_Mode (Stmt); - case Kind is when OE_Beg => Put (" # block start"); diff --git a/src/ortho/mcode/ortho_code-x86-abi.ads b/src/ortho/mcode/ortho_code-x86-abi.ads index 6a07127..e22dc04 100644 --- a/src/ortho/mcode/ortho_code-x86-abi.ads +++ b/src/ortho/mcode/ortho_code-x86-abi.ads @@ -39,7 +39,6 @@ package Ortho_Code.X86.Abi is -- If True, use SSE/SSE2 instructions instead of FPU one. The code is -- still compliant with the ABI (ie FP values are returned in st0). - -- TODO: this is still work in progress. Flag_Sse2 : constant Boolean := True; -- Procedures to layout a subprogram declaration. diff --git a/src/ortho/mcode/ortho_code-x86-insns.adb b/src/ortho/mcode/ortho_code-x86-insns.adb index d57938a..e975455 100644 --- a/src/ortho/mcode/ortho_code-x86-insns.adb +++ b/src/ortho/mcode/ortho_code-x86-insns.adb @@ -15,6 +15,52 @@ -- along with GCC; see the file COPYING. If not, write to the Free -- Software Foundation, 59 Temple Place - Suite 330, Boston, MA -- 02111-1307, USA. + +-- Instruction pass for mcode x86. +-- +-- The purpose of this pass is the transform the AST (the input) into a list +-- of x86 instructions and to allocate registers. +-- +-- The AST given in input is already linearized: ifs, loops, cases have been +-- translated to labels and jumps. So the input is a list of statement to +-- execute, intermixed with declaration blocks. +-- +-- The first purpose of this pass is to translate statements (and expressions) +-- to x86 instructions. This isn't particularly difficult as they are already +-- low-level statements and expression (by design of the language). The +-- algorithm simply try to put as much as possible into an instruction (in +-- order to use the address operand encoding of x86: base, index and scale): +-- AST is split into small trees (sometime as small as a single node) and +-- linearized. Each node represent a fix pattern of one or a few instructions +-- (in some case, like a 64 bit addition, we need more than one x86 +-- instruction). +-- The core functions of this package (Gen_Insn and Gen_Insn_Stmt) do the +-- work: they call Gen_Insn for each operand, then append themself to the +-- result using Link_Stmt. +-- +-- The second purpose of this pass is to perform register allocation. This +-- is done in the same time. +-- There are two sources of constraints for register allocation: +-- - external constraint on the result: for example, the return value of +-- a function must be in a fixed register (defined by the ABI). +-- - instruction constraint on the result: some x86 instructions (like div) +-- specify the result register. This constraint will be forward propagated +-- to next instructions. +-- - instruction constraint on the operand: most x86 instructions set the +-- result in one of the operand register, and some instructions (like shl) +-- have a fixed register for an operand (like the shift count). This +-- constraint has to be backward propagated to previous instructions. +-- Obviously constraints may be incompatible: the result of an instruction +-- may be in a different register than the input of the next instruction. +-- In this case, move instructions are added. +-- It is possible (and quite easily) to run out of registers. In that case +-- some values must be spilt (save) on the stack and will be reloaded later. +-- Registers are allocated statement by statement. So after each statement +-- all registers should be unused (this is a very basic register allocator). +-- +-- Finally, this pass also allocate stack slots for local variables, and +-- compute the size of the frame. + with Interfaces; with Ada.Text_IO; with Ortho_Code.Abi; @@ -24,6 +70,7 @@ with Ortho_Code.Debug; with Ortho_Code.X86.Flags; package body Ortho_Code.X86.Insns is + -- Add STMT to the list of instructions. procedure Link_Stmt (Stmt : O_Enode) is use Ortho_Code.Abi; @@ -35,6 +82,7 @@ package body Ortho_Code.X86.Insns is end if; end Link_Stmt; + -- Return the 'any register' constraint for mode MODE. function Get_Reg_Any (Mode : Mode_Type) return O_Reg is begin case Mode is @@ -104,9 +152,11 @@ package body Ortho_Code.X86.Insns is case Get_Decl_Kind (Decl) is when OD_Local => Decl_Type := Get_Decl_Type (Decl); + -- Align and allocate (on the stack). Stack_Offset := Do_Align (Stack_Offset, Decl_Type); Stack_Offset := Stack_Offset + Get_Type_Size (Decl_Type); Set_Local_Offset (Decl, -Int32 (Stack_Offset)); + -- If the frame gets lager, set the maximum size. if Stack_Offset > Stack_Max then Stack_Max := Stack_Offset; end if; @@ -127,6 +177,7 @@ package body Ortho_Code.X86.Insns is end loop; end Expand_Decls; + -- Condition code for unsigned comparaison. function Ekind_Unsigned_To_Cc (Kind : OE_Kind_Cmp) return O_Reg is begin case Kind is @@ -145,6 +196,7 @@ package body Ortho_Code.X86.Insns is end case; end Ekind_Unsigned_To_Cc; + -- Condition code for signed comparaison. function Ekind_Signed_To_Cc (Kind : OE_Kind_Cmp) return O_Reg is begin case Kind is @@ -211,7 +263,7 @@ package body Ortho_Code.X86.Insns is end case; end Reverse_Cc; - -- Get the register in which a result of MODE is returned. + -- Get the register in which a function result for MODE is returned. function Get_Return_Register (Mode : Mode_Type) return O_Reg is begin case Mode is @@ -289,7 +341,6 @@ package body Ortho_Code.X86.Insns is return Insn_Num; end Get_Insn_Num; - type Reg_Info_Type is record -- Statement number which use this register. -- This is a distance. @@ -301,7 +352,7 @@ package body Ortho_Code.X86.Insns is Stmt : O_Enode; -- If set, this register has been used. - -- All callee-saved registers marked must be saved. + -- All callee-saved registers marked 'used' must be saved in the prolog. Used : Boolean; end record; @@ -390,34 +441,26 @@ package body Ortho_Code.X86.Insns is -- Mark a register as unused. procedure Free_R32 (Reg : O_Reg) is begin - if Regs (Reg).Num = O_Free then - raise Program_Error; - end if; + pragma Assert (Regs (Reg).Num /= O_Free); Regs (Reg).Num := O_Free; end Free_R32; procedure Free_Fp is begin - if Fp_Regs (Fp_Top).Stmt = O_Enode_Null then - raise Program_Error; - end if; - Fp_Regs (Fp_Top).Stmt := O_Enode_Null; + pragma Assert (Fp_Regs (Fp_Top).Num /= O_Free); + Fp_Regs (Fp_Top).Num := O_Free; Fp_Top := Fp_Top + 1; end Free_Fp; procedure Free_Cc is begin - if Reg_Cc.Num = O_Free then - raise Program_Error; - end if; + pragma Assert (Reg_Cc.Num /= O_Free); Reg_Cc.Num := O_Free; end Free_Cc; procedure Free_Xmm (Reg : O_Reg) is begin - if Xmm_Regs (Reg).Num = O_Free then - raise Program_Error; - end if; + pragma Assert (Xmm_Regs (Reg).Num /= O_Free); Xmm_Regs (Reg).Num := O_Free; end Free_Xmm; @@ -491,7 +534,7 @@ package body Ortho_Code.X86.Insns is end case; end Spill_R32; - procedure Alloc_R32 (Reg : O_Reg; Stmt : O_Enode; Num : O_Inum) is + procedure Alloc_R32 (Reg : Regs_R32; Stmt : O_Enode; Num : O_Inum) is begin if Regs (Reg).Num /= O_Free then Spill_R32 (Reg); @@ -879,6 +922,22 @@ package body Ortho_Code.X86.Insns is Link_Stmt (N); end Insert_Arg; + -- Mark all registers that aren't preserved by a call as clobbered, so that + -- they are saved. + procedure Clobber_Caller_Saved_Registers is + begin + Clobber_R32 (R_Ax); + Clobber_R32 (R_Dx); + Clobber_R32 (R_Cx); + -- FIXME: fp regs. + + if Abi.Flag_Sse2 then + for R in Regs_Xmm loop + Clobber_Xmm (R); + end loop; + end if; + end Clobber_Caller_Saved_Registers; + function Insert_Intrinsic (Stmt : O_Enode; Reg : O_Reg; Num : O_Inum) return O_Enode is @@ -929,9 +988,7 @@ package body Ortho_Code.X86.Insns is end case; -- Save caller-saved registers. - Clobber_R32 (R_Ax); - Clobber_R32 (R_Dx); - Clobber_R32 (R_Cx); + Clobber_Caller_Saved_Registers; N := New_Enode (OE_Intrinsic, Mode, O_Tnode_Null, O_Enode (Op), O_Enode_Null); @@ -976,6 +1033,27 @@ package body Ortho_Code.X86.Insns is return Stmt; end Gen_Conv_From_Fp_Insn; + procedure Gen_Stack_Adjust (Off : Int32) + is + use Ortho_Code.Abi; + Stmt : O_Enode; + begin + if Get_Expr_Kind (Last_Link) = OE_Stack_Adjust then + -- The last instruction was already a stack_adjust. Change the + -- value. + Set_Stack_Adjust (Last_Link, + Get_Stack_Adjust (Last_Link) + Off); + if Debug.Flag_Debug_Insn then + Ada.Text_IO.Put (" patched:"); + Disp_Stmt (Last_Link); + end if; + else + Stmt := New_Enode (OE_Stack_Adjust, Mode_Nil, O_Tnode_Null, + O_Enode (Off), O_Enode_Null); + Link_Stmt (Stmt); + end if; + end Gen_Stack_Adjust; + function Gen_Call (Stmt : O_Enode; Reg : O_Reg; Pnum : O_Inum) return O_Enode is @@ -992,8 +1070,7 @@ package body Ortho_Code.X86.Insns is Pad := (Push_Size + Push_Offset) and Uns32 (Flags.Stack_Boundary - 1); if Pad /= 0 then Pad := Uns32 (Flags.Stack_Boundary) - Pad; - Link_Stmt (New_Enode (OE_Stack_Adjust, Mode_Nil, O_Tnode_Null, - O_Enode (Pad), O_Enode_Null)); + Gen_Stack_Adjust (Int32 (Pad)); end if; -- The stack has been adjusted by Pad bytes. Push_Offset := Push_Offset + Pad; @@ -1005,16 +1082,7 @@ package body Ortho_Code.X86.Insns is end if; -- Clobber registers. - Clobber_R32 (R_Ax); - Clobber_R32 (R_Dx); - Clobber_R32 (R_Cx); - -- FIXME: fp regs. - - if Abi.Flag_Sse2 then - for R in Regs_Xmm loop - Clobber_Xmm (R); - end loop; - end if; + Clobber_Caller_Saved_Registers; -- Add the call. Reg_Res := Get_Return_Register (Get_Expr_Mode (Stmt)); @@ -1023,11 +1091,7 @@ package body Ortho_Code.X86.Insns is Res_Stmt := Stmt; if Push_Size + Pad /= 0 then - Res_Stmt := - New_Enode (OE_Stack_Adjust, Get_Expr_Mode (Stmt), O_Tnode_Null, - O_Enode (-Int32 (Push_Size + Pad)), O_Enode_Null); - Set_Expr_Reg (Res_Stmt, Reg_Res); - Link_Stmt (Res_Stmt); + Gen_Stack_Adjust (-Int32 (Push_Size + Pad)); end if; -- The stack has been restored (just after the call). @@ -1589,10 +1653,10 @@ package body Ortho_Code.X86.Insns is | Regs_R64 | Regs_Fp | Regs_Xmm => - Right := Gen_Insn (Right, R_Irm, Num); Left := Gen_Insn (Left, Reg, Num); - Right := Reload (Right, R_Irm, Num); + Right := Gen_Insn (Right, R_Irm, Num); Left := Reload (Left, Reg, Num); + Right := Reload (Right, R_Irm, Num); Reg_Res := Get_Expr_Reg (Left); when others => Error_Gen_Insn (Stmt, Reg); @@ -1667,10 +1731,10 @@ package body Ortho_Code.X86.Insns is else Reg_Res := R_St0; end if; - Right := Gen_Insn (Right, R_Irm, Num); Left := Gen_Insn (Left, Reg_Res, Num); - Right := Reload (Right, R_Irm, Num); + Right := Gen_Insn (Right, R_Irm, Num); Left := Reload (Left, Reg_Res, Num); + Right := Reload (Right, R_Irm, Num); Reg_Res := Get_Expr_Reg (Left); Set_Expr_Right (Stmt, Right); Set_Expr_Left (Stmt, Left); @@ -1874,19 +1938,20 @@ package body Ortho_Code.X86.Insns is | R_Ir | R_Sib | R_Any32 - | Regs_R32 | R_Any64 | R_Any8 + | R_Any_Xmm => + Reg_Res := Get_Reg_Any (Stmt); + when Regs_R32 | Regs_R64 | Regs_Fp - | R_Any_Xmm | Regs_Xmm => - Free_Insn_Regs (Left); - Set_Expr_Reg - (Stmt, Alloc_Reg (Get_Reg_Any (Stmt), Stmt, Pnum)); + Reg_Res := Reg; when others => Error_Gen_Insn (Stmt, Reg); end case; + Free_Insn_Regs (Left); + Set_Expr_Reg (Stmt, Alloc_Reg (Reg_Res, Stmt, Pnum)); Link_Stmt (Stmt); return Stmt; end; @@ -1969,12 +2034,12 @@ package body Ortho_Code.X86.Insns is case Kind is when OE_Asgn => - Left := Gen_Insn (Get_Expr_Operand (Stmt), R_Ir, Num); - Right := Gen_Insn (Get_Assign_Target (Stmt), R_Sib, Num); - Left := Reload (Left, R_Ir, Num); - --Right := Reload (Right, R_Sib, Num); - Set_Expr_Operand (Stmt, Left); - Set_Assign_Target (Stmt, Right); + Right := Gen_Insn (Get_Expr_Operand (Stmt), R_Ir, Num); + Left := Gen_Insn (Get_Assign_Target (Stmt), R_Sib, Num); + Right := Reload (Right, R_Ir, Num); + --Left := Reload (Left, R_Sib, Num); + Set_Expr_Operand (Stmt, Right); + Set_Assign_Target (Stmt, Left); Link_Stmt (Stmt); Free_Insn_Regs (Left); Free_Insn_Regs (Right); @@ -1995,11 +2060,15 @@ package body Ortho_Code.X86.Insns is begin Cur_Block := Stmt; Block_Decl := Get_Block_Decls (Cur_Block); + -- Save current frame size (to be restored at end of block). Set_Block_Max_Stack (Block_Decl, Stack_Offset); + -- Allocate slots for local declarations. Expand_Decls (Block_Decl); end; Link_Stmt (Stmt); when OE_End => + -- Restore current frame size (so deallocate the slots for the + -- local declarations). Swap_Stack_Offset (Get_Block_Decls (Cur_Block)); Cur_Block := Get_Block_Parent (Cur_Block); Link_Stmt (Stmt); @@ -2042,6 +2111,7 @@ package body Ortho_Code.X86.Insns is case Kind is when OE_Beg | OE_End => + -- Stack offset has been explicitely changed for local variables. null; when others => Stack_Offset := Prev_Stack_Offset; |