diff options
Diffstat (limited to 'ortho/mcode/binary_file.adb')
-rw-r--r-- | ortho/mcode/binary_file.adb | 977 |
1 files changed, 0 insertions, 977 deletions
diff --git a/ortho/mcode/binary_file.adb b/ortho/mcode/binary_file.adb deleted file mode 100644 index 6043d7319..000000000 --- a/ortho/mcode/binary_file.adb +++ /dev/null @@ -1,977 +0,0 @@ --- Binary file handling. --- Copyright (C) 2006 Tristan Gingold --- --- GHDL is free software; you can redistribute it and/or modify it under --- the terms of the GNU General Public License as published by the Free --- Software Foundation; either version 2, or (at your option) any later --- version. --- --- GHDL is distributed in the hope that it will be useful, but WITHOUT ANY --- WARRANTY; without even the implied warranty of MERCHANTABILITY or --- FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License --- for more details. --- --- You should have received a copy of the GNU General Public License --- 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. -with System.Storage_Elements; -with Ada.Text_IO; use Ada.Text_IO; -with Ada.Characters.Latin_1; -with Ada.Unchecked_Conversion; -with Hex_Images; use Hex_Images; -with Disassemble; - -package body Binary_File is - Cur_Sect : Section_Acc := null; - - HT : Character renames Ada.Characters.Latin_1.HT; - - function To_Byte_Array_Acc is new Ada.Unchecked_Conversion - (Source => System.Address, Target => Byte_Array_Acc); - - -- Resize a section to SIZE bytes. - procedure Resize (Sect : Section_Acc; Size : Pc_Type) - is - begin - Sect.Data_Max := Size; - Memsegs.Resize (Sect.Seg, Natural (Size)); - Sect.Data := To_Byte_Array_Acc (Memsegs.Get_Address (Sect.Seg)); - end Resize; - - function Get_Scope (Sym : Symbol) return Symbol_Scope is - begin - return Symbols.Table (Sym).Scope; - end Get_Scope; - - procedure Set_Scope (Sym : Symbol; Scope : Symbol_Scope) is - begin - Symbols.Table (Sym).Scope := Scope; - end Set_Scope; - - function Get_Section (Sym : Symbol) return Section_Acc is - begin - return Symbols.Table (Sym).Section; - end Get_Section; - - procedure Set_Section (Sym : Symbol; Sect : Section_Acc) is - begin - Symbols.Table (Sym).Section := Sect; - end Set_Section; - - function Get_Number (Sym : Symbol) return Natural is - begin - return Symbols.Table (Sym).Number; - end Get_Number; - - procedure Set_Number (Sym : Symbol; Num : Natural) is - begin - Symbols.Table (Sym).Number := Num; - end Set_Number; - - function Get_Relocs (Sym : Symbol) return Reloc_Acc is - begin - return Symbols.Table (Sym).Relocs; - end Get_Relocs; - - procedure Set_Relocs (Sym : Symbol; Reloc : Reloc_Acc) is - begin - Symbols.Table (Sym).Relocs := Reloc; - end Set_Relocs; - - function Get_Name (Sym : Symbol) return O_Ident is - begin - return Symbols.Table (Sym).Name; - end Get_Name; - - function Get_Used (Sym : Symbol) return Boolean is - begin - return Symbols.Table (Sym).Used; - end Get_Used; - - procedure Set_Used (Sym : Symbol; Val : Boolean) is - begin - Symbols.Table (Sym).Used := Val; - end Set_Used; - - function Get_Symbol_Value (Sym : Symbol) return Pc_Type is - begin - return Symbols.Table (Sym).Value; - end Get_Symbol_Value; - - procedure Set_Symbol_Value (Sym : Symbol; Val : Pc_Type) is - begin - Symbols.Table (Sym).Value := Val; - end Set_Symbol_Value; - - function S_Defined (Sym : Symbol) return Boolean is - begin - return Get_Scope (Sym) /= Sym_Undef; - end S_Defined; - pragma Unreferenced (S_Defined); - - function S_Local (Sym : Symbol) return Boolean is - begin - return Get_Scope (Sym) = Sym_Local; - end S_Local; - - procedure Create_Section (Sect : out Section_Acc; - Name : String; Flags : Section_Flags) - is - begin - Sect := new Section_Type'(Next => null, - Flags => Flags, - Name => new String'(Name), - Link => null, - Align => 2, - Esize => 0, - Pc => 0, - Insn_Pc => 0, - Data => null, - Data_Max => 0, - First_Reloc => null, - Last_Reloc => null, - Nbr_Relocs => 0, - Number => 0, - Seg => Memsegs.Create, - Vaddr => 0); - if (Flags and Section_Zero) = 0 then - -- Allocate memory for the segment, unless BSS. - Resize (Sect, 8192); - end if; - if (Flags and Section_Strtab) /= 0 then - Sect.Align := 0; - end if; - if Section_Chain = null then - Section_Chain := Sect; - else - Section_Last.Next := Sect; - end if; - Section_Last := Sect; - Nbr_Sections := Nbr_Sections + 1; - end Create_Section; - - procedure Sect_Prealloc (Sect : Section_Acc; L : Pc_Type) - is - New_Max : Pc_Type; - begin - if Sect.Pc + L < Sect.Data_Max then - return; - end if; - New_Max := Sect.Data_Max; - loop - New_Max := New_Max * 2; - exit when Sect.Pc + L < New_Max; - end loop; - Resize (Sect, New_Max); - end Sect_Prealloc; - - procedure Merge_Section (Dest : Section_Acc; Src : Section_Acc) - is - Rel : Reloc_Acc; - begin - -- Sanity checks. - if Src = null or else Dest = Src then - raise Program_Error; - end if; - - Rel := Src.First_Reloc; - - if Rel /= null then - -- Move relocs. - if Dest.Last_Reloc = null then - Dest.First_Reloc := Rel; - Dest.Last_Reloc := Rel; - else - Dest.Last_Reloc.Sect_Next := Rel; - Dest.Last_Reloc := Rel; - end if; - Dest.Nbr_Relocs := Dest.Nbr_Relocs + Src.Nbr_Relocs; - - - -- Reloc reloc, since the pc has changed. - while Rel /= null loop - Rel.Addr := Rel.Addr + Dest.Pc; - Rel := Rel.Sect_Next; - end loop; - end if; - - if Src.Pc > 0 then - Sect_Prealloc (Dest, Src.Pc); - Dest.Data (Dest.Pc .. Dest.Pc + Src.Pc - 1) := - Src.Data (0 .. Src.Pc - 1); - Dest.Pc := Dest.Pc + Src.Pc; - end if; - - Memsegs.Delete (Src.Seg); - Src.Pc := 0; - Src.Data_Max := 0; - Src.Data := null; - Src.First_Reloc := null; - Src.Last_Reloc := null; - Src.Nbr_Relocs := 0; - - -- Remove from section_chain. - if Section_Chain = Src then - Section_Chain := Src.Next; - else - declare - Sect : Section_Acc; - begin - Sect := Section_Chain; - while Sect.Next /= Src loop - Sect := Sect.Next; - end loop; - Sect.Next := Src.Next; - if Section_Last = Src then - Section_Last := Sect; - end if; - end; - end if; - Nbr_Sections := Nbr_Sections - 1; - end Merge_Section; - - procedure Set_Section_Info (Sect : Section_Acc; - Link : Section_Acc; - Align : Natural; - Esize : Natural) - is - begin - Sect.Link := Link; - Sect.Align := Align; - Sect.Esize := Esize; - end Set_Section_Info; - - procedure Set_Current_Section (Sect : Section_Acc) is - begin - -- If the current section does not change, this is a no-op. - if Cur_Sect = Sect then - return; - end if; - - if Dump_Asm then - Put_Line (HT & ".section """ & Sect.Name.all & """"); - end if; - Cur_Sect := Sect; - end Set_Current_Section; - - function Get_Current_Pc return Pc_Type is - begin - return Cur_Sect.Pc; - end Get_Current_Pc; - - function Get_Pc (Sect : Section_Acc) return Pc_Type is - begin - return Sect.Pc; - end Get_Pc; - - - procedure Prealloc (L : Pc_Type) is - begin - Sect_Prealloc (Cur_Sect, L); - end Prealloc; - - procedure Start_Insn is - begin - -- Check there is enough memory for the next instruction. - Sect_Prealloc (Cur_Sect, 16); - if Cur_Sect.Insn_Pc /= 0 then - -- end_insn was not called. - raise Program_Error; - end if; - Cur_Sect.Insn_Pc := Cur_Sect.Pc; - end Start_Insn; - - procedure Get_Symbol_At_Addr (Addr : System.Address; - Line : in out String; - Line_Len : in out Natural) - is - use System; - use System.Storage_Elements; - Off : Pc_Type; - Reloc : Reloc_Acc; - begin - -- Check if addr is in the current section. - if Addr < Cur_Sect.Data (0)'Address - or else Addr > Cur_Sect.Data (Cur_Sect.Pc)'Address - then - raise Program_Error; - --return; - end if; - Off := Pc_Type - (To_Integer (Addr) - To_Integer (Cur_Sect.Data (0)'Address)); - - -- Find a relocation at OFF. - Reloc := Cur_Sect.First_Reloc; - while Reloc /= null loop - if Reloc.Addr = Off then - declare - Str : constant String := Get_Symbol_Name (Reloc.Sym); - begin - Line (Line'First .. Line'First + Str'Length - 1) := Str; - Line_Len := Line_Len + Str'Length; - return; - end; - end if; - Reloc := Reloc.Sect_Next; - end loop; - end Get_Symbol_At_Addr; - - procedure End_Insn - is - Str : String (1 .. 256); - Len : Natural; - Insn_Len : Natural; - begin - --if Insn_Pc = 0 then - -- -- start_insn was not called. - -- raise Program_Error; - --end if; - if Debug_Hex then - Put (HT); - Put ('#'); - for I in Cur_Sect.Insn_Pc .. Cur_Sect.Pc - 1 loop - Put (' '); - Put (Hex_Image (Unsigned_8 (Cur_Sect.Data (I)))); - end loop; - New_Line; - end if; - - if Dump_Asm then - Disassemble.Disassemble_Insn - (Cur_Sect.Data (Cur_Sect.Insn_Pc)'Address, - Unsigned_32 (Cur_Sect.Insn_Pc), - Str, Len, Insn_Len, - Get_Symbol_At_Addr'Access); - Put (HT); - Put_Line (Str (1 .. Len)); - end if; - --if Natural (Cur_Pc - Insn_Pc) /= Insn_Len then - -- raise Program_Error; - --end if; - Cur_Sect.Insn_Pc := 0; - end End_Insn; - - procedure Gen_B8 (B : Byte) is - begin - Cur_Sect.Data (Cur_Sect.Pc) := B; - Cur_Sect.Pc := Cur_Sect.Pc + 1; - end Gen_B8; - - procedure Gen_B16 (B0, B1 : Byte) is - begin - Cur_Sect.Data (Cur_Sect.Pc + 0) := B0; - Cur_Sect.Data (Cur_Sect.Pc + 1) := B1; - Cur_Sect.Pc := Cur_Sect.Pc + 2; - end Gen_B16; - - procedure Gen_Le8 (B : Unsigned_32) is - begin - Cur_Sect.Data (Cur_Sect.Pc) := Byte (B and 16#Ff#); - Cur_Sect.Pc := Cur_Sect.Pc + 1; - end Gen_Le8; - - procedure Gen_Le16 (B : Unsigned_32) is - begin - Cur_Sect.Data (Cur_Sect.Pc + 0) := Byte (Shift_Right (B, 0) and 16#Ff#); - Cur_Sect.Data (Cur_Sect.Pc + 1) := Byte (Shift_Right (B, 8) and 16#Ff#); - Cur_Sect.Pc := Cur_Sect.Pc + 2; - end Gen_Le16; - - procedure Gen_Be16 (B : Unsigned_32) is - begin - Cur_Sect.Data (Cur_Sect.Pc + 0) := Byte (Shift_Right (B, 8) and 16#Ff#); - Cur_Sect.Data (Cur_Sect.Pc + 1) := Byte (Shift_Right (B, 0) and 16#Ff#); - Cur_Sect.Pc := Cur_Sect.Pc + 2; - end Gen_Be16; - - procedure Write_B8 (Sect : Section_Acc; Pc : Pc_Type; V : Unsigned_8) is - begin - Sect.Data (Pc) := Byte (V); - end Write_B8; - - procedure Write_Be16 (Sect : Section_Acc; Pc : Pc_Type; V : Unsigned_32) is - begin - Sect.Data (Pc + 0) := Byte (Shift_Right (V, 8) and 16#Ff#); - Sect.Data (Pc + 1) := Byte (Shift_Right (V, 0) and 16#Ff#); - end Write_Be16; - - procedure Write_Le32 (Sect : Section_Acc; Pc : Pc_Type; V : Unsigned_32) is - begin - Sect.Data (Pc + 0) := Byte (Shift_Right (V, 0) and 16#Ff#); - Sect.Data (Pc + 1) := Byte (Shift_Right (V, 8) and 16#Ff#); - Sect.Data (Pc + 2) := Byte (Shift_Right (V, 16) and 16#Ff#); - Sect.Data (Pc + 3) := Byte (Shift_Right (V, 24) and 16#Ff#); - end Write_Le32; - - procedure Write_Be32 (Sect : Section_Acc; Pc : Pc_Type; V : Unsigned_32) is - begin - Sect.Data (Pc + 0) := Byte (Shift_Right (V, 24) and 16#Ff#); - Sect.Data (Pc + 1) := Byte (Shift_Right (V, 16) and 16#Ff#); - Sect.Data (Pc + 2) := Byte (Shift_Right (V, 8) and 16#Ff#); - Sect.Data (Pc + 3) := Byte (Shift_Right (V, 0) and 16#Ff#); - end Write_Be32; - - procedure Write_16 (Sect : Section_Acc; Pc : Pc_Type; B : Unsigned_32) - is - subtype B2 is Byte_Array_Base (0 .. 1); - function To_B2 is new Ada.Unchecked_Conversion - (Source => Unsigned_16, Target => B2); - begin - Sect.Data (Pc + 0 .. Pc + 1) := To_B2 (Unsigned_16 (B)); - end Write_16; - - procedure Write_32 (Sect : Section_Acc; Pc : Pc_Type; B : Unsigned_32) - is - subtype B4 is Byte_Array_Base (0 .. 3); - function To_B4 is new Ada.Unchecked_Conversion - (Source => Unsigned_32, Target => B4); - begin - Sect.Data (Pc + 0 .. Pc + 3) := To_B4 (B); - end Write_32; - - procedure Gen_16 (B : Unsigned_32) is - begin - Write_16 (Cur_Sect, Cur_Sect.Pc, B); - Cur_Sect.Pc := Cur_Sect.Pc + 2; - end Gen_16; - - procedure Gen_32 (B : Unsigned_32) is - begin - Write_32 (Cur_Sect, Cur_Sect.Pc, B); - Cur_Sect.Pc := Cur_Sect.Pc + 4; - end Gen_32; - - function Read_Le32 (Sect : Section_Acc; Pc : Pc_Type) return Unsigned_32 is - begin - return Shift_Left (Unsigned_32 (Sect.Data (Pc + 0)), 0) - or Shift_Left (Unsigned_32 (Sect.Data (Pc + 1)), 8) - or Shift_Left (Unsigned_32 (Sect.Data (Pc + 2)), 16) - or Shift_Left (Unsigned_32 (Sect.Data (Pc + 3)), 24); - end Read_Le32; - - function Read_Be32 (Sect : Section_Acc; Pc : Pc_Type) return Unsigned_32 is - begin - return Shift_Left (Unsigned_32 (Sect.Data (Pc + 0)), 24) - or Shift_Left (Unsigned_32 (Sect.Data (Pc + 1)), 16) - or Shift_Left (Unsigned_32 (Sect.Data (Pc + 2)), 8) - or Shift_Left (Unsigned_32 (Sect.Data (Pc + 3)), 0); - end Read_Be32; - - procedure Add_Le32 (Sect : Section_Acc; Pc : Pc_Type; V : Unsigned_32) is - begin - Write_Le32 (Sect, Pc, V + Read_Le32 (Sect, Pc)); - end Add_Le32; - - procedure Patch_Le32 (Pc : Pc_Type; V : Unsigned_32) is - begin - if Pc + 4 > Get_Current_Pc then - raise Program_Error; - end if; - Write_Le32 (Cur_Sect, Pc, V); - end Patch_Le32; - - procedure Patch_Be32 (Pc : Pc_Type; V : Unsigned_32) is - begin - if Pc + 4 > Get_Current_Pc then - raise Program_Error; - end if; - Write_Be32 (Cur_Sect, Pc, V); - end Patch_Be32; - - procedure Patch_Be16 (Pc : Pc_Type; V : Unsigned_32) is - begin - if Pc + 2 > Get_Current_Pc then - raise Program_Error; - end if; - Write_Be16 (Cur_Sect, Pc, V); - end Patch_Be16; - - procedure Patch_B8 (Pc : Pc_Type; V : Unsigned_8) is - begin - if Pc >= Get_Current_Pc then - raise Program_Error; - end if; - Write_B8 (Cur_Sect, Pc, V); - end Patch_B8; - - procedure Patch_32 (Pc : Pc_Type; V : Unsigned_32) is - begin - if Pc + 4 > Get_Current_Pc then - raise Program_Error; - end if; - Write_32 (Cur_Sect, Pc, V); - end Patch_32; - - procedure Gen_Le32 (B : Unsigned_32) is - begin - Write_Le32 (Cur_Sect, Cur_Sect.Pc, B); - Cur_Sect.Pc := Cur_Sect.Pc + 4; - end Gen_Le32; - - procedure Gen_Be32 (B : Unsigned_32) is - begin - Write_Be32 (Cur_Sect, Cur_Sect.Pc, B); - Cur_Sect.Pc := Cur_Sect.Pc + 4; - end Gen_Be32; - - procedure Gen_Data_Le8 (B : Unsigned_32) is - begin - if Dump_Asm then - Put_Line (HT & ".byte 0x" & Hex_Image (Unsigned_8 (B))); - end if; - Gen_Le8 (B); - end Gen_Data_Le8; - - procedure Gen_Data_Le16 (B : Unsigned_32) is - begin - if Dump_Asm then - Put_Line (HT & ".half 0x" & Hex_Image (Unsigned_16 (B))); - end if; - Gen_Le16 (B); - end Gen_Data_Le16; - - procedure Gen_Data_32 (Sym : Symbol; Offset : Integer_32) is - begin - if Dump_Asm then - if Sym = Null_Symbol then - Put_Line (HT & ".word 0x" & Hex_Image (Offset)); - else - if Offset = 0 then - Put_Line (HT & ".word " & Get_Symbol_Name (Sym)); - else - Put_Line (HT & ".word " & Get_Symbol_Name (Sym) & " + " - & Hex_Image (Offset)); - end if; - end if; - end if; - case Arch is - when Arch_X86 => - Gen_X86_32 (Sym, Offset); - when Arch_Sparc => - Gen_Sparc_32 (Sym, Offset); - when others => - raise Program_Error; - end case; - end Gen_Data_32; - - function Create_Symbol (Name : O_Ident) return Symbol - is - begin - Symbols.Append (Symbol_Type'(Section => null, - Value => 0, - Scope => Sym_Undef, - Used => False, - Name => Name, - Relocs => null, - Number => 0)); - return Symbols.Last; - end Create_Symbol; - - Last_Label : Natural := 1; - - function Create_Local_Symbol return Symbol is - begin - Symbols.Append (Symbol_Type'(Section => Cur_Sect, - Value => 0, - Scope => Sym_Local, - Used => False, - Name => O_Ident_Nul, - Relocs => null, - Number => Last_Label)); - - Last_Label := Last_Label + 1; - - return Symbols.Last; - end Create_Local_Symbol; - - function Get_Symbol_Name (Sym : Symbol) return String - is - Res : String (1 .. 10); - N : Natural; - P : Natural; - begin - if S_Local (Sym) then - N := Get_Number (Sym); - P := Res'Last; - loop - Res (P) := Character'Val ((N mod 10) + Character'Pos ('0')); - N := N / 10; - P := P - 1; - exit when N = 0; - end loop; - Res (P) := 'L'; - Res (P - 1) := '.'; - return Res (P - 1 .. Res'Last); - else - if Is_Nul (Get_Name (Sym)) then - return "ANON"; - else - return Get_String (Get_Name (Sym)); - end if; - end if; - end Get_Symbol_Name; - - function Get_Symbol_Name_Length (Sym : Symbol) return Natural - is - N : Natural; - begin - if S_Local (Sym) then - N := 10; - for I in 3 .. 8 loop - if Get_Number (Sym) < N then - return I; - end if; - N := N * 10; - end loop; - raise Program_Error; - else - return Get_String_Length (Get_Name (Sym)); - end if; - end Get_Symbol_Name_Length; - - function Get_Symbol (Name : String) return Symbol is - begin - for I in Symbols.First .. Symbols.Last loop - if Get_Symbol_Name (I) = Name then - return I; - end if; - end loop; - return Null_Symbol; - end Get_Symbol; - - function Pow_Align (V : Pc_Type; Align : Natural) return Pc_Type - is - Tmp : Pc_Type; - begin - Tmp := V + 2 ** Align - 1; - return Tmp - (Tmp mod Pc_Type (2 ** Align)); - end Pow_Align; - - procedure Gen_Pow_Align (Align : Natural) is - begin - if Align = 0 then - return; - end if; - if Dump_Asm then - Put_Line (HT & ".align" & Natural'Image (Align)); - end if; - Cur_Sect.Pc := Pow_Align (Cur_Sect.Pc, Align); - end Gen_Pow_Align; - - -- Generate LENGTH bytes set to 0. - procedure Gen_Space (Length : Integer_32) is - begin - if Dump_Asm then - Put_Line (HT & ".space" & Integer_32'Image (Length)); - end if; - Cur_Sect.Pc := Cur_Sect.Pc + Pc_Type (Length); - end Gen_Space; - - procedure Set_Symbol_Pc (Sym : Symbol; Export : Boolean) is - begin - case Get_Scope (Sym) is - when Sym_Local => - if Export then - raise Program_Error; - end if; - when Sym_Private - | Sym_Global => - raise Program_Error; - when Sym_Undef => - if Export then - Set_Scope (Sym, Sym_Global); - else - Set_Scope (Sym, Sym_Private); - end if; - end case; - -- Set value/section. - Set_Symbol_Value (Sym, Cur_Sect.Pc); - Set_Section (Sym, Cur_Sect); - - if Dump_Asm then - if Export then - Put_Line (HT & ".globl " & Get_Symbol_Name (Sym)); - end if; - Put (Get_Symbol_Name (Sym)); - Put_Line (":"); - end if; - end Set_Symbol_Pc; - - procedure Add_Reloc (Sym : Symbol; Kind : Reloc_Kind) - is - Reloc : Reloc_Acc; - begin - Reloc := new Reloc_Type'(Kind => Kind, - Done => False, - Sym_Next => Get_Relocs (Sym), - Sect_Next => null, - Addr => Cur_Sect.Pc, - Sym => Sym); - Set_Relocs (Sym, Reloc); - if Cur_Sect.First_Reloc = null then - Cur_Sect.First_Reloc := Reloc; - else - Cur_Sect.Last_Reloc.Sect_Next := Reloc; - end if; - Cur_Sect.Last_Reloc := Reloc; - Cur_Sect.Nbr_Relocs := Cur_Sect.Nbr_Relocs + 1; - end Add_Reloc; - - procedure Gen_X86_Pc32 (Sym : Symbol) - is - begin - Add_Reloc (Sym, Reloc_Pc32); - Gen_Le32 (16#ff_ff_ff_fc#); - end Gen_X86_Pc32; - - procedure Gen_Sparc_Disp22 (W : Unsigned_32; Sym : Symbol) - is - begin - Add_Reloc (Sym, Reloc_Disp22); - Gen_Be32 (W); - end Gen_Sparc_Disp22; - - procedure Gen_Sparc_Disp30 (W : Unsigned_32; Sym : Symbol) - is - begin - Add_Reloc (Sym, Reloc_Disp30); - Gen_Be32 (W); - end Gen_Sparc_Disp30; - - procedure Gen_Sparc_Hi22 (W : Unsigned_32; - Sym : Symbol; Off : Unsigned_32) - is - pragma Unreferenced (Off); - begin - Add_Reloc (Sym, Reloc_Hi22); - Gen_Be32 (W); - end Gen_Sparc_Hi22; - - procedure Gen_Sparc_Lo10 (W : Unsigned_32; - Sym : Symbol; Off : Unsigned_32) - is - pragma Unreferenced (Off); - begin - Add_Reloc (Sym, Reloc_Lo10); - Gen_Be32 (W); - end Gen_Sparc_Lo10; - - function Conv is new Ada.Unchecked_Conversion - (Source => Integer_32, Target => Unsigned_32); - - procedure Gen_X86_32 (Sym : Symbol; Offset : Integer_32) is - begin - if Sym /= Null_Symbol then - Add_Reloc (Sym, Reloc_32); - end if; - Gen_Le32 (Conv (Offset)); - end Gen_X86_32; - - procedure Gen_Sparc_32 (Sym : Symbol; Offset : Integer_32) is - begin - if Sym /= Null_Symbol then - Add_Reloc (Sym, Reloc_32); - end if; - Gen_Be32 (Conv (Offset)); - end Gen_Sparc_32; - - procedure Gen_Sparc_Ua_32 (Sym : Symbol; Offset : Integer_32) - is - pragma Unreferenced (Offset); - begin - if Sym /= Null_Symbol then - Add_Reloc (Sym, Reloc_Ua_32); - end if; - Gen_Be32 (0); - end Gen_Sparc_Ua_32; - - procedure Gen_Ua_32 (Sym : Symbol; Offset : Integer_32) is - begin - case Arch is - when Arch_X86 => - Gen_X86_32 (Sym, Offset); - when Arch_Sparc => - Gen_Sparc_Ua_32 (Sym, Offset); - when others => - raise Program_Error; - end case; - end Gen_Ua_32; - - procedure Gen_Ppc_24 (V : Unsigned_32; Sym : Symbol) - is - begin - Add_Reloc (Sym, Reloc_Ppc_Addr24); - Gen_32 (V); - end Gen_Ppc_24; - - function Get_Symbol_Vaddr (Sym : Symbol) return Pc_Type is - begin - return Get_Section (Sym).Vaddr + Get_Symbol_Value (Sym); - end Get_Symbol_Vaddr; - - procedure Write_Left_Be32 (Sect : Section_Acc; - Addr : Pc_Type; - Size : Natural; - Val : Unsigned_32) - is - W : Unsigned_32; - Mask : Unsigned_32; - begin - -- Write value. - Mask := Shift_Left (1, Size) - 1; - W := Read_Be32 (Sect, Addr); - Write_Be32 (Sect, Addr, (W and not Mask) or (Val and Mask)); - end Write_Left_Be32; - - procedure Set_Wdisp (Sect : Section_Acc; - Addr : Pc_Type; - Sym : Symbol; - Size : Natural) - is - D : Unsigned_32; - Mask : Unsigned_32; - begin - D := Unsigned_32 (Get_Symbol_Vaddr (Sym) - (Sect.Vaddr + Addr)); - -- Check overflow. - Mask := Shift_Left (1, Size + 2) - 1; - if (D and Shift_Left (1, Size + 1)) = 0 then - if (D and not Mask) /= 0 then - raise Program_Error; - end if; - else - if (D and not Mask) /= not Mask then - raise Program_Error; - end if; - end if; - -- Write value. - Write_Left_Be32 (Sect, Addr, Size, D / 4); - end Set_Wdisp; - - procedure Do_Reloc (Kind : Reloc_Kind; - Sect : Section_Acc; Addr : Pc_Type; Sym : Symbol) - is - begin - if Get_Scope (Sym) = Sym_Undef then - raise Program_Error; - end if; - - case Kind is - when Reloc_32 => - Add_Le32 (Sect, Addr, Unsigned_32 (Get_Symbol_Vaddr (Sym))); - - when Reloc_Pc32 => - Add_Le32 (Sect, Addr, - Unsigned_32 (Get_Symbol_Vaddr (Sym) - - (Sect.Vaddr + Addr))); - when Reloc_Disp22 => - Set_Wdisp (Sect, Addr, Sym, 22); - when Reloc_Disp30 => - Set_Wdisp (Sect, Addr, Sym, 30); - when Reloc_Hi22 => - Write_Left_Be32 (Sect, Addr, 22, - Unsigned_32 (Get_Symbol_Vaddr (Sym) / 1024)); - when Reloc_Lo10 => - Write_Left_Be32 (Sect, Addr, 10, - Unsigned_32 (Get_Symbol_Vaddr (Sym))); - when Reloc_Ua_32 => - Write_Be32 (Sect, Addr, Unsigned_32 (Get_Symbol_Vaddr (Sym))); - when Reloc_Ppc_Addr24 => - raise Program_Error; - end case; - end Do_Reloc; - - function Is_Reloc_Relative (Reloc : Reloc_Acc) return Boolean is - begin - case Reloc.Kind is - when Reloc_Pc32 - | Reloc_Disp22 - | Reloc_Disp30 => - return True; - when others => - return False; - end case; - end Is_Reloc_Relative; - - procedure Apply_Reloc (Sect : Section_Acc; Reloc : Reloc_Acc) is - begin - Do_Reloc (Reloc.Kind, Sect, Reloc.Addr, Reloc.Sym); - end Apply_Reloc; - - procedure Do_Intra_Section_Reloc (Sect : Section_Acc) - is - Prev : Reloc_Acc; - Rel : Reloc_Acc; - Next : Reloc_Acc; - begin - Rel := Sect.First_Reloc; - Prev := null; - while Rel /= null loop - Next := Rel.Sect_Next; - if Get_Scope (Rel.Sym) /= Sym_Undef then - Do_Reloc (Rel.Kind, Sect, Rel.Addr, Rel.Sym); - Rel.Done := True; - - if Get_Section (Rel.Sym) = Sect - and then Is_Reloc_Relative (Rel) - then - -- Remove reloc. - Sect.Nbr_Relocs := Sect.Nbr_Relocs - 1; - if Prev = null then - Sect.First_Reloc := Next; - else - Prev.Sect_Next := Next; - end if; - if Next = null then - Sect.Last_Reloc := Prev; - end if; - Free (Rel); - else - Prev := Rel; - end if; - else - Set_Used (Rel.Sym, True); - Prev := Rel; - end if; - Rel := Next; - end loop; - end Do_Intra_Section_Reloc; - - -- Return VAL rounded up to 2 ^ POW. --- function Align_Pow (Val : Integer; Pow : Natural) return Integer --- is --- N : Integer; --- Tmp : Integer; --- begin --- N := 2 ** Pow; --- Tmp := Val + N - 1; --- return Tmp - (Tmp mod N); --- end Align_Pow; - - procedure Disp_Stats is - begin - Put_Line ("Number of Symbols: " & Symbol'Image (Symbols.Last)); - end Disp_Stats; - - procedure Finish - is - Sect : Section_Acc; - Rel, N_Rel : Reloc_Acc; - begin - Symbols.Free; - Sect := Section_Chain; - while Sect /= null loop - -- Free relocs. - Rel := Sect.First_Reloc; - while Rel /= null loop - N_Rel := Rel.Sect_Next; - Free (Rel); - Rel := N_Rel; - end loop; - Sect.First_Reloc := null; - Sect.Last_Reloc := null; - - Sect := Sect.Next; - end loop; - end Finish; -end Binary_File; |