diff options
author | Eric Botcazou <ebotcazou@adacore.com> | 2024-09-13 21:09:31 +0200 |
---|---|---|
committer | Marc Poulhiès <dkm@gcc.gnu.org> | 2024-10-25 11:08:59 +0200 |
commit | 18232cbcfd2c4f95bf51b4fa9fb65f545c2a0569 (patch) | |
tree | 725e7a1102ae8c21068d4300faa828872fbb7d3e /gcc/ada/libgnat | |
parent | 20441f8a30189040df995b1f4800ceed75a8c23c (diff) | |
download | gcc-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.adb | 150 | ||||
-rw-r--r-- | gcc/ada/libgnat/s-wchwts.ads | 12 |
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; |