This patch factors out the code to calculate the number of iterations required and to generate the iteration loop into separate functions from gimplify_omp_depend for reuse later.

I have also replaced the 'TREE_CODE (*tp) == TREE_LIST && ...' checks used for detecting an iterator clause with a macro OMP_ITERATOR_DECL_P, as it needs to be done frequently.
From 0439fce03c2b5fb2802eaf65831e28f548ca074b Mon Sep 17 00:00:00 2001
From: Kwok Cheung Yeung <kcye...@baylibre.com>
Date: Tue, 12 Mar 2024 20:51:38 +0000
Subject: [PATCH 1/3] openmp: Refactor handling of iterators

Move code to calculate the iteration size and to generate the iterator
expansion loop into separate functions.

Use OMP_ITERATOR_DECL_P to check for iterators in clause declarations.

2024-05-24  Kwok Cheung Yeung  <kcye...@baylibre.com>

        gcc/c-family/
        * c-omp.cc (c_finish_omp_depobj): Use OMP_ITERATOR_DECL_P.

        gcc/c/
        * c-typeck.cc (handle_omp_array_sections): Use OMP_ITERATOR_DECL_P.
        (c_finish_omp_clauses): Likewise.

        gcc/cp/
        * pt.cc (tsubst_omp_clause_decl): Use OMP_ITERATOR_DECL_P.
        * semantics.cc (handle_omp_array_sections): Likewise.
        (finish_omp_clauses): Likewise.

        gcc/
        * gimplify.cc (gimplify_omp_affinity): Use OMP_ITERATOR_DECL_P.
        (compute_iterator_count): New.
        (build_iterator_loop): New.
        (gimplify_omp_depend): Use OMP_ITERATOR_DECL_P, compute_iterator_count
        and build_iterator_loop.
        * tree-inline.cc (copy_tree_body_r): Use OMP_ITERATOR_DECL_P.
        * tree-pretty-print.cc (dump_omp_clause): Likewise.
        * tree.h (OMP_ITERATOR_DECL_P): New macro.
---
 gcc/c-family/c-omp.cc    |   4 +-
 gcc/c/c-typeck.cc        |  13 +-
 gcc/cp/pt.cc             |   4 +-
 gcc/cp/semantics.cc      |   8 +-
 gcc/gimplify.cc          | 326 +++++++++++++++++++--------------------
 gcc/tree-inline.cc       |   5 +-
 gcc/tree-pretty-print.cc |   8 +-
 gcc/tree.h               |   6 +
 8 files changed, 175 insertions(+), 199 deletions(-)

