diff gcc/ada/xoscons.adb @ 111:04ced10e8804

gcc 7
author kono
date Fri, 27 Oct 2017 22:46:09 +0900
parents
children 84e7813d76e9
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/gcc/ada/xoscons.adb	Fri Oct 27 22:46:09 2017 +0900
@@ -0,0 +1,712 @@
+------------------------------------------------------------------------------
+--                                                                          --
+--                          GNAT SYSTEM UTILITIES                           --
+--                                                                          --
+--                              X O S C O N S                               --
+--                                                                          --
+--                                 B o d y                                  --
+--                                                                          --
+--          Copyright (C) 2008-2017, Free Software Foundation, Inc.         --
+--                                                                          --
+-- GNAT is free software;  you can  redistribute it  and/or modify it under --
+-- terms of the  GNU General Public License as published  by the Free Soft- --
+-- ware  Foundation;  either version 3,  or (at your option) any later ver- --
+-- sion.  GNAT is distributed in the hope that it will be useful, but WITH- --
+-- OUT 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  distributed with GNAT; see file COPYING3.  If not, go to --
+-- http://www.gnu.org/licenses for a complete copy of the license.          --
+--                                                                          --
+-- GNAT was originally developed  by the GNAT team at  New York University. --
+-- Extensive contributions were provided by Ada Core Technologies Inc.      --
+--                                                                          --
+------------------------------------------------------------------------------
+
+--  The base name of the template file is given by Argument (1). This program
+--  generates the spec for this specified unit (let's call it UNIT_NAME).
+
+--  It works in conjunction with a C template file which must be preprocessed
+--  and compiled using the cross compiler. Two input files are used:
+--    - the preprocessed C file: UNIT_NAME-tmplt.i
+--    - the generated assembly file: UNIT_NAME-tmplt.s
+
+--  The generated files are UNIT_NAME.ads and UNIT_NAME.h
+
+with Ada.Characters.Handling;    use Ada.Characters.Handling;
+with Ada.Command_Line;           use Ada.Command_Line;
+with Ada.Exceptions;             use Ada.Exceptions;
+with Ada.Streams.Stream_IO;      use Ada.Streams.Stream_IO;
+with Ada.Strings.Fixed;          use Ada.Strings.Fixed;
+with Ada.Strings.Maps;           use Ada.Strings.Maps;
+with Ada.Strings.Maps.Constants; use Ada.Strings.Maps.Constants;
+with Ada.Text_IO;                use Ada.Text_IO;
+
+pragma Warnings (Off);
+--  System.Unsigned_Types is an internal GNAT unit
+with System.Unsigned_Types;   use System.Unsigned_Types;
+pragma Warnings (On);
+
+with GNAT.OS_Lib;
+with GNAT.String_Split; use GNAT.String_Split;
+with GNAT.Table;
+
+with XUtil; use XUtil;
+
+procedure XOSCons is
+
+   use Ada.Strings;
+
+   Unit_Name : constant String := Argument (1);
+   Tmpl_Name : constant String := Unit_Name & "-tmplt";
+
+   -------------------------------------------------
+   -- Information retrieved from assembly listing --
+   -------------------------------------------------
+
+   type String_Access is access all String;
+   --  Note: we can't use GNAT.Strings for this definition, since that unit
+   --  is not available in older base compilers.
+
+   --  We need to deal with integer values that can be signed or unsigned, so
+   --  we need to accommodate the maximum range of both cases.
+
+   type Int_Value_Type is record
+      Positive  : Boolean;
+      Abs_Value : Long_Unsigned := 0;
+   end record;
+
+   function ">" (V1, V2 : Int_Value_Type) return Boolean;
+   function "<" (V1, V2 : Int_Value_Type) return Boolean;
+
+   type Asm_Info_Kind is
+     (CND,     --  Named number (decimal)
+      CNU,     --  Named number (decimal, unsigned)
+      CNS,     --  Named number (freeform text)
+      C,       --  Constant object
+      SUB,     --  Subtype
+      TXT);    --  Literal text
+   --  Recognized markers found in assembly file. These markers are produced by
+   --  the same-named macros from the C template.
+
+   subtype Asm_Int_Kind is Asm_Info_Kind range CND .. CNU;
+   --  Asm_Info_Kind values with int values in input
+
+   subtype Named_Number is Asm_Info_Kind range CND .. CNS;
+   --  Asm_Info_Kind values with named numbers in output
+
+   type Asm_Info (Kind : Asm_Info_Kind := TXT) is record
+      Line_Number   : Integer;
+      --  Line number in C source file
+
+      Constant_Name : String_Access;
+      --  Name of constant to be defined
+
+      Constant_Type : String_Access;
+      --  Type of constant (case of Kind = C)
+
+      Value_Len     : Natural := 0;
+      --  Length of text representation of constant's value
+
+      Text_Value    : String_Access;
+      --  Value for CNS / C constant
+
+      Int_Value     : Int_Value_Type;
+      --  Value for CND / CNU constant
+
+      Comment       : String_Access;
+      --  Additional descriptive comment for constant, or free-form text (TXT)
+   end record;
+
+   package Asm_Infos is new GNAT.Table
+     (Table_Component_Type => Asm_Info,
+      Table_Index_Type     => Integer,
+      Table_Low_Bound      => 1,
+      Table_Initial        => 100,
+      Table_Increment      => 10);
+
+   Max_Constant_Name_Len  : Natural := 0;
+   Max_Constant_Value_Len : Natural := 0;
+   Max_Constant_Type_Len  : Natural := 0;
+   --  Lengths of longest name and longest value
+
+   Size_Of_Unsigned_Int : Integer := 0;
+   --  Size of unsigned int on target
+
+   type Language is (Lang_Ada, Lang_C);
+
+   function Parse_Int (S : String; K : Asm_Int_Kind) return Int_Value_Type;
+   --  Parse a decimal number, preceded by an optional '$' or '#' character,
+   --  and return its value.
+
+   procedure Output_Info
+     (Lang       : Language;
+      OFile      : Sfile;
+      Info_Index : Integer);
+   --  Output information from the indicated asm info line
+
+   procedure Parse_Asm_Line (Line : String);
+   --  Parse one information line from the assembly source
+
+   function Contains_Template_Name (S : String) return Boolean;
+   --  True if S contains Tmpl_Name, possibly with different casing
+
+   function Spaces (Count : Integer) return String;
+   --  If Count is positive, return a string of Count spaces, else return
+   --  an empty string.
+
+   ---------
+   -- ">" --
+   ---------
+
+   function ">" (V1, V2 : Int_Value_Type) return Boolean is
+      P1 : Boolean renames V1.Positive;
+      P2 : Boolean renames V2.Positive;
+      A1 : Long_Unsigned renames V1.Abs_Value;
+      A2 : Long_Unsigned renames V2.Abs_Value;
+   begin
+      return (P1 and then not P2)
+        or else (P1 and then P2 and then A1 > A2)
+        or else (not P1 and then not P2 and then A1 < A2);
+   end ">";
+
+   ---------
+   -- "<" --
+   ---------
+
+   function "<" (V1, V2 : Int_Value_Type) return Boolean is
+   begin
+      return not (V1 > V2) and then not (V1 = V2);
+   end "<";
+
+   ----------------------------
+   -- Contains_Template_Name --
+   ----------------------------
+
+   function Contains_Template_Name (S : String) return Boolean is
+   begin
+      if Index (Source => To_Lower (S), Pattern => Tmpl_Name) > 0 then
+         return True;
+      else
+         return False;
+      end if;
+   end Contains_Template_Name;
+
+   -----------------
+   -- Output_Info --
+   -----------------
+
+   procedure Output_Info
+     (Lang       : Language;
+      OFile      : Sfile;
+      Info_Index : Integer)
+   is
+      Info : Asm_Info renames Asm_Infos.Table (Info_Index);
+
+      procedure Put (S : String);
+      --  Write S to OFile
+
+      ---------
+      -- Put --
+      ---------
+
+      procedure Put (S : String) is
+      begin
+         Put (OFile, S);
+      end Put;
+
+   --  Start of processing for Output_Info
+
+   begin
+      case Info.Kind is
+         when TXT =>
+
+            --  Handled in the common code for comments below
+
+            null;
+
+         when SUB =>
+            case Lang is
+               when Lang_Ada =>
+                  Put ("   subtype " & Info.Constant_Name.all
+                       & " is Interfaces.C."
+                       & Info.Text_Value.all & ";");
+               when Lang_C =>
+                  Put ("#define " & Info.Constant_Name.all & " "
+                       & Info.Text_Value.all);
+            end case;
+
+         when others =>
+
+            --  All named number cases
+
+            case Lang is
+               when Lang_Ada =>
+                  Put ("   " & Info.Constant_Name.all);
+                  Put (Spaces (Max_Constant_Name_Len
+                                 - Info.Constant_Name'Length));
+
+                  if Info.Kind in Named_Number then
+                     Put (" : constant := ");
+                  else
+                     Put (" : constant " & Info.Constant_Type.all);
+                     Put (Spaces (Max_Constant_Type_Len
+                                    - Info.Constant_Type'Length));
+                     Put (" := ");
+                  end if;
+
+               when Lang_C =>
+                  Put ("#define " & Info.Constant_Name.all & " ");
+                  Put (Spaces (Max_Constant_Name_Len
+                                 - Info.Constant_Name'Length));
+            end case;
+
+            if Info.Kind in Asm_Int_Kind then
+               if not Info.Int_Value.Positive then
+                  Put ("-");
+               end if;
+
+               Put (Trim (Info.Int_Value.Abs_Value'Img, Side => Left));
+
+            else
+               declare
+                  Is_String : constant Boolean :=
+                                Info.Kind = C
+                                  and then Info.Constant_Type.all = "String";
+
+               begin
+                  if Is_String then
+                     Put ("""");
+                  end if;
+
+                  Put (Info.Text_Value.all);
+
+                  if Is_String then
+                     Put ("""");
+                  end if;
+               end;
+            end if;
+
+            if Lang = Lang_Ada then
+               Put (";");
+
+               if Info.Comment'Length > 0 then
+                  Put (Spaces (Max_Constant_Value_Len - Info.Value_Len));
+                  Put (" --  ");
+               end if;
+            end if;
+      end case;
+
+      if Lang = Lang_Ada then
+         Put (Info.Comment.all);
+      end if;
+
+      New_Line (OFile);
+   end Output_Info;
+
+   --------------------
+   -- Parse_Asm_Line --
+   --------------------
+
+   procedure Parse_Asm_Line (Line : String) is
+      Index1, Index2 : Integer := Line'First;
+
+      function Field_Alloc return String_Access;
+      --  Allocate and return a copy of Line (Index1 .. Index2 - 1)
+
+      procedure Find_Colon (Index : in out Integer);
+      --  Increment Index until the next colon in Line
+
+      -----------------
+      -- Field_Alloc --
+      -----------------
+
+      function Field_Alloc return String_Access is
+      begin
+         return new String'(Line (Index1 .. Index2 - 1));
+      end Field_Alloc;
+
+      ----------------
+      -- Find_Colon --
+      ----------------
+
+      procedure Find_Colon (Index : in out Integer) is
+      begin
+         loop
+            Index := Index + 1;
+            exit when Index > Line'Last or else Line (Index) = ':';
+         end loop;
+      end Find_Colon;
+
+   --  Start of processing for Parse_Asm_Line
+
+   begin
+      Find_Colon (Index2);
+
+      declare
+         Info : Asm_Info (Kind => Asm_Info_Kind'Value
+                                    (Line (Line'First .. Index2 - 1)));
+      begin
+         Index1 := Index2 + 1;
+         Find_Colon (Index2);
+
+         Info.Line_Number :=
+           Integer (Parse_Int (Line (Index1 .. Index2 - 1), CNU).Abs_Value);
+
+         case Info.Kind is
+            when C
+               | CND
+               | CNS
+               | CNU
+               | SUB
+            =>
+               Index1 := Index2 + 1;
+               Find_Colon (Index2);
+
+               Info.Constant_Name := Field_Alloc;
+
+               if Info.Kind /= SUB
+                    and then
+                  Info.Constant_Name'Length > Max_Constant_Name_Len
+               then
+                  Max_Constant_Name_Len := Info.Constant_Name'Length;
+               end if;
+
+               Index1 := Index2 + 1;
+               Find_Colon (Index2);
+
+               if Info.Kind = C then
+                  Info.Constant_Type := Field_Alloc;
+
+                  if Info.Constant_Type'Length > Max_Constant_Type_Len then
+                     Max_Constant_Type_Len := Info.Constant_Type'Length;
+                  end if;
+
+                  Index1 := Index2 + 1;
+                  Find_Colon (Index2);
+               end if;
+
+               if Info.Kind = CND or else Info.Kind = CNU then
+                  Info.Int_Value :=
+                    Parse_Int (Line (Index1 .. Index2 - 1), Info.Kind);
+                  Info.Value_Len := Info.Int_Value.Abs_Value'Img'Length - 1;
+
+                  if not Info.Int_Value.Positive then
+                     Info.Value_Len := Info.Value_Len + 1;
+                  end if;
+
+               else
+                  Info.Text_Value := Field_Alloc;
+                  Info.Value_Len  := Info.Text_Value'Length;
+               end if;
+
+               if Info.Constant_Name.all = "SIZEOF_unsigned_int" then
+                  Size_Of_Unsigned_Int :=
+                    8 * Integer (Info.Int_Value.Abs_Value);
+               end if;
+
+            when others =>
+               null;
+         end case;
+
+         Index1 := Index2 + 1;
+         Index2 := Line'Last + 1;
+         Info.Comment := Field_Alloc;
+
+         if Info.Kind = TXT then
+            Info.Text_Value := Info.Comment;
+
+         --  Update Max_Constant_Value_Len, but only if this constant has a
+         --  comment (else the value is allowed to be longer).
+
+         elsif Info.Comment'Length > 0 then
+            if Info.Value_Len > Max_Constant_Value_Len then
+               Max_Constant_Value_Len := Info.Value_Len;
+            end if;
+         end if;
+
+         Asm_Infos.Append (Info);
+      end;
+
+   exception
+      when E : others =>
+         Put_Line
+           (Standard_Error, "can't parse " & Line);
+         Put_Line
+           (Standard_Error, "exception raised: " & Exception_Information (E));
+   end Parse_Asm_Line;
+
+   ----------------
+   -- Parse_Cond --
+   ----------------
+
+   procedure Parse_Cond
+     (If_Line            : String;
+      Cond               : Boolean;
+      Tmpl_File          : Ada.Text_IO.File_Type;
+      Ada_Ofile, C_Ofile : Sfile;
+      Current_Line       : in out Integer)
+   is
+      function Get_Value (Name : String) return Int_Value_Type;
+      --  Returns the value of the variable Name
+
+      ---------------
+      -- Get_Value --
+      ---------------
+
+      function Get_Value (Name : String) return Int_Value_Type is
+      begin
+         if Is_Subset (To_Set (Name), Decimal_Digit_Set) then
+            return Parse_Int (Name, CND);
+
+         else
+            for K in 1 .. Asm_Infos.Last loop
+               if Asm_Infos.Table (K).Constant_Name /= null then
+                  if Name = Asm_Infos.Table (K).Constant_Name.all then
+                     return Asm_Infos.Table (K).Int_Value;
+                  end if;
+               end if;
+            end loop;
+
+            --  Not found returns 0
+
+            return (True, 0);
+         end if;
+      end Get_Value;
+
+      --  Local variables
+
+      Sline  : Slice_Set;
+      Line   : String (1 .. 256);
+      Last   : Integer;
+      Value1 : Int_Value_Type;
+      Value2 : Int_Value_Type;
+      Res    : Boolean;
+
+   --  Start of processing for Parse_Cond
+
+   begin
+      Create (Sline, If_Line, " ");
+
+      if Slice_Count (Sline) /= 4 then
+         Put_Line (Standard_Error, "can't parse " & If_Line);
+      end if;
+
+      Value1 := Get_Value (Slice (Sline, 2));
+      Value2 := Get_Value (Slice (Sline, 4));
+
+      if Slice (Sline, 3) = ">" then
+         Res := Cond and (Value1 > Value2);
+
+      elsif Slice (Sline, 3) = "<" then
+         Res := Cond and (Value1 < Value2);
+
+      elsif Slice (Sline, 3) = "=" then
+         Res := Cond and (Value1 = Value2);
+
+      elsif Slice (Sline, 3) = "/=" then
+         Res := Cond and (Value1 /= Value2);
+
+      else
+         --  No other operator can be used
+
+         Put_Line (Standard_Error, "unknown operator in " & If_Line);
+         Res := False;
+      end if;
+
+      Current_Line := Current_Line + 1;
+
+      loop
+         Get_Line (Tmpl_File, Line, Last);
+         Current_Line := Current_Line + 1;
+         exit when Line (1 .. Last) = "@END_IF";
+
+         if Last > 4 and then Line (1 .. 4) = "@IF " then
+            Parse_Cond
+              (Line (1 .. Last), Res,
+               Tmpl_File, Ada_Ofile, C_Ofile, Current_Line);
+
+         elsif Line (1 .. Last) = "@ELSE" then
+            Res := Cond and not Res;
+
+         elsif Res then
+            Put_Line (Ada_OFile, Line (1 .. Last));
+            Put_Line (C_OFile, Line (1 .. Last));
+         end if;
+      end loop;
+   end Parse_Cond;
+
+   ---------------
+   -- Parse_Int --
+   ---------------
+
+   function Parse_Int
+     (S : String;
+      K : Asm_Int_Kind) return Int_Value_Type
+   is
+      First  : Integer := S'First;
+      Result : Int_Value_Type;
+
+   begin
+      --  On some platforms, immediate integer values are prefixed with
+      --  a $ or # character in assembly output.
+
+      if S (First) = '$' or else S (First) = '#' then
+         First := First + 1;
+      end if;
+
+      if S (First) = '-' then
+         Result.Positive := False;
+         First := First + 1;
+      else
+         Result.Positive := True;
+      end if;
+
+      Result.Abs_Value := Long_Unsigned'Value (S (First .. S'Last));
+
+      if not Result.Positive and then K = CNU then
+
+         --  Negative value, but unsigned expected: take 2's complement
+         --  reciprocical value.
+
+         Result.Abs_Value := ((not Result.Abs_Value) + 1)
+                               and
+                             (Shift_Left (1, Size_Of_Unsigned_Int) - 1);
+         Result.Positive  := True;
+      end if;
+
+      return Result;
+
+   exception
+      when others =>
+         Put_Line (Standard_Error, "can't parse decimal value: " & S);
+         raise;
+   end Parse_Int;
+
+   ------------
+   -- Spaces --
+   ------------
+
+   function Spaces (Count : Integer) return String is
+   begin
+      if Count <= 0 then
+         return "";
+      else
+         return (1 .. Count => ' ');
+      end if;
+   end Spaces;
+
+   --  Local declarations
+
+   --  Input files
+
+   Tmpl_File_Name : constant String := Tmpl_Name & ".i";
+   Asm_File_Name  : constant String := Tmpl_Name & ".s";
+
+   --  Output files
+
+   Ada_File_Name : constant String := Unit_Name & ".ads";
+   C_File_Name   : constant String := Unit_Name & ".h";
+
+   Asm_File  : Ada.Text_IO.File_Type;
+   Tmpl_File : Ada.Text_IO.File_Type;
+   Ada_OFile : Sfile;
+   C_OFile   : Sfile;
+
+   Line : String (1 .. 256);
+   Last : Integer;
+   --  Line being processed
+
+   Current_Line : Integer;
+   Current_Info : Integer;
+   In_Comment   : Boolean;
+   In_Template  : Boolean;
+
+--  Start of processing for XOSCons
+
+begin
+   --  Load values from assembly file
+
+   Open (Asm_File, In_File, Asm_File_Name);
+   while not End_Of_File (Asm_File) loop
+      Get_Line (Asm_File, Line, Last);
+      if Last > 2 and then Line (1 .. 2) = "->" then
+         Parse_Asm_Line (Line (3 .. Last));
+      end if;
+   end loop;
+
+   Close (Asm_File);
+
+   --  Load C template and output definitions
+
+   Open   (Tmpl_File, In_File,  Tmpl_File_Name);
+   Create (Ada_OFile, Out_File, Ada_File_Name);
+   Create (C_OFile,   Out_File, C_File_Name);
+
+   Current_Line := 0;
+   Current_Info := Asm_Infos.First;
+   In_Comment   := False;
+
+   while not End_Of_File (Tmpl_File) loop
+      <<Get_One_Line>>
+      Get_Line (Tmpl_File, Line, Last);
+
+      if Last >= 2 and then Line (1 .. 2) = "# " then
+         declare
+            Index : Integer;
+
+         begin
+            Index := 3;
+            while Index <= Last and then Line (Index) in '0' .. '9' loop
+               Index := Index + 1;
+            end loop;
+
+            if Contains_Template_Name (Line (Index + 1 .. Last)) then
+               Current_Line := Integer'Value (Line (3 .. Index - 1));
+               In_Template  := True;
+               goto Get_One_Line;
+            else
+               In_Template := False;
+            end if;
+         end;
+
+      elsif In_Template then
+         if In_Comment then
+            if Line (1 .. Last) = "*/" then
+               Put_Line (C_OFile, Line (1 .. Last));
+               In_Comment := False;
+
+            elsif Last > 4 and then Line (1 .. 4) = "@IF " then
+               Parse_Cond
+                 (Line (1 .. Last), True,
+                  Tmpl_File, Ada_Ofile, C_Ofile, Current_Line);
+
+            else
+               Put_Line (Ada_OFile, Line (1 .. Last));
+               Put_Line (C_OFile, Line (1 .. Last));
+            end if;
+
+         elsif Line (1 .. Last) = "/*" then
+            Put_Line (C_OFile, Line (1 .. Last));
+            In_Comment := True;
+
+         elsif Asm_Infos.Table (Current_Info).Line_Number = Current_Line then
+            if Fixed.Index (Line, "/*NOGEN*/") = 0 then
+               Output_Info (Lang_Ada, Ada_OFile, Current_Info);
+               Output_Info (Lang_C,   C_OFile,   Current_Info);
+            end if;
+
+            Current_Info := Current_Info + 1;
+         end if;
+
+         Current_Line := Current_Line + 1;
+      end if;
+   end loop;
+
+   Close (Tmpl_File);
+
+exception
+   when E : others =>
+      Put_Line ("raised " & Ada.Exceptions.Exception_Information (E));
+      GNAT.OS_Lib.OS_Exit (1);
+end XOSCons;