aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTristan Gingold <tgingold@free.fr>2015-09-13 16:02:28 +0200
committerTristan Gingold <tgingold@free.fr>2015-09-13 16:02:28 +0200
commit357e381d7dff9279157186afd6a6c19124a804f5 (patch)
tree7b957b17e16a908bc0415bc0f35d65965fa176de
parentbdc8e297b8cc2c36cf9e9ce41007cade06d2ee3f (diff)
downloadghdl-357e381d7dff9279157186afd6a6c19124a804f5.tar.gz
ghdl-357e381d7dff9279157186afd6a6c19124a804f5.tar.bz2
ghdl-357e381d7dff9279157186afd6a6c19124a804f5.zip
Add mach-O binary file writer.
-rw-r--r--src/ortho/mcode/binary_file-macho.adb345
-rw-r--r--src/ortho/mcode/binary_file-macho.ads23
-rw-r--r--src/ortho/mcode/macho.ads103
3 files changed, 471 insertions, 0 deletions
diff --git a/src/ortho/mcode/binary_file-macho.adb b/src/ortho/mcode/binary_file-macho.adb
new file mode 100644
index 000000000..dbfc8825d
--- /dev/null
+++ b/src/ortho/mcode/binary_file-macho.adb
@@ -0,0 +1,345 @@
+-- Binary file Mach-O writer.
+-- Copyright (C) 2015 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 Macho; use Macho;
+
+package body Binary_File.Macho is
+ procedure Write (Fd : GNAT.OS_Lib.File_Descriptor)
+ is
+ use GNAT.OS_Lib;
+
+ -- If true, discard local symbols;
+ Flag_Discard_Local : Boolean := True;
+
+ procedure Xwrite (Data : System.Address; Len : Natural) is
+ begin
+ if Write (Fd, Data, Len) /= Len then
+ raise Write_Error;
+ end if;
+ end Xwrite;
+
+ function Symbol_Discarded (S : Symbol) return Boolean is
+ begin
+ case Get_Scope (S) is
+ when Sym_Local =>
+ if Flag_Discard_Local then
+ return True;
+ end if;
+ when Sym_Private =>
+ null;
+ when Sym_Global =>
+ null;
+ when Sym_Undef =>
+ if not Get_Used (S) then
+ return True;
+ end if;
+ end case;
+ return False;
+ end Symbol_Discarded;
+
+ procedure Fill_Name (Dest : out String; Src : String)
+ is
+ subtype D_Type is String (1 .. Dest'Length);
+ D : D_Type renames Dest;
+ subtype S_Type is String (1 .. Src'Length);
+ S : S_Type renames Src;
+ begin
+ if S'Length < D'Length then
+ D (1 .. S'Length) := S;
+ D (S'Length + 1 .. D'Last) := (others => ASCII.NUL);
+ else
+ D := S (1 .. D'Last);
+ end if;
+ end Fill_Name;
+
+ type Section_Info_Type is record
+ Sect : Section_Acc;
+ -- Index of the section symbol (in symtab).
+ end record;
+ type Section_Info_Array is array (Natural range <>) of Section_Info_Type;
+ Sects_Info : Section_Info_Array (1 .. Nbr_Sections);
+ type Section_32_Array is array (Natural range <>) of Section_32;
+ Sects_Hdr : Section_32_Array (1 .. Nbr_Sections);
+ Nbr_Sect : Natural;
+ Sect : Section_Acc;
+
+ -- Various offsets.
+ File_Offset : Natural;
+ Seg_Offset : Natural;
+ Symtab_Offset : Natural;
+ Strtab_Offset : Natural;
+ Sizeof_Cmds : Natural;
+
+ -- Number of symtab entries.
+ Nbr_Symbols : Natural;
+
+ Str_Size : Natural;
+
+ -- If true, do local relocs.
+ Flag_Reloc : constant Boolean := True;
+ begin
+ -- If relocations are not performs, then local symbols cannot be
+ -- discarded.
+ if not Flag_Reloc then
+ Flag_Discard_Local := False;
+ end if;
+
+ -- Count sections.
+ Sect := Section_Chain;
+ Nbr_Sect := 0;
+ while Sect /= null loop
+ Nbr_Sect := Nbr_Sect + 1;
+ Sects_Info (Nbr_Sect).Sect := Sect;
+ Sect.Number := Nbr_Sect;
+ Sect := Sect.Next;
+ end loop;
+
+ -- Set sections offset.
+ Sizeof_Cmds := Lc_Size + Segment_Command_32_Size
+ + Nbr_Sect * Section_32_Size
+ + Lc_Size + Symtab_Command_Size;
+ File_Offset := Header_32_Size + Sizeof_Cmds;
+ Seg_Offset := File_Offset;
+ for I in 1 .. Nbr_Sect loop
+ Sect := Sects_Info (I).Sect;
+ if Sect.Data /= null then
+ -- FIXME: alignment ?
+ Sects_Hdr (I).Offset := Unsigned_32 (File_Offset);
+ File_Offset := File_Offset + Natural (Sect.Pc);
+ else
+ Sects_Hdr (I).Offset := 0;
+ end if;
+ end loop;
+
+ -- Relocs
+ -- FIXME: todo.
+
+ Symtab_Offset := File_Offset;
+ Str_Size := 0;
+ Nbr_Symbols := 0;
+ for I in Symbols.First .. Symbols.Last loop
+ if not Symbol_Discarded (I) then
+ Nbr_Symbols := Nbr_Symbols + 1;
+ Set_Number (I, Nbr_Symbols);
+ Str_Size := Str_Size + Get_Symbol_Name_Length (I) + 1;
+ else
+ Set_Number (I, 0);
+ end if;
+ end loop;
+
+ File_Offset := File_Offset + Nbr_Symbols * Nlist_32_Size;
+ Strtab_Offset := File_Offset;
+
+ -- Write file header.
+ declare
+ Hdr : Header_32;
+ begin
+ Hdr := (Magic => Magic,
+ Cputype => Cputype_I386,
+ Cpusubtype => Cpusubtype_I386_All,
+ Filetype => Mh_Object,
+ Ncmds => 2,
+ Sizeofcmds => Unsigned_32 (Sizeof_Cmds),
+ Flags => 0);
+ Xwrite (Hdr'Address, Header_32_Size);
+ end;
+
+ -- Write segment and section commands.
+ declare
+ Lc : Load_Command;
+ Seg : Segment_Command_32;
+ begin
+ Lc := (Cmd => Lc_Segment_32,
+ Cmdsize => Unsigned_32 (Lc_Size + Segment_Command_32_Size
+ + Nbr_Sect * Section_32_Size));
+ Xwrite (Lc'Address, Lc_Size);
+ Seg := (Segname => (others => ASCII.NUL),
+ Vmaddr => 0,
+ Vmsize => 0, -- FIXME
+ Fileoff => Unsigned_32 (Seg_Offset),
+ Filesize => Unsigned_32 (Symtab_Offset - Seg_Offset),
+ Maxprot => 7, -- rwx
+ Initprot => 7,
+ Nsects => Unsigned_32 (Nbr_Sect),
+ Flags => 0);
+ Xwrite (Seg'Address, Segment_Command_32_Size);
+ end;
+
+ -- Write section headers.
+ for I in 1 .. Nbr_Sect loop
+ Sect := Sects_Info (I).Sect;
+ declare
+ Hdr : Section_32 renames Sects_Hdr (I);
+ Secname_Raw : constant String := Sect.Name.all;
+ subtype S_Type is String (1 .. Secname_Raw'Length);
+ Secname : S_Type renames Secname_Raw;
+ begin
+ if Secname = ".text" then
+ Fill_Name (Hdr.Sectname, "__text");
+ Fill_Name (Hdr.Segname, "__TEXT");
+ elsif Secname = ".rodata" then
+ Fill_Name (Hdr.Sectname, "__const");
+ Fill_Name (Hdr.Segname, "__TEXT");
+ elsif (Sect.Flags and Section_Debug) /= 0 then
+ if Secname'Length > 7
+ and then Secname (1 .. 7) = ".debug_"
+ then
+ Fill_Name (Hdr.Sectname,
+ "__debug_" & Secname (8 .. Secname'Last));
+ else
+ Fill_Name (Hdr.Sectname, Sect.Name.all);
+ end if;
+ Fill_Name (Hdr.Segname, "__DWARF");
+ else
+ Fill_Name (Hdr.Sectname, Secname);
+ Fill_Name (Hdr.Segname, "");
+ end if;
+ Hdr.Addr := Unsigned_32 (Sect.Vaddr);
+ Hdr.Size := Unsigned_32 (Sect.Pc);
+ Hdr.Align := Unsigned_32 (Sect.Align);
+ Hdr.Reloff := 0;
+ Hdr.Nreloc := 0;
+ Hdr.Flags := 0;
+ Hdr.Reserved1 := 0;
+ Hdr.Reserved2 := 0;
+ Xwrite (Hdr'Address, Section_32_Size);
+ end;
+ end loop;
+
+ -- Write symtab command
+ declare
+ Lc : Load_Command;
+ Symtab : Symtab_Command;
+ begin
+ Lc := (Cmd => Lc_Symtab,
+ Cmdsize => Unsigned_32 (Lc_Size + Symtab_Command_Size));
+ Xwrite (Lc'Address, Lc_Size);
+ Symtab := (Symoff => Unsigned_32 (Symtab_Offset),
+ Nsyms => Unsigned_32 (Nbr_Symbols),
+ Stroff => Unsigned_32 (Strtab_Offset),
+ Strsize => Unsigned_32 (Str_Size));
+ Xwrite (Symtab'Address, Symtab_Command_Size);
+ end;
+
+ -- Write sections content.
+ for I in 1 .. Nbr_Sect loop
+ Sect := Sects_Info (I).Sect;
+ if Sect.Data /= null then
+ Xwrite (Sect.Data (0)'Address, Natural (Sect.Pc));
+ end if;
+ end loop;
+
+ -- FIXME: write relocs.
+
+ -- Write symbols.
+ declare
+ Str_Offset : Natural;
+
+ generic
+ with procedure Handle (S : Symbol);
+ procedure Foreach_Symbol;
+
+ procedure Foreach_Symbol is
+ begin
+ -- First, the local and private symbols.
+ for I in Symbols.First .. Symbols.Last loop
+ case Get_Scope (I) is
+ when Sym_Local =>
+ if not Flag_Discard_Local then
+ Handle (I);
+ end if;
+ when Sym_Private =>
+ Handle (I);
+ when Sym_Global
+ | Sym_Undef =>
+ null;
+ end case;
+ end loop;
+
+ -- Then global symbols
+ for I in Symbols.First .. Symbols.Last loop
+ case Get_Scope (I) is
+ when Sym_Local
+ | Sym_Private =>
+ null;
+ when Sym_Global =>
+ Handle (I);
+ when Sym_Undef =>
+ null;
+ end case;
+ end loop;
+ -- Then undef symbols.
+ for I in Symbols.First .. Symbols.Last loop
+ case Get_Scope (I) is
+ when Sym_Local
+ | Sym_Private =>
+ null;
+ when Sym_Global =>
+ null;
+ when Sym_Undef =>
+ if Get_Used (I) then
+ Handle (I);
+ end if;
+ end case;
+ end loop;
+ end Foreach_Symbol;
+
+ procedure Write_Symbol (S : Symbol)
+ is
+ Sym : Nlist_32;
+ begin
+ Sym := (N_Strx => Unsigned_32 (Str_Offset),
+ N_Type => 0,
+ N_Sect => 0,
+ N_Desc => 0,
+ N_Value => Unsigned_32 (Get_Symbol_Value (S)));
+ Str_Offset := Str_Offset + Get_Symbol_Name_Length (S) + 1;
+ if Get_Scope (S) = Sym_Undef then
+ Sym.N_Type := N_Undf;
+ else
+ if Get_Scope (S) = Sym_Global then
+ Sym.N_Type := N_Sect + N_Ext;
+ else
+ Sym.N_Type := N_Sect;
+ end if;
+ Sym.N_Sect := Unsigned_8 (Get_Section (S).Number);
+ Sym.N_Value :=
+ Sym.N_Value + Unsigned_32 (Get_Section (S).Vaddr);
+ end if;
+ Xwrite (Sym'Address, Nlist_32_Size);
+ end Write_Symbol;
+
+ procedure Write_String (Sym : Symbol)
+ is
+ Str : constant String := Get_Symbol_Name (Sym) & ASCII.NUL;
+ begin
+ Xwrite (Str'Address, Str'Length);
+ end Write_String;
+
+ procedure Write_All_Symbols is new
+ Foreach_Symbol (Write_Symbol);
+ procedure Write_All_Strings is new
+ Foreach_Symbol (Write_String);
+ begin
+ Str_Offset := 0;
+
+ Write_All_Symbols;
+ Write_All_Strings;
+ end;
+ end Write;
+
+end Binary_File.Macho;
diff --git a/src/ortho/mcode/binary_file-macho.ads b/src/ortho/mcode/binary_file-macho.ads
new file mode 100644
index 000000000..404327c9a
--- /dev/null
+++ b/src/ortho/mcode/binary_file-macho.ads
@@ -0,0 +1,23 @@
+-- Binary file Mach-O writer.
+-- Copyright (C) 2015 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 GNAT.OS_Lib;
+
+package Binary_File.Macho is
+ procedure Write (Fd : GNAT.OS_Lib.File_Descriptor);
+end Binary_File.Macho;
+
diff --git a/src/ortho/mcode/macho.ads b/src/ortho/mcode/macho.ads
new file mode 100644
index 000000000..e080a430f
--- /dev/null
+++ b/src/ortho/mcode/macho.ads
@@ -0,0 +1,103 @@
+-- Macho definitions.
+-- Copyright (C) 2015 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 Interfaces; use Interfaces;
+with System; use System;
+
+package Macho is
+ type Header_32 is record
+ Magic : Unsigned_32;
+ Cputype : Unsigned_32;
+ Cpusubtype : Unsigned_32;
+ Filetype : Unsigned_32;
+ Ncmds : Unsigned_32;
+ Sizeofcmds : Unsigned_32;
+ Flags : Unsigned_32;
+ end record;
+
+ -- Size of Filehdr.
+ Header_32_Size : constant Natural := Header_32'Size / Storage_Unit;
+
+ -- Magic numbers.
+ Magic : constant Unsigned_32 := 16#feed_face#;
+ Cputype_I386 : constant Unsigned_32 := 7;
+ Cpusubtype_I386_All : constant Unsigned_32 := 3;
+
+ Mh_Object : constant Unsigned_32 := 1;
+ Mh_Execute : constant Unsigned_32 := 2;
+
+ -- Load commands.
+ type Load_Command is record
+ Cmd : Unsigned_32;
+ Cmdsize : Unsigned_32;
+ end record;
+ Lc_Size : constant Natural := Load_Command'Size / Storage_Unit;
+
+ Lc_Segment_32 : constant Unsigned_32 := 1;
+ type Segment_Command_32 is record
+ Segname : String (1 .. 16);
+ Vmaddr : Unsigned_32;
+ Vmsize : Unsigned_32;
+ Fileoff : Unsigned_32;
+ Filesize : Unsigned_32;
+ Maxprot : Unsigned_32;
+ Initprot : Unsigned_32;
+ Nsects : Unsigned_32;
+ Flags : Unsigned_32;
+ end record;
+ Segment_Command_32_Size : constant Natural :=
+ Segment_Command_32'Size / Storage_Unit;
+
+ type Section_32 is record
+ Sectname : String (1 .. 16);
+ Segname : String (1 .. 16);
+ Addr : Unsigned_32;
+ Size : Unsigned_32;
+ Offset : Unsigned_32;
+ Align : Unsigned_32;
+ Reloff : Unsigned_32;
+ Nreloc : Unsigned_32;
+ Flags : Unsigned_32;
+ Reserved1 : Unsigned_32;
+ Reserved2 : Unsigned_32;
+ end record;
+ Section_32_Size : constant Natural := Section_32'Size / Storage_Unit;
+
+ Lc_Symtab : constant Unsigned_32 := 2;
+ type Symtab_Command is record
+ Symoff : Unsigned_32;
+ Nsyms : Unsigned_32;
+ Stroff : Unsigned_32;
+ Strsize : Unsigned_32;
+ end record;
+ Symtab_Command_Size : constant Natural :=
+ Symtab_Command'Size / Storage_Unit;
+
+ type Nlist_32 is record
+ N_Strx : Unsigned_32;
+ N_Type : Unsigned_8;
+ N_Sect : Unsigned_8;
+ N_Desc : Unsigned_16;
+ N_Value : Unsigned_32;
+ end record;
+
+ Nlist_32_Size : constant Natural := Nlist_32'Size / Storage_Unit;
+
+ N_Undf : constant Unsigned_8 := 16#00#;
+ N_Ext : constant Unsigned_8 := 16#01#;
+ N_Sect : constant Unsigned_8 := 16#0e#;
+end Macho;