]>
git.ipfire.org Git - thirdparty/gcc.git/blob - gcc/c-family/c-cilkplus.c
1 /* This file contains routines to construct and validate Cilk Plus
2 constructs within the C and C++ front ends.
4 Copyright (C) 2013-2016 Free Software Foundation, Inc.
5 Contributed by Aldy Hernandez <aldyh@redhat.com>.
7 This file is part of GCC.
9 GCC is free software; you can redistribute it and/or modify it
10 under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 3, or (at your option)
14 GCC is distributed in the hope that it will be useful, but
15 WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 General Public License for more details.
19 You should have received a copy of the GNU General Public License
20 along with GCC; see the file COPYING3. If not see
21 <http://www.gnu.org/licenses/>. */
25 #include "coretypes.h"
28 /* Validate the body of a _Cilk_for construct or a <#pragma simd> for
31 Returns true if there were no errors, false otherwise. */
34 c_check_cilk_loop (location_t loc
, tree decl
)
36 if (TREE_THIS_VOLATILE (decl
))
38 error_at (loc
, "iteration variable cannot be volatile");
44 /* Validate and emit code for <#pragma simd> clauses. */
47 c_finish_cilk_clauses (tree clauses
)
49 for (tree c
= clauses
; c
; c
= OMP_CLAUSE_CHAIN (c
))
53 /* If a variable appears in a linear clause it cannot appear in
54 any other OMP clause. */
55 if (OMP_CLAUSE_CODE (c
) == OMP_CLAUSE_LINEAR
)
56 for (tree c2
= clauses
; c2
; c2
= OMP_CLAUSE_CHAIN (c2
))
60 enum omp_clause_code code
= OMP_CLAUSE_CODE (c2
);
64 case OMP_CLAUSE_LINEAR
:
65 case OMP_CLAUSE_PRIVATE
:
66 case OMP_CLAUSE_FIRSTPRIVATE
:
67 case OMP_CLAUSE_LASTPRIVATE
:
68 case OMP_CLAUSE_REDUCTION
:
71 case OMP_CLAUSE_SAFELEN
:
78 if (OMP_CLAUSE_DECL (c
) == OMP_CLAUSE_DECL (c2
))
80 error_at (OMP_CLAUSE_LOCATION (c2
),
81 "variable appears in more than one clause");
82 inform (OMP_CLAUSE_LOCATION (c
),
83 "other clause defined here");
84 // Remove problematic clauses.
85 OMP_CLAUSE_CHAIN (prev
) = OMP_CLAUSE_CHAIN (c2
);
94 /* Calculate number of iterations of CILK_FOR. */
97 cilk_for_number_of_iterations (tree cilk_for
)
99 tree t
, v
, n1
, n2
, step
, type
, init
, cond
, incr
, itype
;
100 enum tree_code cond_code
;
101 location_t loc
= EXPR_LOCATION (cilk_for
);
103 init
= TREE_VEC_ELT (OMP_FOR_INIT (cilk_for
), 0);
104 v
= TREE_OPERAND (init
, 0);
105 cond
= TREE_VEC_ELT (OMP_FOR_COND (cilk_for
), 0);
106 incr
= TREE_VEC_ELT (OMP_FOR_INCR (cilk_for
), 0);
107 type
= TREE_TYPE (v
);
109 gcc_assert (TREE_CODE (TREE_TYPE (v
)) == INTEGER_TYPE
110 || TREE_CODE (TREE_TYPE (v
)) == POINTER_TYPE
);
111 n1
= TREE_OPERAND (init
, 1);
112 cond_code
= TREE_CODE (cond
);
113 n2
= TREE_OPERAND (cond
, 1);
121 if (POINTER_TYPE_P (TREE_TYPE (n2
)))
122 n2
= fold_build_pointer_plus_hwi_loc (loc
, n2
, 1);
124 n2
= fold_build2_loc (loc
, PLUS_EXPR
, TREE_TYPE (n2
), n2
,
125 build_int_cst (TREE_TYPE (n2
), 1));
129 if (POINTER_TYPE_P (TREE_TYPE (n2
)))
130 n2
= fold_build_pointer_plus_hwi_loc (loc
, n2
, -1);
132 n2
= fold_build2_loc (loc
, MINUS_EXPR
, TREE_TYPE (n2
), n2
,
133 build_int_cst (TREE_TYPE (n2
), 1));
141 switch (TREE_CODE (incr
))
143 case PREINCREMENT_EXPR
:
144 case POSTINCREMENT_EXPR
:
145 step
= build_int_cst (TREE_TYPE (v
), 1);
147 case PREDECREMENT_EXPR
:
148 case POSTDECREMENT_EXPR
:
149 step
= build_int_cst (TREE_TYPE (v
), -1);
152 t
= TREE_OPERAND (incr
, 1);
153 gcc_assert (TREE_OPERAND (t
, 0) == v
);
154 switch (TREE_CODE (t
))
157 step
= TREE_OPERAND (t
, 1);
159 case POINTER_PLUS_EXPR
:
160 step
= fold_convert (ssizetype
, TREE_OPERAND (t
, 1));
163 step
= TREE_OPERAND (t
, 1);
164 step
= fold_build1_loc (loc
, NEGATE_EXPR
, TREE_TYPE (step
), step
);
175 if (POINTER_TYPE_P (itype
))
176 itype
= signed_type_for (itype
);
177 if (cond_code
== NE_EXPR
)
179 /* For NE_EXPR, we need to find out if the iterator increases
180 or decreases from whether step is positive or negative. */
182 if (TYPE_UNSIGNED (stype
))
183 stype
= signed_type_for (stype
);
184 cond
= fold_build2_loc (loc
, GE_EXPR
, boolean_type_node
,
185 fold_convert_loc (loc
, stype
, step
),
186 build_int_cst (stype
, 0));
187 t
= fold_build3_loc (loc
, COND_EXPR
, itype
, cond
,
188 build_int_cst (itype
, -1),
189 build_int_cst (itype
, 1));
192 t
= build_int_cst (itype
, (cond_code
== LT_EXPR
? -1 : 1));
193 t
= fold_build2_loc (loc
, PLUS_EXPR
, itype
,
194 fold_convert_loc (loc
, itype
, step
), t
);
195 t
= fold_build2_loc (loc
, PLUS_EXPR
, itype
, t
,
196 fold_convert_loc (loc
, itype
, n2
));
197 t
= fold_build2_loc (loc
, MINUS_EXPR
, itype
, t
,
198 fold_convert_loc (loc
, itype
, n1
));
199 if (TYPE_UNSIGNED (itype
) && cond_code
== GT_EXPR
)
200 t
= fold_build2_loc (loc
, TRUNC_DIV_EXPR
, itype
,
201 fold_build1_loc (loc
, NEGATE_EXPR
, itype
, t
),
202 fold_build1_loc (loc
, NEGATE_EXPR
, itype
,
203 fold_convert_loc (loc
, itype
,
205 else if (TYPE_UNSIGNED (itype
) && cond_code
== NE_EXPR
)
208 = fold_build2_loc (loc
, TRUNC_DIV_EXPR
, itype
, t
,
209 fold_convert_loc (loc
, itype
, step
));
211 = fold_build2_loc (loc
, TRUNC_DIV_EXPR
, itype
,
212 fold_build1_loc (loc
, NEGATE_EXPR
, itype
, t
),
213 fold_build1_loc (loc
, NEGATE_EXPR
, itype
,
214 fold_convert_loc (loc
, itype
,
216 t
= fold_build3_loc (loc
, COND_EXPR
, itype
, cond
, t1
, t2
);
219 t
= fold_build2_loc (loc
, TRUNC_DIV_EXPR
, itype
, t
,
220 fold_convert_loc (loc
, itype
, step
));
221 cond
= fold_build2_loc (loc
, cond_code
, boolean_type_node
, n1
, n2
);
222 t
= fold_build3_loc (loc
, COND_EXPR
, itype
, cond
, t
,
223 build_int_cst (itype
, 0));