Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On Mon, Nov 25, 2019 at 03:39:32PM +0100, Jakub Jelinek wrote: > I guess the question is, shall we store the minimum precision needed > somewhere by finish_enum_value_list (perhaps only bother if the precision > of the underlying type is not the same) or compute it each time again > (which would mean for each bitfield of enum type walk the list of all the > enum values)? And, if we should store it somewhere, any preferences where? > These days C++ FE has TYPE_LANG_SPECIFIC just for classes, so options are > just use the class TYPE_LANG_SPECIFIC and just stick the min/max values in > some trees in there, restore having different lang specific variants and > store just minimum precision in there, add hash map from tree (hashed using > TYPE_UID) to precision, something else? Or yet another possibility, only compute it on demand and use a hash map as cache. Jakub
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On Fri, Oct 20, 2017 at 11:59:39AM -0400, Jason Merrill wrote: > > Still, warning when a bit-field can't hold all enumerators instead of > > all values may be a good idea. I've looked into it, and it does require > > recalculating the maximum and minimum enumerator value, since the bounds > > of the underlying type are saved in TYPE_MIN_VALUE and TYPE_MAX_VALUE > > when the enumeration is scoped, instead of the min/max enumerator value. > > > > Would adding that separate warning level be part of a separate patch, or > > should I add it to this one? > > I think making that behavior the default would be appropriate. The > current behavior for scoped enums seems like a bug; sure, all values > of the underlying type are valid, but that's also true for "unsigned i > : 4", and we don't warn about that. Note, the chosen switch name -Wbitfield-enum-conversion looks wrong to me, that is a (non-default) warning option that clang implements, but it actually is completely different warning, which GCC doesn't implement and the warning GCC has is something clang on the other side doesn't implement. So reusing the same option name for something different is bad. The warning clang++ implements is warning when storing a value with enum type into a bitfield with implicit cast, if the number of bitfield bits (the bitfield could very well be just int x : 2; or unsigned y : 3; etc.) is smaller than what is needed for the range of the enum. So, if we want to add a warning option for this warning, it should be something like -Wbitfield-enum-width or so. Agreed on that we just should only consider the range of enum values and not the underlying type for the warning, whether the warning has a warning switch or not. I think the reason why for C++ the warning is implemented by checking the precision of the underlying type is that this way was the cheapest, at least when scoped enums or enums with fixed underlying type or mode attribute isn't used, then finish_enum_value_list already computed the needed precision. While e.g. in the C FE the same warning is implemented by remembering the minimum and maximum values in type_lang_specific (although, that seems like a waste, why it doesn't just store the minimum precision instead of the two trees). finish_enum_value_list already computes almost everything even when using fixed underlying type or mode attribute, except the signop sgn = tree_int_cst_sgn (minnode) >= 0 ? UNSIGNED : SIGNED; int lowprec = tree_int_cst_min_precision (minnode, sgn); int highprec = tree_int_cst_min_precision (maxnode, sgn); int precision = MAX (lowprec, highprec); I guess the question is, shall we store the minimum precision needed somewhere by finish_enum_value_list (perhaps only bother if the precision of the underlying type is not the same) or compute it each time again (which would mean for each bitfield of enum type walk the list of all the enum values)? And, if we should store it somewhere, any preferences where? These days C++ FE has TYPE_LANG_SPECIFIC just for classes, so options are just use the class TYPE_LANG_SPECIFIC and just stick the min/max values in some trees in there, restore having different lang specific variants and store just minimum precision in there, add hash map from tree (hashed using TYPE_UID) to precision, something else? Jakub
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On Wed, Oct 18, 2017 at 3:15 PM, Sam van Kampen via gcc-patcheswrote: > On Wed, Oct 18, 2017 at 09:46:08AM -0600, Martin Sebor wrote: >> > Fair enough, I didn't know whether to change the way it currently was >> > triggered. Do you think it should fall under -Wextra (I don't think it >> > falls under -Wall, since it isn't "easy to avoid or modify to prevent >> > the warning" because it may be valid and wanted behavior), or should it >> > be enabled by no other flag? >> >> I think it depends on the implementation of the warning. With >> the current (fairly restrictive) behavior I'd say it should be >> disabled by default. But if it were to be changed to more closely >> match the Clang behavior and only warn for bit-field declarations >> that cannot represent all enumerators of the enumerated type, then >> including it in -Wall would seem helpful to me. >> >> I.e., Clang doesn't warn on this and IIUC that's what the reporter >> of the bug also expects: >> >> enum E: unsigned { E3 = 15 }; >> >> struct S { E i: 4; }; >> >> (There is value in warning on this as well, but I think most users >> will not be interested in it, so making the warning a two-level one >> where level 1 warns same as Clang and level 2 same as GCC does now >> might give us the best of both worlds). > > I see what you mean - that is the behavior I wanted to implement in the > first place, but Jonathan Wakely rightly pointed out that when an > enumeration is scoped, all values of its underlying type are valid > enumeration values, and so the bit-field you declare in 'S' _is_ too > small to hold all values of 'enum E'. > > Here's the corresponding text from draft N4659 of the C++17 standard, > §10.2/8 [dcl.enum] > > For an enumeration whose underlying type is fixed, the values of the > enumeration are the values of the underlying type. [...] It is possible > to define an enumeration that has values not defined by any of its > enumerators. > > Still, warning when a bit-field can't hold all enumerators instead of > all values may be a good idea. I've looked into it, and it does require > recalculating the maximum and minimum enumerator value, since the bounds > of the underlying type are saved in TYPE_MIN_VALUE and TYPE_MAX_VALUE > when the enumeration is scoped, instead of the min/max enumerator value. > > Would adding that separate warning level be part of a separate patch, or > should I add it to this one? I think making that behavior the default would be appropriate. The current behavior for scoped enums seems like a bug; sure, all values of the underlying type are valid, but that's also true for "unsigned i : 4", and we don't warn about that. Jason
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On 10/19/2017 12:38 PM, Eric Gallager wrote: On 10/16/17, Martin Seborwrote: On 10/16/2017 06:37 AM, Sam van Kampen via gcc-patches wrote: ..I just realised that the clang flag is -Wbitfield-enum-conversion, not -Wenum-bitfield-conversion. Please apply the patch below instead, which has replaced the two words to remove the inconsistency. 2017-10-16 Sam van Kampen * c-family/c.opt: Add a warning flag for struct bit-fields being too small to hold enumerated types. * cp/class.c: Likewise. * doc/invoke.texi: Add documentation for said warning flag. Index: gcc/c-family/c.opt === --- gcc/c-family/c.opt (revision 253784) +++ gcc/c-family/c.opt (working copy) @@ -346,6 +346,10 @@ Wframe-address C ObjC C++ ObjC++ Var(warn_frame_address) Warning LangEnabledBy(C ObjC C++ ObjC++,Wall) Warn when __builtin_frame_address or __builtin_return_address is used unsafely. +Wbitfield-enum-conversion +C++ Var(warn_bitfield_enum_conversion) Init(1) Warning +Warn about struct bit-fields being too small to hold enumerated types. Warning by default is usually reserved for problems that are most likely indicative of bugs. Does this rise to that level? (It seems that it should be in the same class as -Wconversion). The warning is already enabled by default; this patch just adds the new flag controlling it. I don't think it's worth changing it away from warning by default when it already warns by default now. The C++ warning in its present form is considered unhelpful by users (as evident from the bug report). That it warns by default is part of the problem. GCC (in C mode), Clang, and Intel ICC warn only for enums whose enumerators cannot fit in the bitfield. (In Clang the warning is in addition disabled by default and has to be explicitly enabled.) The suggestion is to add a two-level warning and adjust G++ to warn the same way at level 1 as other compilers do (and to include this level in -Wall, or if it's considered appropriate and necessary, enable it by default), and warn as it does now only at level 2 (with that level having to be explicitly requested). I would further suggest to do the same in GCC (in C mode), if only for consistency. Martin
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On 10/16/17, Martin Seborwrote: > On 10/16/2017 06:37 AM, Sam van Kampen via gcc-patches wrote: >> ..I just realised that the clang flag is -Wbitfield-enum-conversion, not >> -Wenum-bitfield-conversion. Please apply the patch below instead, which >> has replaced the two words to remove the inconsistency. >> >> 2017-10-16 Sam van Kampen >> >> * c-family/c.opt: Add a warning flag for struct bit-fields >> being too small to hold enumerated types. >> * cp/class.c: Likewise. >> * doc/invoke.texi: Add documentation for said warning flag. >> >> Index: gcc/c-family/c.opt >> === >> --- gcc/c-family/c.opt (revision 253784) >> +++ gcc/c-family/c.opt (working copy) >> @@ -346,6 +346,10 @@ Wframe-address >> C ObjC C++ ObjC++ Var(warn_frame_address) Warning LangEnabledBy(C ObjC >> C++ ObjC++,Wall) >> Warn when __builtin_frame_address or __builtin_return_address is used >> unsafely. >> >> +Wbitfield-enum-conversion >> +C++ Var(warn_bitfield_enum_conversion) Init(1) Warning >> +Warn about struct bit-fields being too small to hold enumerated types. > > Warning by default is usually reserved for problems that are > most likely indicative of bugs. Does this rise to that level? > (It seems that it should be in the same class as -Wconversion). The warning is already enabled by default; this patch just adds the new flag controlling it. I don't think it's worth changing it away from warning by default when it already warns by default now. > >> + >> Wbuiltin-declaration-mismatch >> C ObjC C++ ObjC++ Var(warn_builtin_declaraion_mismatch) Init(1) Warning >> Warn when a built-in function is declared with the wrong signature. >> Index: gcc/cp/class.c >> === >> --- gcc/cp/class.c (revision 253784) >> +++ gcc/cp/class.c (working copy) >> @@ -3278,10 +3278,11 @@ check_bitfield_decl (tree field) >> && tree_int_cst_lt (TYPE_SIZE (type), w))) >> warning_at (DECL_SOURCE_LOCATION (field), 0, >> "width of %qD exceeds its type", field); >> - else if (TREE_CODE (type) == ENUMERAL_TYPE >> + else if (warn_bitfield_enum_conversion >> + && TREE_CODE (type) == ENUMERAL_TYPE >> && (0 > (compare_tree_int >> (w, TYPE_PRECISION (ENUM_UNDERLYING_TYPE (type)) >> -warning_at (DECL_SOURCE_LOCATION (field), 0, >> +warning_at (DECL_SOURCE_LOCATION (field), >> OPT_Wbitfield_enum_conversion, >> "%qD is too small to hold all values of %q#T", >> field, type); > > I would suggest to include a bit more detail in the message, such > as the smallest number of bits needed to represent every value of > the enumeration. It's also often helpful to include a note > pointing to the relevant declaration (in this case it could be > the bit-field). > >> } >> Index: gcc/doc/invoke.texi >> === >> --- gcc/doc/invoke.texi (revision 253784) >> +++ gcc/doc/invoke.texi (working copy) >> @@ -264,7 +264,7 @@ Objective-C and Objective-C++ Dialects}. >> -Walloca -Walloca-larger-than=@var{n} @gol >> -Wno-aggressive-loop-optimizations -Warray-bounds >> -Warray-bounds=@var{n} @gol >> -Wno-attributes -Wbool-compare -Wbool-operation @gol >> --Wno-builtin-declaration-mismatch @gol >> +-Wbitfield-enum-conversion -Wno-builtin-declaration-mismatch @gol >> -Wno-builtin-macro-redefined -Wc90-c99-compat -Wc99-c11-compat @gol >> -Wc++-compat -Wc++11-compat -Wc++14-compat @gol >> -Wcast-align -Wcast-align=strict -Wcast-qual @gol >> @@ -5431,6 +5431,12 @@ Incrementing a boolean is invalid in C++17, and de >> >> This warning is enabled by @option{-Wall}. >> >> +@item -Wbitfield-enum-conversion >> +@opindex Wbitfield-enum-conversion >> +@opindex Wno-bitfield-enum-conversion >> +Warn about a bit-field potentially being too small to hold all values >> +of an enumerated type. This warning is enabled by default. >> + > > The brief description makes me wonder what this really means. What > is the meaning of "potentially?" What (what expressions) triggers > the warning? Presumably it's initialization and assignment. Does > explicit or implicit conversion also trigger is? I would suggest > to expand the documentation to obviate these questions, and perhaps > also include an example. > > Martin >
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On 10/18/2017 01:15 PM, Sam van Kampen wrote: On Wed, Oct 18, 2017 at 09:46:08AM -0600, Martin Sebor wrote: Fair enough, I didn't know whether to change the way it currently was triggered. Do you think it should fall under -Wextra (I don't think it falls under -Wall, since it isn't "easy to avoid or modify to prevent the warning" because it may be valid and wanted behavior), or should it be enabled by no other flag? I think it depends on the implementation of the warning. With the current (fairly restrictive) behavior I'd say it should be disabled by default. But if it were to be changed to more closely match the Clang behavior and only warn for bit-field declarations that cannot represent all enumerators of the enumerated type, then including it in -Wall would seem helpful to me. I.e., Clang doesn't warn on this and IIUC that's what the reporter of the bug also expects: enum E: unsigned { E3 = 15 }; struct S { E i: 4; }; (There is value in warning on this as well, but I think most users will not be interested in it, so making the warning a two-level one where level 1 warns same as Clang and level 2 same as GCC does now might give us the best of both worlds). I see what you mean - that is the behavior I wanted to implement in the first place, but Jonathan Wakely rightly pointed out that when an enumeration is scoped, all values of its underlying type are valid enumeration values, and so the bit-field you declare in 'S' _is_ too small to hold all values of 'enum E'. Here's the corresponding text from draft N4659 of the C++17 standard, §10.2/8 [dcl.enum] For an enumeration whose underlying type is fixed, the values of the enumeration are the values of the underlying type. [...] It is possible to define an enumeration that has values not defined by any of its enumerators. Still, warning when a bit-field can't hold all enumerators instead of all values may be a good idea. I've looked into it, and it does require recalculating the maximum and minimum enumerator value, since the bounds of the underlying type are saved in TYPE_MIN_VALUE and TYPE_MAX_VALUE when the enumeration is scoped, instead of the min/max enumerator value. Would adding that separate warning level be part of a separate patch, or should I add it to this one? I think it makes sense to submit the feature it in its final form, whatever you decide that will be. Martin
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On Wed, Oct 18, 2017 at 09:46:08AM -0600, Martin Sebor wrote: > > Fair enough, I didn't know whether to change the way it currently was > > triggered. Do you think it should fall under -Wextra (I don't think it > > falls under -Wall, since it isn't "easy to avoid or modify to prevent > > the warning" because it may be valid and wanted behavior), or should it > > be enabled by no other flag? > > I think it depends on the implementation of the warning. With > the current (fairly restrictive) behavior I'd say it should be > disabled by default. But if it were to be changed to more closely > match the Clang behavior and only warn for bit-field declarations > that cannot represent all enumerators of the enumerated type, then > including it in -Wall would seem helpful to me. > > I.e., Clang doesn't warn on this and IIUC that's what the reporter > of the bug also expects: > > enum E: unsigned { E3 = 15 }; > > struct S { E i: 4; }; > > (There is value in warning on this as well, but I think most users > will not be interested in it, so making the warning a two-level one > where level 1 warns same as Clang and level 2 same as GCC does now > might give us the best of both worlds). I see what you mean - that is the behavior I wanted to implement in the first place, but Jonathan Wakely rightly pointed out that when an enumeration is scoped, all values of its underlying type are valid enumeration values, and so the bit-field you declare in 'S' _is_ too small to hold all values of 'enum E'. Here's the corresponding text from draft N4659 of the C++17 standard, §10.2/8 [dcl.enum] For an enumeration whose underlying type is fixed, the values of the enumeration are the values of the underlying type. [...] It is possible to define an enumeration that has values not defined by any of its enumerators. Still, warning when a bit-field can't hold all enumerators instead of all values may be a good idea. I've looked into it, and it does require recalculating the maximum and minimum enumerator value, since the bounds of the underlying type are saved in TYPE_MIN_VALUE and TYPE_MAX_VALUE when the enumeration is scoped, instead of the min/max enumerator value. Would adding that separate warning level be part of a separate patch, or should I add it to this one?
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
Fair enough, I didn't know whether to change the way it currently was triggered. Do you think it should fall under -Wextra (I don't think it falls under -Wall, since it isn't "easy to avoid or modify to prevent the warning" because it may be valid and wanted behavior), or should it be enabled by no other flag? I think it depends on the implementation of the warning. With the current (fairly restrictive) behavior I'd say it should be disabled by default. But if it were to be changed to more closely match the Clang behavior and only warn for bit-field declarations that cannot represent all enumerators of the enumerated type, then including it in -Wall would seem helpful to me. I.e., Clang doesn't warn on this and IIUC that's what the reporter of the bug also expects: enum E: unsigned { E3 = 15 }; struct S { E i: 4; }; (There is value in warning on this as well, but I think most users will not be interested in it, so making the warning a two-level one where level 1 warns same as Clang and level 2 same as GCC does now might give us the best of both worlds). The warning points to the bit-field ('field:2' is too small to hold...), but I've added a note in an updated patch that tells the user what the precision of the enum's underlying type is. That will be useful, thanks! I've changed the description in both c.opt and invoke.texi to be more specific, it now notes that it warns when 1. An enum in a bit-field is a scoped enum... 2. ...which has an underlying type which has a higher precision than the width of the bit-field. Sounds good. Martin
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On Mon, Oct 16, 2017 at 11:31:10PM +, Joseph Myers wrote: > On Mon, 16 Oct 2017, Sam van Kampen via gcc-patches wrote: > > > +Wbitfield-enum-conversion > > +C++ Var(warn_bitfield_enum_conversion) Init(1) Warning > > +Warn about struct bit-fields being too small to hold enumerated types. > > Any option supported for C++ should also be supported for ObjC++ unless > there is a clear reason it cannot work for ObjC++. > [...] > The documentation of the option also needs to indicate that it's for > C++/ObjC++ only, similar to other such options. I've added the ObjC++ flag to the warning declaration above and I've expanded the documentation to note that it only applies to C++/ObjC++. > The patch also needs to add a testcase to the testsuite that verifies the > warnings issues in appropriate cases (and that warnings are not issued > when the bit-field is large enough). I've added such a testcase as well. I'll send the updated version of the patch, I'm just waiting to hear back from Martin on whether the flag should fall under -Wextra or fall under no flag and be disabled unless explicitly specified. > -- > Joseph S. Myers > jos...@codesourcery.com Thanks for the feedback, Sam
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On Mon, Oct 16, 2017 at 08:56:05PM -0600, Martin Sebor wrote: > On 10/16/2017 06:37 AM, Sam van Kampen via gcc-patches wrote: > > ..I just realised that the clang flag is -Wbitfield-enum-conversion, not > > -Wenum-bitfield-conversion. Please apply the patch below instead, which > > has replaced the two words to remove the inconsistency. > > > > 2017-10-16 Sam van Kampen> > > > * c-family/c.opt: Add a warning flag for struct bit-fields > > being too small to hold enumerated types. > > * cp/class.c: Likewise. > > * doc/invoke.texi: Add documentation for said warning flag. > > > > Index: gcc/c-family/c.opt > > === > > --- gcc/c-family/c.opt (revision 253784) > > +++ gcc/c-family/c.opt (working copy) > > @@ -346,6 +346,10 @@ Wframe-address > > C ObjC C++ ObjC++ Var(warn_frame_address) Warning LangEnabledBy(C ObjC C++ > > ObjC++,Wall) > > Warn when __builtin_frame_address or __builtin_return_address is used > > unsafely. > > > > +Wbitfield-enum-conversion > > +C++ Var(warn_bitfield_enum_conversion) Init(1) Warning > > +Warn about struct bit-fields being too small to hold enumerated types. > > Warning by default is usually reserved for problems that are > most likely indicative of bugs. Does this rise to that level? > (It seems that it should be in the same class as -Wconversion). > Fair enough, I didn't know whether to change the way it currently was triggered. Do you think it should fall under -Wextra (I don't think it falls under -Wall, since it isn't "easy to avoid or modify to prevent the warning" because it may be valid and wanted behavior), or should it be enabled by no other flag? > > + > > Wbuiltin-declaration-mismatch > > C ObjC C++ ObjC++ Var(warn_builtin_declaraion_mismatch) Init(1) Warning > > Warn when a built-in function is declared with the wrong signature. > > Index: gcc/cp/class.c > > === > > --- gcc/cp/class.c (revision 253784) > > +++ gcc/cp/class.c (working copy) > > @@ -3278,10 +3278,11 @@ check_bitfield_decl (tree field) > >&& tree_int_cst_lt (TYPE_SIZE (type), w))) > > warning_at (DECL_SOURCE_LOCATION (field), 0, > > "width of %qD exceeds its type", field); > > - else if (TREE_CODE (type) == ENUMERAL_TYPE > > + else if (warn_bitfield_enum_conversion > > + && TREE_CODE (type) == ENUMERAL_TYPE > >&& (0 > (compare_tree_int > > (w, TYPE_PRECISION (ENUM_UNDERLYING_TYPE (type)) > > - warning_at (DECL_SOURCE_LOCATION (field), 0, > > + warning_at (DECL_SOURCE_LOCATION (field), OPT_Wbitfield_enum_conversion, > > "%qD is too small to hold all values of %q#T", > > field, type); > > I would suggest to include a bit more detail in the message, such > as the smallest number of bits needed to represent every value of > the enumeration. It's also often helpful to include a note > pointing to the relevant declaration (in this case it could be > the bit-field). > The warning points to the bit-field ('field:2' is too small to hold...), but I've added a note in an updated patch that tells the user what the precision of the enum's underlying type is. > > } > > Index: gcc/doc/invoke.texi > > === > > --- gcc/doc/invoke.texi (revision 253784) > > +++ gcc/doc/invoke.texi (working copy) > > @@ -264,7 +264,7 @@ Objective-C and Objective-C++ Dialects}. > > -Walloca -Walloca-larger-than=@var{n} @gol > > -Wno-aggressive-loop-optimizations -Warray-bounds -Warray-bounds=@var{n} > > @gol > > -Wno-attributes -Wbool-compare -Wbool-operation @gol > > --Wno-builtin-declaration-mismatch @gol > > +-Wbitfield-enum-conversion -Wno-builtin-declaration-mismatch @gol > > -Wno-builtin-macro-redefined -Wc90-c99-compat -Wc99-c11-compat @gol > > -Wc++-compat -Wc++11-compat -Wc++14-compat @gol > > -Wcast-align -Wcast-align=strict -Wcast-qual @gol > > @@ -5431,6 +5431,12 @@ Incrementing a boolean is invalid in C++17, and de > > > > This warning is enabled by @option{-Wall}. > > > > +@item -Wbitfield-enum-conversion > > +@opindex Wbitfield-enum-conversion > > +@opindex Wno-bitfield-enum-conversion > > +Warn about a bit-field potentially being too small to hold all values > > +of an enumerated type. This warning is enabled by default. > > + > > The brief description makes me wonder what this really means. What > is the meaning of "potentially?" What (what expressions) triggers > the warning? Presumably it's initialization and assignment. Does > explicit or implicit conversion also trigger is? I would suggest > to expand the documentation to obviate these questions, and perhaps > also include an example. > I've changed the description in both c.opt and invoke.texi to be more specific, it now notes that
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On 10/16/2017 06:37 AM, Sam van Kampen via gcc-patches wrote: ..I just realised that the clang flag is -Wbitfield-enum-conversion, not -Wenum-bitfield-conversion. Please apply the patch below instead, which has replaced the two words to remove the inconsistency. 2017-10-16 Sam van Kampen* c-family/c.opt: Add a warning flag for struct bit-fields being too small to hold enumerated types. * cp/class.c: Likewise. * doc/invoke.texi: Add documentation for said warning flag. Index: gcc/c-family/c.opt === --- gcc/c-family/c.opt (revision 253784) +++ gcc/c-family/c.opt (working copy) @@ -346,6 +346,10 @@ Wframe-address C ObjC C++ ObjC++ Var(warn_frame_address) Warning LangEnabledBy(C ObjC C++ ObjC++,Wall) Warn when __builtin_frame_address or __builtin_return_address is used unsafely. +Wbitfield-enum-conversion +C++ Var(warn_bitfield_enum_conversion) Init(1) Warning +Warn about struct bit-fields being too small to hold enumerated types. Warning by default is usually reserved for problems that are most likely indicative of bugs. Does this rise to that level? (It seems that it should be in the same class as -Wconversion). + Wbuiltin-declaration-mismatch C ObjC C++ ObjC++ Var(warn_builtin_declaraion_mismatch) Init(1) Warning Warn when a built-in function is declared with the wrong signature. Index: gcc/cp/class.c === --- gcc/cp/class.c (revision 253784) +++ gcc/cp/class.c (working copy) @@ -3278,10 +3278,11 @@ check_bitfield_decl (tree field) && tree_int_cst_lt (TYPE_SIZE (type), w))) warning_at (DECL_SOURCE_LOCATION (field), 0, "width of %qD exceeds its type", field); - else if (TREE_CODE (type) == ENUMERAL_TYPE + else if (warn_bitfield_enum_conversion + && TREE_CODE (type) == ENUMERAL_TYPE && (0 > (compare_tree_int (w, TYPE_PRECISION (ENUM_UNDERLYING_TYPE (type)) - warning_at (DECL_SOURCE_LOCATION (field), 0, + warning_at (DECL_SOURCE_LOCATION (field), OPT_Wbitfield_enum_conversion, "%qD is too small to hold all values of %q#T", field, type); I would suggest to include a bit more detail in the message, such as the smallest number of bits needed to represent every value of the enumeration. It's also often helpful to include a note pointing to the relevant declaration (in this case it could be the bit-field). } Index: gcc/doc/invoke.texi === --- gcc/doc/invoke.texi (revision 253784) +++ gcc/doc/invoke.texi (working copy) @@ -264,7 +264,7 @@ Objective-C and Objective-C++ Dialects}. -Walloca -Walloca-larger-than=@var{n} @gol -Wno-aggressive-loop-optimizations -Warray-bounds -Warray-bounds=@var{n} @gol -Wno-attributes -Wbool-compare -Wbool-operation @gol --Wno-builtin-declaration-mismatch @gol +-Wbitfield-enum-conversion -Wno-builtin-declaration-mismatch @gol -Wno-builtin-macro-redefined -Wc90-c99-compat -Wc99-c11-compat @gol -Wc++-compat -Wc++11-compat -Wc++14-compat @gol -Wcast-align -Wcast-align=strict -Wcast-qual @gol @@ -5431,6 +5431,12 @@ Incrementing a boolean is invalid in C++17, and de This warning is enabled by @option{-Wall}. +@item -Wbitfield-enum-conversion +@opindex Wbitfield-enum-conversion +@opindex Wno-bitfield-enum-conversion +Warn about a bit-field potentially being too small to hold all values +of an enumerated type. This warning is enabled by default. + The brief description makes me wonder what this really means. What is the meaning of "potentially?" What (what expressions) triggers the warning? Presumably it's initialization and assignment. Does explicit or implicit conversion also trigger is? I would suggest to expand the documentation to obviate these questions, and perhaps also include an example. Martin
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
On Mon, 16 Oct 2017, Sam van Kampen via gcc-patches wrote: > +Wbitfield-enum-conversion > +C++ Var(warn_bitfield_enum_conversion) Init(1) Warning > +Warn about struct bit-fields being too small to hold enumerated types. Any option supported for C++ should also be supported for ObjC++ unless there is a clear reason it cannot work for ObjC++. > +@item -Wbitfield-enum-conversion > +@opindex Wbitfield-enum-conversion > +@opindex Wno-bitfield-enum-conversion > +Warn about a bit-field potentially being too small to hold all values > +of an enumerated type. This warning is enabled by default. The documentation of the option also needs to indicate that it's for C++/ObjC++ only, similar to other such options. The patch also needs to add a testcase to the testsuite that verifies the warnings issues in appropriate cases (and that warnings are not issued when the bit-field is large enough). -- Joseph S. Myers jos...@codesourcery.com
Re: [patch, c++] Add a warning flag for the enum bit-field declaration warning in bug #61414.
..I just realised that the clang flag is -Wbitfield-enum-conversion, not -Wenum-bitfield-conversion. Please apply the patch below instead, which has replaced the two words to remove the inconsistency. 2017-10-16 Sam van Kampen* c-family/c.opt: Add a warning flag for struct bit-fields being too small to hold enumerated types. * cp/class.c: Likewise. * doc/invoke.texi: Add documentation for said warning flag. Index: gcc/c-family/c.opt === --- gcc/c-family/c.opt (revision 253784) +++ gcc/c-family/c.opt (working copy) @@ -346,6 +346,10 @@ Wframe-address C ObjC C++ ObjC++ Var(warn_frame_address) Warning LangEnabledBy(C ObjC C++ ObjC++,Wall) Warn when __builtin_frame_address or __builtin_return_address is used unsafely. +Wbitfield-enum-conversion +C++ Var(warn_bitfield_enum_conversion) Init(1) Warning +Warn about struct bit-fields being too small to hold enumerated types. + Wbuiltin-declaration-mismatch C ObjC C++ ObjC++ Var(warn_builtin_declaraion_mismatch) Init(1) Warning Warn when a built-in function is declared with the wrong signature. Index: gcc/cp/class.c === --- gcc/cp/class.c (revision 253784) +++ gcc/cp/class.c (working copy) @@ -3278,10 +3278,11 @@ check_bitfield_decl (tree field) && tree_int_cst_lt (TYPE_SIZE (type), w))) warning_at (DECL_SOURCE_LOCATION (field), 0, "width of %qD exceeds its type", field); - else if (TREE_CODE (type) == ENUMERAL_TYPE + else if (warn_bitfield_enum_conversion + && TREE_CODE (type) == ENUMERAL_TYPE && (0 > (compare_tree_int (w, TYPE_PRECISION (ENUM_UNDERLYING_TYPE (type)) - warning_at (DECL_SOURCE_LOCATION (field), 0, + warning_at (DECL_SOURCE_LOCATION (field), OPT_Wbitfield_enum_conversion, "%qD is too small to hold all values of %q#T", field, type); } Index: gcc/doc/invoke.texi === --- gcc/doc/invoke.texi (revision 253784) +++ gcc/doc/invoke.texi (working copy) @@ -264,7 +264,7 @@ Objective-C and Objective-C++ Dialects}. -Walloca -Walloca-larger-than=@var{n} @gol -Wno-aggressive-loop-optimizations -Warray-bounds -Warray-bounds=@var{n} @gol -Wno-attributes -Wbool-compare -Wbool-operation @gol --Wno-builtin-declaration-mismatch @gol +-Wbitfield-enum-conversion -Wno-builtin-declaration-mismatch @gol -Wno-builtin-macro-redefined -Wc90-c99-compat -Wc99-c11-compat @gol -Wc++-compat -Wc++11-compat -Wc++14-compat @gol -Wcast-align -Wcast-align=strict -Wcast-qual @gol @@ -5431,6 +5431,12 @@ Incrementing a boolean is invalid in C++17, and de This warning is enabled by @option{-Wall}. +@item -Wbitfield-enum-conversion +@opindex Wbitfield-enum-conversion +@opindex Wno-bitfield-enum-conversion +Warn about a bit-field potentially being too small to hold all values +of an enumerated type. This warning is enabled by default. + @item -Wduplicated-branches @opindex Wno-duplicated-branches @opindex Wduplicated-branches