diff --git a/gcc/c-family/c-omp.cc b/gcc/c-family/c-omp.cc
index c0e02aa422f..b56e49da62c 100644
--- a/gcc/c-family/c-omp.cc
+++ b/gcc/c-family/c-omp.cc
@@ -744,9 +744,7 @@ c_finish_omp_depobj (location_t loc, tree depobj,
          kind = OMP_CLAUSE_DEPEND_KIND (clause);
          t = OMP_CLAUSE_DECL (clause);
          gcc_assert (t);
-         if (TREE_CODE (t) == TREE_LIST
-             && TREE_PURPOSE (t)
-             && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+         if (OMP_ITERATOR_DECL_P (t))
            {
              error_at (OMP_CLAUSE_LOCATION (clause),
                        "%<iterator%> modifier may not be specified on "
diff --git a/gcc/c/c-typeck.cc b/gcc/c/c-typeck.cc
index 7ecca9f58c6..b0fe80cf224 100644
--- a/gcc/c/c-typeck.cc
+++ b/gcc/c/c-typeck.cc
@@ -14218,9 +14218,7 @@ handle_omp_array_sections (tree &c, enum 
c_omp_region_type ort)
   tree *tp = &OMP_CLAUSE_DECL (c);
   if ((OMP_CLAUSE_CODE (c) == OMP_CLAUSE_DEPEND
        || OMP_CLAUSE_CODE (c) == OMP_CLAUSE_AFFINITY)
-      && TREE_CODE (*tp) == TREE_LIST
-      && TREE_PURPOSE (*tp)
-      && TREE_CODE (TREE_PURPOSE (*tp)) == TREE_VEC)
+      && OMP_ITERATOR_DECL_P (*tp))
     tp = &TREE_VALUE (*tp);
   tree first = handle_omp_array_sections_1 (c, *tp, types,
                                            maybe_zero_len, first_non_one,
@@ -15409,9 +15407,7 @@ c_finish_omp_clauses (tree clauses, enum 
c_omp_region_type ort)
        case OMP_CLAUSE_DEPEND:
        case OMP_CLAUSE_AFFINITY:
          t = OMP_CLAUSE_DECL (c);
-         if (TREE_CODE (t) == TREE_LIST
-             && TREE_PURPOSE (t)
-             && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+         if (OMP_ITERATOR_DECL_P (t))
            {
              if (TREE_PURPOSE (t) != last_iterators)
                last_iterators_remove
@@ -15511,10 +15507,7 @@ c_finish_omp_clauses (tree clauses, enum 
c_omp_region_type ort)
                      break;
                    }
                }
-             if (TREE_CODE (OMP_CLAUSE_DECL (c)) == TREE_LIST
-                 && TREE_PURPOSE (OMP_CLAUSE_DECL (c))
-                 && (TREE_CODE (TREE_PURPOSE (OMP_CLAUSE_DECL (c)))
-                     == TREE_VEC))
+             if (OMP_ITERATOR_DECL_P (OMP_CLAUSE_DECL (c)))
                TREE_VALUE (OMP_CLAUSE_DECL (c)) = t;
              else
                OMP_CLAUSE_DECL (c) = t;
diff --git a/gcc/cp/pt.cc b/gcc/cp/pt.cc
index e77c48e463e..26db4f6e0cf 100644
--- a/gcc/cp/pt.cc
+++ b/gcc/cp/pt.cc
@@ -17520,9 +17520,7 @@ tsubst_omp_clause_decl (tree decl, tree args, 
tsubst_flags_t complain,
     return decl;
 
   /* Handle OpenMP iterators.  */
-  if (TREE_CODE (decl) == TREE_LIST
-      && TREE_PURPOSE (decl)
-      && TREE_CODE (TREE_PURPOSE (decl)) == TREE_VEC)
+  if (OMP_ITERATOR_DECL_P (decl))
     {
       tree ret;
       if (iterator_cache[0] == TREE_PURPOSE (decl))
diff --git a/gcc/cp/semantics.cc b/gcc/cp/semantics.cc
index f90c304a65b..a48b3d2fcc5 100644
--- a/gcc/cp/semantics.cc
+++ b/gcc/cp/semantics.cc
@@ -5884,9 +5884,7 @@ handle_omp_array_sections (tree &c, enum 
c_omp_region_type ort)
   tree *tp = &OMP_CLAUSE_DECL (c);
   if ((OMP_CLAUSE_CODE (c) == OMP_CLAUSE_DEPEND
        || OMP_CLAUSE_CODE (c) == OMP_CLAUSE_AFFINITY)
-      && TREE_CODE (*tp) == TREE_LIST
-      && TREE_PURPOSE (*tp)
-      && TREE_CODE (TREE_PURPOSE (*tp)) == TREE_VEC)
+      && OMP_ITERATOR_DECL_P (*tp))
     tp = &TREE_VALUE (*tp);
   tree first = handle_omp_array_sections_1 (c, *tp, types,
                                            maybe_zero_len, first_non_one,
@@ -8191,9 +8189,7 @@ finish_omp_clauses (tree clauses, enum c_omp_region_type 
ort)
        case OMP_CLAUSE_DEPEND:
        case OMP_CLAUSE_AFFINITY:
          t = OMP_CLAUSE_DECL (c);
-         if (TREE_CODE (t) == TREE_LIST
-             && TREE_PURPOSE (t)
-             && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+         if (OMP_ITERATOR_DECL_P (t))
            {
              if (TREE_PURPOSE (t) != last_iterators)
                last_iterators_remove
diff --git a/gcc/gimplify.cc b/gcc/gimplify.cc
index b0ed58ed0f9..cb7358640f0 100644
--- a/gcc/gimplify.cc
+++ b/gcc/gimplify.cc
@@ -8640,9 +8640,7 @@ gimplify_omp_affinity (tree *list_p, gimple_seq *pre_p)
     if (OMP_CLAUSE_CODE (c) == OMP_CLAUSE_AFFINITY)
       {
        tree t = OMP_CLAUSE_DECL (c);
-       if (TREE_CODE (t) == TREE_LIST
-                   && TREE_PURPOSE (t)
-                   && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+       if (OMP_ITERATOR_DECL_P (t))
          {
            if (TREE_VALUE (t) == null_pointer_node)
              continue;
@@ -8747,6 +8745,159 @@ gimplify_omp_affinity (tree *list_p, gimple_seq *pre_p)
   return;
 }
 
+/* Returns a tree expression containing the total iteration count of the
+   iterator clause decl T.  */
+
+static tree
+compute_iterator_count (tree t, gimple_seq *pre_p)
+{
+  tree tcnt = size_one_node;
+  for (tree it = TREE_PURPOSE (t); it; it = TREE_CHAIN (it))
+    {
+      if (gimplify_expr (&TREE_VEC_ELT (it, 1), pre_p, NULL,
+                        is_gimple_val, fb_rvalue) == GS_ERROR
+         || gimplify_expr (&TREE_VEC_ELT (it, 2), pre_p, NULL,
+                           is_gimple_val, fb_rvalue) == GS_ERROR
+         || gimplify_expr (&TREE_VEC_ELT (it, 3), pre_p, NULL,
+                           is_gimple_val, fb_rvalue) == GS_ERROR
+         || (gimplify_expr (&TREE_VEC_ELT (it, 4), pre_p, NULL,
+                            is_gimple_val, fb_rvalue) == GS_ERROR))
+       return NULL_TREE;
+      tree var = TREE_VEC_ELT (it, 0);
+      tree begin = TREE_VEC_ELT (it, 1);
+      tree end = TREE_VEC_ELT (it, 2);
+      tree step = TREE_VEC_ELT (it, 3);
+      tree orig_step = TREE_VEC_ELT (it, 4);
+      tree type = TREE_TYPE (var);
+      tree stype = TREE_TYPE (step);
+      location_t loc = DECL_SOURCE_LOCATION (var);
+      tree endmbegin;
+      /* Compute count for this iterator as
+        orig_step > 0
+        ? (begin < end ? (end - begin + (step - 1)) / step : 0)
+        : (begin > end ? (end - begin + (step + 1)) / step : 0)
+        and compute product of those for the entire clause.  */
+      if (POINTER_TYPE_P (type))
+       endmbegin = fold_build2_loc (loc, POINTER_DIFF_EXPR, stype, end, begin);
+      else
+       endmbegin = fold_build2_loc (loc, MINUS_EXPR, type, end, begin);
+      tree stepm1 = fold_build2_loc (loc, MINUS_EXPR, stype, step,
+                                    build_int_cst (stype, 1));
+      tree stepp1 = fold_build2_loc (loc, PLUS_EXPR, stype, step,
+                                    build_int_cst (stype, 1));
+      tree pos = fold_build2_loc (loc, PLUS_EXPR, stype,
+                                 unshare_expr (endmbegin), stepm1);
+      pos = fold_build2_loc (loc, TRUNC_DIV_EXPR, stype, pos, step);
+      tree neg = fold_build2_loc (loc, PLUS_EXPR, stype, endmbegin, stepp1);
+      if (TYPE_UNSIGNED (stype))
+       {
+         neg = fold_build1_loc (loc, NEGATE_EXPR, stype, neg);
+         step = fold_build1_loc (loc, NEGATE_EXPR, stype, step);
+       }
+      neg = fold_build2_loc (loc, TRUNC_DIV_EXPR, stype, neg, step);
+      step = NULL_TREE;
+      tree cond = fold_build2_loc (loc, LT_EXPR, boolean_type_node, begin, 
end);
+      pos = fold_build3_loc (loc, COND_EXPR, stype, cond, pos,
+                            build_int_cst (stype, 0));
+      cond = fold_build2_loc (loc, LT_EXPR, boolean_type_node, end, begin);
+      neg = fold_build3_loc (loc, COND_EXPR, stype, cond, neg,
+                            build_int_cst (stype, 0));
+      tree osteptype = TREE_TYPE (orig_step);
+      cond = fold_build2_loc (loc, GT_EXPR, boolean_type_node, orig_step,
+                             build_int_cst (osteptype, 0));
+      tree cnt = fold_build3_loc (loc, COND_EXPR, stype, cond, pos, neg);
+      cnt = fold_convert_loc (loc, sizetype, cnt);
+      if (gimplify_expr (&cnt, pre_p, NULL, is_gimple_val,
+                        fb_rvalue) == GS_ERROR)
+       return NULL_TREE;
+      tcnt = size_binop_loc (loc, MULT_EXPR, tcnt, cnt);
+    }
+  if (gimplify_expr (&tcnt, pre_p, NULL, is_gimple_val, fb_rvalue) == GS_ERROR)
+    return NULL_TREE;
+
+  return tcnt;
+}
+
+/* Build loops iterating over the space defined by the iterators in clause C.
+   Returns a pointer to the BIND_EXPR_BODY in the innermost loop body.
+   LAST_BIND is set to point to the BIND_EXPR containing the whole loop.  */
+
+static tree *
+build_iterator_loop (tree c, gimple_seq *pre_p, tree *last_bind)
+{
+  tree t = OMP_CLAUSE_DECL (c);
+  gcc_assert (OMP_ITERATOR_DECL_P (t));
+
+  if (*last_bind)
+    gimplify_and_add (*last_bind, pre_p);
+  tree block = TREE_VEC_ELT (TREE_PURPOSE (t), 5);
+  *last_bind = build3 (BIND_EXPR, void_type_node,
+                      BLOCK_VARS (block), NULL, block);
+  TREE_SIDE_EFFECTS (*last_bind) = 1;
+  SET_EXPR_LOCATION (*last_bind, OMP_CLAUSE_LOCATION (c));
+  tree *p = &BIND_EXPR_BODY (*last_bind);
+  for (tree it = TREE_PURPOSE (t); it; it = TREE_CHAIN (it))
+    {
+      tree var = TREE_VEC_ELT (it, 0);
+      tree begin = TREE_VEC_ELT (it, 1);
+      tree end = TREE_VEC_ELT (it, 2);
+      tree step = TREE_VEC_ELT (it, 3);
+      tree orig_step = TREE_VEC_ELT (it, 4);
+      tree type = TREE_TYPE (var);
+      location_t loc = DECL_SOURCE_LOCATION (var);
+      /* Emit:
+        var = begin;
+        goto cond_label;
+        beg_label:
+        ...
+        var = var + step;
+        cond_label:
+        if (orig_step > 0) {
+          if (var < end) goto beg_label;
+        } else {
+          if (var > end) goto beg_label;
+        }
+        for each iterator, with inner iterators added to
+        the ... above.  */
+      tree beg_label = create_artificial_label (loc);
+      tree cond_label = NULL_TREE;
+      tree tem = build2_loc (loc, MODIFY_EXPR, void_type_node, var, begin);
+      append_to_statement_list_force (tem, p);
+      tem = build_and_jump (&cond_label);
+      append_to_statement_list_force (tem, p);
+      tem = build1 (LABEL_EXPR, void_type_node, beg_label);
+      append_to_statement_list (tem, p);
+      tree bind = build3 (BIND_EXPR, void_type_node, NULL_TREE,
+                         NULL_TREE, NULL_TREE);
+      TREE_SIDE_EFFECTS (bind) = 1;
+      SET_EXPR_LOCATION (bind, loc);
+      append_to_statement_list_force (bind, p);
+      if (POINTER_TYPE_P (type))
+       tem = build2_loc (loc, POINTER_PLUS_EXPR, type,
+                         var, fold_convert_loc (loc, sizetype, step));
+      else
+       tem = build2_loc (loc, PLUS_EXPR, type, var, step);
+      tem = build2_loc (loc, MODIFY_EXPR, void_type_node, var, tem);
+      append_to_statement_list_force (tem, p);
+      tem = build1 (LABEL_EXPR, void_type_node, cond_label);
+      append_to_statement_list (tem, p);
+      tree cond = fold_build2_loc (loc, LT_EXPR, boolean_type_node, var, end);
+      tree pos = fold_build3_loc (loc, COND_EXPR, void_type_node, cond,
+                                 build_and_jump (&beg_label), void_node);
+      cond = fold_build2_loc (loc, GT_EXPR, boolean_type_node, var, end);
+      tree neg = fold_build3_loc (loc, COND_EXPR, void_type_node, cond,
+                                 build_and_jump (&beg_label), void_node);
+      tree osteptype = TREE_TYPE (orig_step);
+      cond = fold_build2_loc (loc, GT_EXPR, boolean_type_node, orig_step,
+                             build_int_cst (osteptype, 0));
+      tem = fold_build3_loc (loc, COND_EXPR, void_type_node, cond, pos, neg);
+      append_to_statement_list_force (tem, p);
+      p = &BIND_EXPR_BODY (bind);
+    }
+
+  return p;
+}
+
 /* If *LIST_P contains any OpenMP depend clauses with iterators,
    lower all the depend clauses by populating corresponding depend
    array.  Returns 0 if there are no such depend clauses, or
@@ -8791,89 +8942,12 @@ gimplify_omp_depend (tree *list_p, gimple_seq *pre_p)
        tree t = OMP_CLAUSE_DECL (c);
        if (first_loc == UNKNOWN_LOCATION)
          first_loc = OMP_CLAUSE_LOCATION (c);
-       if (TREE_CODE (t) == TREE_LIST
-           && TREE_PURPOSE (t)
-           && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+       if (OMP_ITERATOR_DECL_P (t))
          {
            if (TREE_PURPOSE (t) != last_iter)
              {
-               tree tcnt = size_one_node;
-               for (tree it = TREE_PURPOSE (t); it; it = TREE_CHAIN (it))
-                 {
-                   if (gimplify_expr (&TREE_VEC_ELT (it, 1), pre_p, NULL,
-                                      is_gimple_val, fb_rvalue) == GS_ERROR
-                       || gimplify_expr (&TREE_VEC_ELT (it, 2), pre_p, NULL,
-                                         is_gimple_val, fb_rvalue) == GS_ERROR
-                       || gimplify_expr (&TREE_VEC_ELT (it, 3), pre_p, NULL,
-                                         is_gimple_val, fb_rvalue) == GS_ERROR
-                       || (gimplify_expr (&TREE_VEC_ELT (it, 4), pre_p, NULL,
-                                          is_gimple_val, fb_rvalue)
-                           == GS_ERROR))
-                     return 2;
-                   tree var = TREE_VEC_ELT (it, 0);
-                   tree begin = TREE_VEC_ELT (it, 1);
-                   tree end = TREE_VEC_ELT (it, 2);
-                   tree step = TREE_VEC_ELT (it, 3);
-                   tree orig_step = TREE_VEC_ELT (it, 4);
-                   tree type = TREE_TYPE (var);
-                   tree stype = TREE_TYPE (step);
-                   location_t loc = DECL_SOURCE_LOCATION (var);
-                   tree endmbegin;
-                   /* Compute count for this iterator as
-                      orig_step > 0
-                      ? (begin < end ? (end - begin + (step - 1)) / step : 0)
-                      : (begin > end ? (end - begin + (step + 1)) / step : 0)
-                      and compute product of those for the entire depend
-                      clause.  */
-                   if (POINTER_TYPE_P (type))
-                     endmbegin = fold_build2_loc (loc, POINTER_DIFF_EXPR,
-                                                  stype, end, begin);
-                   else
-                     endmbegin = fold_build2_loc (loc, MINUS_EXPR, type,
-                                                  end, begin);
-                   tree stepm1 = fold_build2_loc (loc, MINUS_EXPR, stype,
-                                                  step,
-                                                  build_int_cst (stype, 1));
-                   tree stepp1 = fold_build2_loc (loc, PLUS_EXPR, stype, step,
-                                                  build_int_cst (stype, 1));
-                   tree pos = fold_build2_loc (loc, PLUS_EXPR, stype,
-                                               unshare_expr (endmbegin),
-                                               stepm1);
-                   pos = fold_build2_loc (loc, TRUNC_DIV_EXPR, stype,
-                                          pos, step);
-                   tree neg = fold_build2_loc (loc, PLUS_EXPR, stype,
-                                               endmbegin, stepp1);
-                   if (TYPE_UNSIGNED (stype))
-                     {
-                       neg = fold_build1_loc (loc, NEGATE_EXPR, stype, neg);
-                       step = fold_build1_loc (loc, NEGATE_EXPR, stype, step);
-                     }
-                   neg = fold_build2_loc (loc, TRUNC_DIV_EXPR, stype,
-                                          neg, step);
-                   step = NULL_TREE;
-                   tree cond = fold_build2_loc (loc, LT_EXPR,
-                                                boolean_type_node,
-                                                begin, end);
-                   pos = fold_build3_loc (loc, COND_EXPR, stype, cond, pos,
-                                          build_int_cst (stype, 0));
-                   cond = fold_build2_loc (loc, LT_EXPR, boolean_type_node,
-                                           end, begin);
-                   neg = fold_build3_loc (loc, COND_EXPR, stype, cond, neg,
-                                          build_int_cst (stype, 0));
-                   tree osteptype = TREE_TYPE (orig_step);
-                   cond = fold_build2_loc (loc, GT_EXPR, boolean_type_node,
-                                           orig_step,
-                                           build_int_cst (osteptype, 0));
-                   tree cnt = fold_build3_loc (loc, COND_EXPR, stype,
-                                               cond, pos, neg);
-                   cnt = fold_convert_loc (loc, sizetype, cnt);
-                   if (gimplify_expr (&cnt, pre_p, NULL, is_gimple_val,
-                                      fb_rvalue) == GS_ERROR)
-                     return 2;
-                   tcnt = size_binop_loc (loc, MULT_EXPR, tcnt, cnt);
-                 }
-               if (gimplify_expr (&tcnt, pre_p, NULL, is_gimple_val,
-                                  fb_rvalue) == GS_ERROR)
+               tree tcnt = compute_iterator_count (t, pre_p);
+               if (!tcnt)
                  return 2;
                last_iter = TREE_PURPOSE (t);
                last_count = tcnt;
@@ -9027,92 +9101,10 @@ gimplify_omp_depend (tree *list_p, gimple_seq *pre_p)
            gcc_unreachable ();
          }
        tree t = OMP_CLAUSE_DECL (c);
-       if (TREE_CODE (t) == TREE_LIST
-           && TREE_PURPOSE (t)
-           && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+       if (OMP_ITERATOR_DECL_P (t))
          {
            if (TREE_PURPOSE (t) != last_iter)
-             {
-               if (last_bind)
-                 gimplify_and_add (last_bind, pre_p);
-               tree block = TREE_VEC_ELT (TREE_PURPOSE (t), 5);
-               last_bind = build3 (BIND_EXPR, void_type_node,
-                                   BLOCK_VARS (block), NULL, block);
-               TREE_SIDE_EFFECTS (last_bind) = 1;
-               SET_EXPR_LOCATION (last_bind, OMP_CLAUSE_LOCATION (c));
-               tree *p = &BIND_EXPR_BODY (last_bind);
-               for (tree it = TREE_PURPOSE (t); it; it = TREE_CHAIN (it))
-                 {
-                   tree var = TREE_VEC_ELT (it, 0);
-                   tree begin = TREE_VEC_ELT (it, 1);
-                   tree end = TREE_VEC_ELT (it, 2);
-                   tree step = TREE_VEC_ELT (it, 3);
-                   tree orig_step = TREE_VEC_ELT (it, 4);
-                   tree type = TREE_TYPE (var);
-                   location_t loc = DECL_SOURCE_LOCATION (var);
-                   /* Emit:
-                      var = begin;
-                      goto cond_label;
-                      beg_label:
-                      ...
-                      var = var + step;
-                      cond_label:
-                      if (orig_step > 0) {
-                        if (var < end) goto beg_label;
-                      } else {
-                        if (var > end) goto beg_label;
-                      }
-                      for each iterator, with inner iterators added to
-                      the ... above.  */
-                   tree beg_label = create_artificial_label (loc);
-                   tree cond_label = NULL_TREE;
-                   tem = build2_loc (loc, MODIFY_EXPR, void_type_node,
-                                     var, begin);
-                   append_to_statement_list_force (tem, p);
-                   tem = build_and_jump (&cond_label);
-                   append_to_statement_list_force (tem, p);
-                   tem = build1 (LABEL_EXPR, void_type_node, beg_label);
-                   append_to_statement_list (tem, p);
-                   tree bind = build3 (BIND_EXPR, void_type_node, NULL_TREE,
-                                       NULL_TREE, NULL_TREE);
-                   TREE_SIDE_EFFECTS (bind) = 1;
-                   SET_EXPR_LOCATION (bind, loc);
-                   append_to_statement_list_force (bind, p);
-                   if (POINTER_TYPE_P (type))
-                     tem = build2_loc (loc, POINTER_PLUS_EXPR, type,
-                                       var, fold_convert_loc (loc, sizetype,
-                                                              step));
-                   else
-                     tem = build2_loc (loc, PLUS_EXPR, type, var, step);
-                   tem = build2_loc (loc, MODIFY_EXPR, void_type_node,
-                                     var, tem);
-                   append_to_statement_list_force (tem, p);
-                   tem = build1 (LABEL_EXPR, void_type_node, cond_label);
-                   append_to_statement_list (tem, p);
-                   tree cond = fold_build2_loc (loc, LT_EXPR,
-                                                boolean_type_node,
-                                                var, end);
-                   tree pos
-                     = fold_build3_loc (loc, COND_EXPR, void_type_node,
-                                        cond, build_and_jump (&beg_label),
-                                        void_node);
-                   cond = fold_build2_loc (loc, GT_EXPR, boolean_type_node,
-                                           var, end);
-                   tree neg
-                     = fold_build3_loc (loc, COND_EXPR, void_type_node,
-                                        cond, build_and_jump (&beg_label),
-                                        void_node);
-                   tree osteptype = TREE_TYPE (orig_step);
-                   cond = fold_build2_loc (loc, GT_EXPR, boolean_type_node,
-                                           orig_step,
-                                           build_int_cst (osteptype, 0));
-                   tem = fold_build3_loc (loc, COND_EXPR, void_type_node,
-                                          cond, pos, neg);
-                   append_to_statement_list_force (tem, p);
-                   p = &BIND_EXPR_BODY (bind);
-                 }
-               last_body = p;
-             }
+             last_body = build_iterator_loop (c, pre_p, &last_bind);
            last_iter = TREE_PURPOSE (t);
            if (TREE_CODE (TREE_VALUE (t)) == COMPOUND_EXPR)
              {
diff --git a/gcc/tree-inline.cc b/gcc/tree-inline.cc
index f31a34ac410..05dea9473a0 100644
--- a/gcc/tree-inline.cc
+++ b/gcc/tree-inline.cc
@@ -1453,10 +1453,7 @@ copy_tree_body_r (tree *tp, int *walk_subtrees, void 
*data)
                   || OMP_CLAUSE_CODE (*tp) == OMP_CLAUSE_DEPEND))
        {
          tree t = OMP_CLAUSE_DECL (*tp);
-         if (t
-             && TREE_CODE (t) == TREE_LIST
-             && TREE_PURPOSE (t)
-             && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+         if (t && OMP_ITERATOR_DECL_P (t))
            {
              *walk_subtrees = 0;
              OMP_CLAUSE_DECL (*tp) = copy_node (t);
diff --git a/gcc/tree-pretty-print.cc b/gcc/tree-pretty-print.cc
index f9ad8562078..011f44bfd3d 100644
--- a/gcc/tree-pretty-print.cc
+++ b/gcc/tree-pretty-print.cc
@@ -805,9 +805,7 @@ dump_omp_clause (pretty_printer *pp, tree clause, int spc, 
dump_flags_t flags)
       pp_string (pp, "affinity(");
       {
        tree t = OMP_CLAUSE_DECL (clause);
-       if (TREE_CODE (t) == TREE_LIST
-           && TREE_PURPOSE (t)
-           && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+       if (OMP_ITERATOR_DECL_P (t))
          {
            dump_omp_iterators (pp, TREE_PURPOSE (t), spc, flags);
            pp_colon (pp);
@@ -847,9 +845,7 @@ dump_omp_clause (pretty_printer *pp, tree clause, int spc, 
dump_flags_t flags)
        }
       {
        tree t = OMP_CLAUSE_DECL (clause);
-       if (TREE_CODE (t) == TREE_LIST
-           && TREE_PURPOSE (t)
-           && TREE_CODE (TREE_PURPOSE (t)) == TREE_VEC)
+       if (OMP_ITERATOR_DECL_P (t))
          {
            dump_omp_iterators (pp, TREE_PURPOSE (t), spc, flags);
            pp_colon (pp);
diff --git a/gcc/tree.h b/gcc/tree.h
index ee2aae332a4..e8568a69f95 100644
--- a/gcc/tree.h
+++ b/gcc/tree.h
@@ -2147,6 +2147,12 @@ class auto_suppress_location_wrappers
 #define OMP_CLAUSE_OPERAND(NODE, I)                            \
        OMP_CLAUSE_ELT_CHECK (NODE, I)
 
+/* True if the clause decl NODE contains an iterator.  */
+#define OMP_ITERATOR_DECL_P(NODE)                              \
+       (TREE_CODE (NODE) == TREE_LIST                          \
+        && TREE_PURPOSE (NODE)                                 \
+        && TREE_CODE (TREE_PURPOSE (NODE)) == TREE_VEC)
+
 /* In a BLOCK (scope) node:
    Variables declared in the scope NODE.  */
 #define BLOCK_VARS(NODE) (BLOCK_CHECK (NODE)->block.vars)
-- 
2.34.1

Reply via email to