aboutsummaryrefslogtreecommitdiff
path: root/gcc/ada/libgnat
diff options
context:
space:
mode:
authorEric Botcazou <ebotcazou@adacore.com>2024-09-13 21:09:31 +0200
committerMarc Poulhiès <dkm@gcc.gnu.org>2024-10-25 11:08:59 +0200
commit18232cbcfd2c4f95bf51b4fa9fb65f545c2a0569 (patch)
tree725e7a1102ae8c21068d4300faa828872fbb7d3e /gcc/ada/libgnat
parent20441f8a30189040df995b1f4800ceed75a8c23c (diff)
downloadgcc-18232cbcfd2c4f95bf51b4fa9fb65f545c2a0569.zip
gcc-18232cbcfd2c4f95bf51b4fa9fb65f545c2a0569.tar.gz
gcc-18232cbcfd2c4f95bf51b4fa9fb65f545c2a0569.tar.bz2
ada: Fix fallout of change to 'Wide_Wide_Value for enumeration types
The literals of enumeration types are always normalized, even though they contain wide characters (but the normalization leaves these unchanged), so a normalization routine that is aware of wide characters must be run on the input string for 'Wide_Wide_Value. gcc/ada/ChangeLog: PR ada/115507 * rtsfind.ads (RE_Id): Add RE_Enum_[Wide_]Wide_String_To_String. (RE_Unit_Table): Add entries for the new values. * exp_attr.adb (Is_User_Defined_Enumeration_Type): New predicate. (Expand_N_Attribute_Reference) <Attribute_Wide_Value>: Build a call to RE_Enum_Wide_String_To_String for user-defined enumeration types. <Attribute_Wide_Wide_Value>: Likewise with RE_Enum_Wide_Wide_String_To_String. * exp_imgv.adb (Expand_Value_Attribute): Adjust to above. * libgnat/s-wchwts.ads (Enum_Wide_String_To_String): New function. (Enum_Wide_Wide_String_To_String): Likewise. * libgnat/s-wchwts.adb: Add clauses for System.Case_Util. (Normalize_String): New local procedure. (Enum_Wide_String_To_String): New function body. (Enum_Wide_Wide_String_To_String): Likewise.
Diffstat (limited to 'gcc/ada/libgnat')
-rw-r--r--gcc/ada/libgnat/s-wchwts.adb150
-rw-r--r--gcc/ada/libgnat/s-wchwts.ads12
2 files changed, 160 insertions, 2 deletions
diff --git a/gcc/ada/libgnat/s-wchwts.adb b/gcc/ada/libgnat/s-wchwts.adb
index db2fd6a..771e7dd 100644
--- a/gcc/ada/libgnat/s-wchwts.adb
+++ b/gcc/ada/libgnat/s-wchwts.adb
@@ -29,8 +29,9 @@
-- --
------------------------------------------------------------------------------
-with System.WCh_Con; use System.WCh_Con;
-with System.WCh_Cnv; use System.WCh_Cnv;
+with System.Case_Util; use System.Case_Util;
+with System.WCh_Con; use System.WCh_Con;
+with System.WCh_Cnv; use System.WCh_Cnv;
package body System.WCh_WtS is
@@ -38,6 +39,14 @@ package body System.WCh_WtS is
-- Local Subprograms --
-----------------------
+ procedure Normalize_String
+ (S : in out String;
+ EM : WC_Encoding_Method);
+ -- If S does not represent a character literal, then any lower case
+ -- characters in S are changed to their upper case counterparts, while
+ -- wide characters are unchanged. EM indicates their encoding method.
+ -- This is the wide counterpart of System.Val_Util.Normalize_String.
+
procedure Store_UTF_32_Character
(U : UTF_32_Code;
S : out String;
@@ -48,6 +57,113 @@ package body System.WCh_WtS is
-- point to the last character stored. Raises CE if character cannot be
-- stored using the given encoding method.
+ ----------------------
+ -- Normalize_String --
+ ----------------------
+
+ procedure Normalize_String
+ (S : in out String;
+ EM : WC_Encoding_Method)
+ is
+ procedure Skip_Wide (S : String; P : in out Natural);
+ -- On entry S (P) points to an ESC character for a wide character escape
+ -- sequence or an upper half character if the encoding method uses the
+ -- upper bit, or a left bracket if the brackets encoding method is in
+ -- use. On exit, P is bumped past the wide character sequence.
+
+ ---------------
+ -- Skip_Wide --
+ ---------------
+
+ procedure Skip_Wide (S : String; P : in out Natural) is
+ function Skip_Char return Character;
+ -- Function to skip one character of wide character escape sequence
+
+ ---------------
+ -- Skip_Char --
+ ---------------
+
+ function Skip_Char return Character is
+ begin
+ P := P + 1;
+ return S (P - 1);
+ end Skip_Char;
+
+ function WC_Skip is new Char_Sequence_To_UTF_32 (Skip_Char);
+
+ Discard : UTF_32_Code;
+ pragma Warnings (Off, Discard);
+
+ -- Start of processing for Skip_Wide
+
+ begin
+ -- Capture invalid wide characters errors since we are going to
+ -- discard the result anyway. We just want to move past it.
+
+ begin
+ Discard := WC_Skip (Skip_Char, EM);
+ exception
+ when Constraint_Error =>
+ null;
+ end;
+ end Skip_Wide;
+
+ F, L, Ptr : Natural;
+
+ begin
+ F := S'First;
+ L := S'Last;
+
+ -- Case of empty string
+
+ if F > L then
+ return;
+ end if;
+
+ -- Scan for leading spaces
+
+ while F < L and then S (F) = ' ' loop
+ F := F + 1;
+ end loop;
+
+ -- Case of no nonspace characters found. Decrease L to ensure L < F
+ -- without risking an overflow if F is Integer'Last.
+
+ if S (F) = ' ' then
+ L := L - 1;
+ return;
+ end if;
+
+ -- Scan for trailing spaces
+
+ while S (L) = ' ' loop
+ L := L - 1;
+ end loop;
+
+ -- Convert to upper case if S is not a character literal
+
+ if S (F) /= ''' then
+ Ptr := F;
+
+ while Ptr <= L loop
+ -- This mimics the handling of wide characters in a call to
+ -- Casing.Set_Casing (All_Upper_Case) in the compiler.
+
+ if S (Ptr) = ASCII.ESC
+ or else S (Ptr) = '['
+ or else (EM in WC_Upper_Half_Encoding_Method
+ and then Character'Pos (S (Ptr)) >= 16#80#)
+ then
+ Skip_Wide (S, Ptr);
+
+ else
+ S (Ptr) := To_Upper (S (Ptr));
+ Ptr := Ptr + 1;
+ end if;
+ end loop;
+ end if;
+ end Normalize_String;
+
----------------------------
-- Store_UTF_32_Character --
----------------------------
@@ -78,6 +194,36 @@ package body System.WCh_WtS is
Store_Chars (U, EM);
end Store_UTF_32_Character;
+ --------------------------------
+ -- Enum_Wide_String_To_String --
+ --------------------------------
+
+ function Enum_Wide_String_To_String
+ (S : Wide_String;
+ EM : WC_Encoding_Method) return String
+ is
+ Result : String := Wide_String_To_String (S, EM);
+
+ begin
+ Normalize_String (Result, EM);
+ return Result;
+ end Enum_Wide_String_To_String;
+
+ -------------------------------------
+ -- Enum_Wide_Wide_String_To_String --
+ -------------------------------------
+
+ function Enum_Wide_Wide_String_To_String
+ (S : Wide_Wide_String;
+ EM : WC_Encoding_Method) return String
+ is
+ Result : String := Wide_Wide_String_To_String (S, EM);
+
+ begin
+ Normalize_String (Result, EM);
+ return Result;
+ end Enum_Wide_Wide_String_To_String;
+
---------------------------
-- Wide_String_To_String --
---------------------------
diff --git a/gcc/ada/libgnat/s-wchwts.ads b/gcc/ada/libgnat/s-wchwts.ads
index 63ee804..549a0b6 100644
--- a/gcc/ada/libgnat/s-wchwts.ads
+++ b/gcc/ada/libgnat/s-wchwts.ads
@@ -55,9 +55,21 @@ package System.WCh_WtS is
-- for characters greater than 16#FF#. The lowest index of the returned
-- String is equal to S'First.
+ function Enum_Wide_String_To_String
+ (S : Wide_String;
+ EM : System.WCh_Con.WC_Encoding_Method) return String;
+ -- Same processing, except that the string is normalized to be usable
+ -- with the Wide_Value attribute of user-defined enumeration types.
+
function Wide_Wide_String_To_String
(S : Wide_Wide_String;
EM : System.WCh_Con.WC_Encoding_Method) return String;
-- Same processing, except for Wide_Wide_String
+ function Enum_Wide_Wide_String_To_String
+ (S : Wide_Wide_String;
+ EM : System.WCh_Con.WC_Encoding_Method) return String;
+ -- Same processing, except that the string is normalized to be usable
+ -- with the Wide_Wide_Value attribute of user-defined enumeration types.
+
end System.WCh_WtS;