------------------------------------------------------------------------------ -- -- -- GNAT COMPILER COMPONENTS -- -- -- -- S T R I N G T -- -- -- -- B o d y -- -- -- -- Copyright (C) 1992-2021, 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. -- -- -- ------------------------------------------------------------------------------ with Alloc; with Output; use Output; with Table; package body Stringt is -- The following table stores the sequence of character codes for the -- stored string constants. The entries are referenced from the -- separate Strings table. package String_Chars is new Table.Table ( Table_Component_Type => Char_Code, Table_Index_Type => Int, Table_Low_Bound => 0, Table_Initial => Alloc.String_Chars_Initial, Table_Increment => Alloc.String_Chars_Increment, Table_Name => "String_Chars"); -- The String_Id values reference entries in the Strings table, which -- contains String_Entry records that record the length of each stored -- string and its starting location in the String_Chars table. type String_Entry is record String_Index : Int; Length : Nat; end record; package Strings is new Table.Table ( Table_Component_Type => String_Entry, Table_Index_Type => String_Id'Base, Table_Low_Bound => First_String_Id, Table_Initial => Alloc.Strings_Initial, Table_Increment => Alloc.Strings_Increment, Table_Name => "Strings"); -- Note: it is possible that two entries in the Strings table can share -- string data in the String_Chars table, and in particular this happens -- when Start_String is called with a parameter that is the last string -- currently allocated in the table. Strings_Last : String_Id := First_String_Id; String_Chars_Last : Int := 0; -- Strings_Last and String_Chars_Last are used by procedure Mark and -- Release to get a snapshot of the tables and to restore them to their -- previous situation. ------------ -- Append -- ------------ procedure Append (Buf : in out Bounded_String; S : String_Id) is begin for X in 1 .. String_Length (S) loop Append (Buf, Get_Character (Get_String_Char (S, X))); end loop; end Append; ---------------- -- End_String -- ---------------- function End_String return String_Id is begin return Strings.Last; end End_String; --------------------- -- Get_String_Char -- --------------------- function Get_String_Char (Id : String_Id; Index : Int) return Char_Code is begin pragma Assert (Id in First_String_Id .. Strings.Last and then Index in 1 .. Strings.Table (Id).Length); return String_Chars.Table (Strings.Table (Id).String_Index + Index - 1); end Get_String_Char; ---------------- -- Initialize -- ---------------- procedure Initialize is begin String_Chars.Init; Strings.Init; -- Set up the null string Start_String; Null_String_Id := End_String; end Initialize; ---------- -- Lock -- ---------- procedure Lock is begin String_Chars.Release; String_Chars.Locked := True; Strings.Release; Strings.Locked := True; end Lock; ---------- -- Mark -- ---------- procedure Mark is begin Strings_Last := Strings.Last; String_Chars_Last := String_Chars.Last; end Mark; ------------- -- Release -- ------------- procedure Release is begin Strings.Set_Last (Strings_Last); String_Chars.Set_Last (String_Chars_Last); end Release; ------------------ -- Start_String -- ------------------ -- Version to start completely new string procedure Start_String is begin Strings.Append ((String_Index => String_Chars.Last + 1, Length => 0)); end Start_String; -- Version to start from initially stored string procedure Start_String (S : String_Id) is begin Strings.Increment_Last; -- Case of initial string value is at the end of the string characters -- table, so it does not need copying, instead it can be shared. if Strings.Table (S).String_Index + Strings.Table (S).Length = String_Chars.Last + 1 then Strings.Table (Strings.Last).String_Index := Strings.Table (S).String_Index; -- Case of initial string value must be copied to new string else Strings.Table (Strings.Last).String_Index := String_Chars.Last + 1; for J in 1 .. Strings.Table (S).Length loop String_Chars.Append (String_Chars.Table (Strings.Table (S).String_Index + (J - 1))); end loop; end if; -- In either case the result string length is copied from the argument Strings.Table (Strings.Last).Length := Strings.Table (S).Length; end Start_String; ----------------------- -- Store_String_Char -- ----------------------- procedure Store_String_Char (C : Char_Code) is begin String_Chars.Append (C); Strings.Table (Strings.Last).Length := Strings.Table (Strings.Last).Length + 1; end Store_String_Char; procedure Store_String_Char (C : Character) is begin Store_String_Char (Get_Char_Code (C)); end Store_String_Char; ------------------------ -- Store_String_Chars -- ------------------------ procedure Store_String_Chars (S : String) is begin for J in S'First .. S'Last loop Store_String_Char (Get_Char_Code (S (J))); end loop; end Store_String_Chars; procedure Store_String_Chars (S : String_Id) is -- We are essentially doing this: -- for J in 1 .. String_Length (S) loop -- Store_String_Char (Get_String_Char (S, J)); -- end loop; -- but when the string is long it's more efficient to grow the -- String_Chars table all at once. S_First : constant Int := Strings.Table (S).String_Index; S_Len : constant Nat := String_Length (S); Old_Last : constant Int := String_Chars.Last; New_Last : constant Int := Old_Last + S_Len; begin String_Chars.Set_Last (New_Last); String_Chars.Table (Old_Last + 1 .. New_Last) := String_Chars.Table (S_First .. S_First + S_Len - 1); Strings.Table (Strings.Last).Length := Strings.Table (Strings.Last).Length + S_Len; end Store_String_Chars; ---------------------- -- Store_String_Int -- ---------------------- procedure Store_String_Int (N : Int) is begin if N < 0 then Store_String_Char ('-'); Store_String_Int (-N); else if N > 9 then Store_String_Int (N / 10); end if; Store_String_Char (Character'Val (Character'Pos ('0') + N mod 10)); end if; end Store_String_Int; -------------------------- -- String_Chars_Address -- -------------------------- function String_Chars_Address return System.Address is begin return String_Chars.Table (0)'Address; end String_Chars_Address; ------------------ -- String_Equal -- ------------------ function String_Equal (L, R : String_Id) return Boolean is Len : constant Nat := Strings.Table (L).Length; begin if Len /= Strings.Table (R).Length then return False; else for J in 1 .. Len loop if Get_String_Char (L, J) /= Get_String_Char (R, J) then return False; end if; end loop; return True; end if; end String_Equal; ----------------------------- -- String_From_Name_Buffer -- ----------------------------- function String_From_Name_Buffer (Buf : Bounded_String := Global_Name_Buffer) return String_Id is begin Start_String; Store_String_Chars (+Buf); return End_String; end String_From_Name_Buffer; ------------------- -- String_Length -- ------------------- function String_Length (Id : String_Id) return Nat is begin return Strings.Table (Id).Length; end String_Length; -------------------- -- String_To_Name -- -------------------- function String_To_Name (S : String_Id) return Name_Id is Buf : Bounded_String; begin Append (Buf, S); return Name_Find (Buf); end String_To_Name; --------------------------- -- String_To_Name_Buffer -- --------------------------- procedure String_To_Name_Buffer (S : String_Id) is begin Name_Len := 0; Append (Global_Name_Buffer, S); end String_To_Name_Buffer; --------------------- -- Strings_Address -- --------------------- function Strings_Address return System.Address is begin return Strings.Table (First_String_Id)'Address; end Strings_Address; --------------- -- To_String -- --------------- function To_String (S : String_Id) return String is Buf : Bounded_String; begin Append (Buf, S); return To_String (Buf); end To_String; ------------ -- Unlock -- ------------ procedure Unlock is begin String_Chars.Locked := False; Strings.Locked := False; end Unlock; ------------------------- -- Unstore_String_Char -- ------------------------- procedure Unstore_String_Char is begin String_Chars.Decrement_Last; Strings.Table (Strings.Last).Length := Strings.Table (Strings.Last).Length - 1; end Unstore_String_Char; --------------------- -- Write_Char_Code -- --------------------- procedure Write_Char_Code (Code : Char_Code) is procedure Write_Hex_Byte (J : Char_Code); -- Write single hex byte (value in range 0 .. 255) as two digits -------------------- -- Write_Hex_Byte -- -------------------- procedure Write_Hex_Byte (J : Char_Code) is Hexd : constant array (Char_Code range 0 .. 15) of Character := "0123456789abcdef"; begin Write_Char (Hexd (J / 16)); Write_Char (Hexd (J mod 16)); end Write_Hex_Byte; -- Start of processing for Write_Char_Code begin if Code in 16#20# .. 16#7E# then Write_Char (Character'Val (Code)); else Write_Char ('['); Write_Char ('"'); if Code > 16#FF_FFFF# then Write_Hex_Byte (Code / 2 ** 24); end if; if Code > 16#FFFF# then Write_Hex_Byte ((Code / 2 ** 16) mod 256); end if; if Code > 16#FF# then Write_Hex_Byte ((Code / 256) mod 256); end if; Write_Hex_Byte (Code mod 256); Write_Char ('"'); Write_Char (']'); end if; end Write_Char_Code; ------------------------------ -- Write_String_Table_Entry -- ------------------------------ procedure Write_String_Table_Entry (Id : String_Id) is C : Char_Code; begin if Id = No_String then Write_Str ("no string"); else Write_Char ('"'); for J in 1 .. String_Length (Id) loop C := Get_String_Char (Id, J); if C = Character'Pos ('"') then Write_Str (""""""); else Write_Char_Code (C); end if; -- If string is very long, quit if J >= 1000 then -- arbitrary limit Write_Str ("""...etc (length = "); Write_Int (String_Length (Id)); Write_Str (")"); return; end if; end loop; Write_Char ('"'); end if; end Write_String_Table_Entry; end Stringt;