On Wed, 2023-12-06 at 02:31 -0300, Alexandre Oliva wrote:
> On Nov 22, 2023, Alexandre Oliva <ol...@adacore.com> wrote:
> 
> > Ah, nice, that's a great idea, I wish I'd thought of that!  Will
> > do.
> 
> Sorry it took me so long, here it is.  I added two tests, so that,
> regardless of the defaults, we get both circumstances tested, without
> repetition.
> 
> Regstrapped on x86_64-linux-gnu.  Also tested on arm-eabi.  Ok to
> install?

Thanks for the updated patch.

Looks good to me.

Dave

> 
> 
> analyzer: deal with -fshort-enums
> 
> On platforms that enable -fshort-enums by default, various switch-
> enum
> analyzer tests fail, because apply_constraints_for_gswitch doesn't
> expect the integral promotion type cast.  I've arranged for the code
> to cope with those casts.
> 
> 
> for  gcc/analyzer/ChangeLog
> 
>         * region-model.cc (has_nondefault_case_for_value_p): Take
>         enumerate type as a parameter.
>         (region_model::apply_constraints_for_gswitch): Cope with
>         integral promotion type casts.
> 
> for  gcc/testsuite/ChangeLog
> 
>         * gcc.dg/analyzer/switch-short-enum-1.c: New.
>         * gcc.dg/analyzer/switch-no-short-enum-1.c: New.
> ---
>  gcc/analyzer/region-model.cc                       |   27 +++-
>  .../gcc.dg/analyzer/switch-no-short-enum-1.c       |  141
> ++++++++++++++++++++
>  .../gcc.dg/analyzer/switch-short-enum-1.c          |  140
> ++++++++++++++++++++
>  3 files changed, 304 insertions(+), 4 deletions(-)
>  create mode 100644 gcc/testsuite/gcc.dg/analyzer/switch-no-short-
> enum-1.c
>  create mode 100644 gcc/testsuite/gcc.dg/analyzer/switch-short-enum-
> 1.c
> 
> diff --git a/gcc/analyzer/region-model.cc b/gcc/analyzer/region-
> model.cc
> index 2157ad2578b85..6a7a8bc9f4884 100644
> --- a/gcc/analyzer/region-model.cc
> +++ b/gcc/analyzer/region-model.cc
> @@ -5387,10 +5387,10 @@ has_nondefault_case_for_value_p (const
> gswitch *switch_stmt, tree int_cst)
>     has nondefault cases handling all values in the enum.  */
>  
>  static bool
> -has_nondefault_cases_for_all_enum_values_p (const gswitch
> *switch_stmt)
> +has_nondefault_cases_for_all_enum_values_p (const gswitch
> *switch_stmt,
> +                                           tree type)
>  {
>    gcc_assert (switch_stmt);
> -  tree type = TREE_TYPE (gimple_switch_index (switch_stmt));
>    gcc_assert (TREE_CODE (type) == ENUMERAL_TYPE);
>  
>    for (tree enum_val_iter = TYPE_VALUES (type);
> @@ -5426,6 +5426,23 @@ apply_constraints_for_gswitch (const
> switch_cfg_superedge &edge,
>  {
>    tree index  = gimple_switch_index (switch_stmt);
>    const svalue *index_sval = get_rvalue (index, ctxt);
> +  bool check_index_type = true;
> +
> +  /* With -fshort-enum, there may be a type cast.  */
> +  if (ctxt && index_sval->get_kind () == SK_UNARYOP
> +      && TREE_CODE (index_sval->get_type ()) == INTEGER_TYPE)
> +    {
> +      const unaryop_svalue *unaryop = as_a <const unaryop_svalue *>
> (index_sval);
> +      if (unaryop->get_op () == NOP_EXPR
> +         && is_a <const initial_svalue *> (unaryop->get_arg ()))
> +       if (const initial_svalue *initvalop = (as_a <const
> initial_svalue *>
> +                                              (unaryop->get_arg
> ())))
> +         if (TREE_CODE (initvalop->get_type ()) == ENUMERAL_TYPE)
> +           {
> +             index_sval = initvalop;
> +             check_index_type = false;
> +           }
> +    }
>  
>    /* If we're switching based on an enum type, assume that the user
> is only
>       working with values from the enum.  Hence if this is an
> @@ -5437,12 +5454,14 @@ apply_constraints_for_gswitch (const
> switch_cfg_superedge &edge,
>        ctxt
>        /* Must be an enum value.  */
>        && index_sval->get_type ()
> -      && TREE_CODE (TREE_TYPE (index)) == ENUMERAL_TYPE
> +      && (!check_index_type
> +         || TREE_CODE (TREE_TYPE (index)) == ENUMERAL_TYPE)
>        && TREE_CODE (index_sval->get_type ()) == ENUMERAL_TYPE
>        /* If we have a constant, then we can check it directly.  */
>        && index_sval->get_kind () != SK_CONSTANT
>        && edge.implicitly_created_default_p ()
> -      && has_nondefault_cases_for_all_enum_values_p (switch_stmt)
> +      && has_nondefault_cases_for_all_enum_values_p (switch_stmt,
> +                                                    index_sval-
> >get_type ())
>        /* Don't do this if there's a chance that the index is
>          attacker-controlled.  */
>        && !ctxt->possibly_tainted_p (index_sval))
> diff --git a/gcc/testsuite/gcc.dg/analyzer/switch-no-short-enum-1.c
> b/gcc/testsuite/gcc.dg/analyzer/switch-no-short-enum-1.c
> new file mode 100644
> index 0000000000000..98f6d91f97481
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/analyzer/switch-no-short-enum-1.c
> @@ -0,0 +1,141 @@
> +/* { dg-do compile } */
> +/* { dg-additional-options "-fno-short-enums" } */
> +/* { dg-skip-if "default" { ! short_enums } } */
> +
> +#include "analyzer-decls.h"
> +
> +/* Verify the handling of "switch (enum_value)".  */
> +
> +enum e
> +{
> + E_VAL0,
> + E_VAL1,
> + E_VAL2
> +};
> +
> +/* Verify that we assume that "switch (enum)" doesn't follow
> implicit
> +   "default" if all enum values have cases  */
> +
> +int test_all_values_covered_implicit_default_1 (enum e x)
> +{
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    case E_VAL2:
> +      return 1945;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +}
> +
> +int test_all_values_covered_implicit_default_2 (enum e x)
> +{
> +  int result;
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      result = 1066;
> +      break;
> +    case E_VAL1:
> +      result = 1776;
> +      break;
> +    case E_VAL2:
> +      result = 1945;
> +      break;
> +    }
> +  return result; /* { dg-bogus "uninitialized" } */
> +}
> +
> +/* Verify that we consider paths that use the implicit default when
> not
> +   all enum values are covered by cases.  */
> +
> +int test_missing_values_implicit_default_1 (enum e x)
> +{
> +  switch (x) /* { dg-message "following 'default:' branch" } */
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    }
> +  __analyzer_dump_path (); /* { dg-message "path" } */
> +  return 0;
> +}
> +
> +int test_missing_values_implicit_default_2 (enum e x)
> +{
> +  int result;
> +  switch (x) /* { dg-message "following 'default:' branch" } */
> +    {
> +    case E_VAL0:
> +      result = 1066;
> +      break;
> +    case E_VAL1:
> +      result = 1776;
> +      break;
> +    }
> +  return result; /* { dg-warning "uninitialized" } */
> +}
> +
> +/* Verify that explicit "default" isn't rejected.  */
> +
> +int test_all_values_covered_explicit_default_1 (enum e x)
> +{
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    case E_VAL2:
> +      return 1945;
> +    default:
> +      __analyzer_dump_path (); /* { dg-message "path" } */
> +      return 0;
> +    }
> +}
> +
> +int test_missing_values_explicit_default_1 (enum e x)
> +{
> +  switch (x)
> +    {
> +    default:
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +  return 0;
> +}
> +
> +int test_missing_values_explicit_default_2 (enum e x)
> +{
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    default:
> +      __analyzer_dump_path (); /* { dg-message "path" } */
> +      return 1945;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +  return 0;
> +}
> +
> +int test_just_default (enum e x)
> +{
> +  switch (x)
> +    {
> +    default:
> +      __analyzer_dump_path (); /* { dg-message "path" } */
> +      return 42;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +  return 0;  
> +}
> +
> diff --git a/gcc/testsuite/gcc.dg/analyzer/switch-short-enum-1.c
> b/gcc/testsuite/gcc.dg/analyzer/switch-short-enum-1.c
> new file mode 100644
> index 0000000000000..384113fde5cbf
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/analyzer/switch-short-enum-1.c
> @@ -0,0 +1,140 @@
> +/* { dg-do compile } */
> +/* { dg-additional-options "-fshort-enums" } */
> +/* { dg-skip-if "default" { short_enums } } */
> +
> +#include "analyzer-decls.h"
> +
> +/* Verify the handling of "switch (enum_value)".  */
> +
> +enum e
> +{
> + E_VAL0,
> + E_VAL1,
> + E_VAL2
> +};
> +
> +/* Verify that we assume that "switch (enum)" doesn't follow
> implicit
> +   "default" if all enum values have cases  */
> +
> +int test_all_values_covered_implicit_default_1 (enum e x)
> +{
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    case E_VAL2:
> +      return 1945;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +}
> +
> +int test_all_values_covered_implicit_default_2 (enum e x)
> +{
> +  int result;
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      result = 1066;
> +      break;
> +    case E_VAL1:
> +      result = 1776;
> +      break;
> +    case E_VAL2:
> +      result = 1945;
> +      break;
> +    }
> +  return result; /* { dg-bogus "uninitialized" } */
> +}
> +
> +/* Verify that we consider paths that use the implicit default when
> not
> +   all enum values are covered by cases.  */
> +
> +int test_missing_values_implicit_default_1 (enum e x)
> +{
> +  switch (x) /* { dg-message "following 'default:' branch" } */
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    }
> +  __analyzer_dump_path (); /* { dg-message "path" } */
> +  return 0;
> +}
> +
> +int test_missing_values_implicit_default_2 (enum e x)
> +{
> +  int result;
> +  switch (x) /* { dg-message "following 'default:' branch" } */
> +    {
> +    case E_VAL0:
> +      result = 1066;
> +      break;
> +    case E_VAL1:
> +      result = 1776;
> +      break;
> +    }
> +  return result; /* { dg-warning "uninitialized" } */
> +}
> +
> +/* Verify that explicit "default" isn't rejected.  */
> +
> +int test_all_values_covered_explicit_default_1 (enum e x)
> +{
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    case E_VAL2:
> +      return 1945;
> +    default:
> +      __analyzer_dump_path (); /* { dg-message "path" } */
> +      return 0;
> +    }
> +}
> +
> +int test_missing_values_explicit_default_1 (enum e x)
> +{
> +  switch (x)
> +    {
> +    default:
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +  return 0;
> +}
> +
> +int test_missing_values_explicit_default_2 (enum e x)
> +{
> +  switch (x)
> +    {
> +    case E_VAL0:
> +      return 1066;
> +    case E_VAL1:
> +      return 1776;
> +    default:
> +      __analyzer_dump_path (); /* { dg-message "path" } */
> +      return 1945;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +  return 0;
> +}
> +
> +int test_just_default (enum e x)
> +{
> +  switch (x)
> +    {
> +    default:
> +      __analyzer_dump_path (); /* { dg-message "path" } */
> +      return 42;
> +    }
> +  __analyzer_dump_path (); /* { dg-bogus "path" } */
> +  return 0;  
> +}
> 
> 

Reply via email to