diff options
| author | Tristan Gingold <tgingold@free.fr> | 2015-11-14 21:54:50 +0100 | 
|---|---|---|
| committer | Tristan Gingold <tgingold@free.fr> | 2015-11-14 21:54:50 +0100 | 
| commit | 20ea3682027c0725a02797665b58dc4adb382851 (patch) | |
| tree | 76250347bbc66272781e9d7e34efeaa88ac39696 /src | |
| parent | 826d00295979c7b5e88e1150c191ebc2f9302f52 (diff) | |
| download | ghdl-20ea3682027c0725a02797665b58dc4adb382851.tar.gz ghdl-20ea3682027c0725a02797665b58dc4adb382851.tar.bz2 ghdl-20ea3682027c0725a02797665b58dc4adb382851.zip | |
mcode x86: fix regressions.
Diffstat (limited to 'src')
| -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 0f3e01ab9..1caa6c0b4 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 03f550ac9..1eb3652ef 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 a45c9f802..17a47f41c 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 f8ee88a8e..971c57a68 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 8af6a57aa..2be10fe0e 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 6a07127e2..e22dc04ba 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 d57938a1d..e975455b6 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; | 
