annotate gcc/brig/brigfrontend/brig-atomic-inst-handler.cc @ 131:84e7813d76e9

gcc-8.2
author mir3636
date Thu, 25 Oct 2018 07:37:49 +0900
parents 04ced10e8804
children 1830386684a0
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 /* brig-atomic-inst-handler.cc -- brig atomic instruction handling
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
2 Copyright (C) 2016-2018 Free Software Foundation, Inc.
111
kono
parents:
diff changeset
3
kono
parents:
diff changeset
4 Contributed by Pekka Jaaskelainen <pekka.jaaskelainen@parmance.com>
kono
parents:
diff changeset
5 for General Processor Tech.
kono
parents:
diff changeset
6 This file is part of GCC.
kono
parents:
diff changeset
7
kono
parents:
diff changeset
8 GCC is free software; you can redistribute it and/or modify it under
kono
parents:
diff changeset
9 the terms of the GNU General Public License as published by the Free
kono
parents:
diff changeset
10 Software Foundation; either version 3, or (at your option) any later
kono
parents:
diff changeset
11 version.
kono
parents:
diff changeset
12
kono
parents:
diff changeset
13 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
kono
parents:
diff changeset
14 WARRANTY; without even the implied warranty of MERCHANTABILITY or
kono
parents:
diff changeset
15 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
kono
parents:
diff changeset
16 for more details.
kono
parents:
diff changeset
17
kono
parents:
diff changeset
18 You should have received a copy of the GNU General Public License
kono
parents:
diff changeset
19 along with GCC; see the file COPYING3. If not see
kono
parents:
diff changeset
20 <http://www.gnu.org/licenses/>. */
kono
parents:
diff changeset
21
kono
parents:
diff changeset
22 #include <sstream>
kono
parents:
diff changeset
23
kono
parents:
diff changeset
24 #include "brig-code-entry-handler.h"
kono
parents:
diff changeset
25 #include "brig-util.h"
kono
parents:
diff changeset
26 #include "fold-const.h"
kono
parents:
diff changeset
27 #include "diagnostic.h"
kono
parents:
diff changeset
28 #include "tree-pretty-print.h"
kono
parents:
diff changeset
29 #include "print-tree.h"
kono
parents:
diff changeset
30 #include "convert.h"
kono
parents:
diff changeset
31 #include "langhooks.h"
kono
parents:
diff changeset
32 #include "gimple-expr.h"
kono
parents:
diff changeset
33 #include "stringpool.h"
kono
parents:
diff changeset
34 #include "brig-builtins.h"
kono
parents:
diff changeset
35
kono
parents:
diff changeset
36 brig_atomic_inst_handler::brig_atomic_inst_handler (brig_to_generic &parent)
kono
parents:
diff changeset
37 : brig_code_entry_handler (parent)
kono
parents:
diff changeset
38 {
kono
parents:
diff changeset
39 }
kono
parents:
diff changeset
40
kono
parents:
diff changeset
41 size_t
kono
parents:
diff changeset
42 brig_atomic_inst_handler::generate_tree (const BrigInstBase &inst,
kono
parents:
diff changeset
43 BrigAtomicOperation8_t atomic_opcode)
kono
parents:
diff changeset
44 {
kono
parents:
diff changeset
45 tree_stl_vec operands = build_operands (inst);
kono
parents:
diff changeset
46 const int first_input
kono
parents:
diff changeset
47 = gccbrig_hsa_opcode_op_output_p (inst.opcode, 0) ? 1 : 0;
kono
parents:
diff changeset
48
kono
parents:
diff changeset
49 tree instr_type = gccbrig_tree_type_for_hsa_type (inst.type);
kono
parents:
diff changeset
50
kono
parents:
diff changeset
51 /* Utilize the atomic data types (from C++11 support) for implementing
kono
parents:
diff changeset
52 atomic operations. */
kono
parents:
diff changeset
53
kono
parents:
diff changeset
54 tree atomic_type = build_qualified_type (instr_type, TYPE_QUAL_ATOMIC);
kono
parents:
diff changeset
55
kono
parents:
diff changeset
56 gcc_assert (atomic_type != NULL_TREE);
kono
parents:
diff changeset
57
kono
parents:
diff changeset
58 tree signal_handle = operands[first_input];
kono
parents:
diff changeset
59 tree atomic_ptype = build_pointer_type (atomic_type);
kono
parents:
diff changeset
60 tree casted_to_ptr = convert_to_pointer (atomic_ptype, signal_handle);
kono
parents:
diff changeset
61
kono
parents:
diff changeset
62 tree src0 = NULL_TREE;
kono
parents:
diff changeset
63 if (atomic_opcode != BRIG_ATOMIC_LD)
kono
parents:
diff changeset
64 src0 = operands[first_input + 1];
kono
parents:
diff changeset
65
kono
parents:
diff changeset
66 tree instr_expr = NULL_TREE;
kono
parents:
diff changeset
67
kono
parents:
diff changeset
68 tree ptype = build_pointer_type (instr_type);
kono
parents:
diff changeset
69 tree ptr = convert_to_pointer (ptype, operands[first_input]);
kono
parents:
diff changeset
70
kono
parents:
diff changeset
71 if (atomic_opcode == BRIG_ATOMIC_ST)
kono
parents:
diff changeset
72 {
kono
parents:
diff changeset
73 tree mem_ref = build2 (MEM_REF, atomic_type, casted_to_ptr,
kono
parents:
diff changeset
74 build_int_cst (atomic_ptype, 0));
kono
parents:
diff changeset
75 instr_expr = build2 (MODIFY_EXPR, atomic_type, mem_ref, src0);
kono
parents:
diff changeset
76 }
kono
parents:
diff changeset
77 else if (atomic_opcode == BRIG_ATOMIC_LD
kono
parents:
diff changeset
78 || (atomic_opcode >= BRIG_ATOMIC_WAIT_EQ
kono
parents:
diff changeset
79 && atomic_opcode <= BRIG_ATOMIC_WAITTIMEOUT_GTE))
kono
parents:
diff changeset
80 {
kono
parents:
diff changeset
81 tree mem_ref = build2 (MEM_REF, atomic_type, casted_to_ptr,
kono
parents:
diff changeset
82 build_int_cst (atomic_ptype, 0));
kono
parents:
diff changeset
83 /* signal_wait* instructions can return spuriously before the
kono
parents:
diff changeset
84 condition becomes true. Therefore it's legal to return
kono
parents:
diff changeset
85 right away. TODO: builtin calls which can be
kono
parents:
diff changeset
86 implemented with a power efficient sleep-wait. */
kono
parents:
diff changeset
87 instr_expr = mem_ref;
kono
parents:
diff changeset
88 }
kono
parents:
diff changeset
89 else if (atomic_opcode == BRIG_ATOMIC_CAS)
kono
parents:
diff changeset
90 {
kono
parents:
diff changeset
91 /* Special case for CAS due to the two args. */
kono
parents:
diff changeset
92 tree built_in = NULL_TREE;
kono
parents:
diff changeset
93 switch (gccbrig_hsa_type_bit_size (inst.type))
kono
parents:
diff changeset
94 {
kono
parents:
diff changeset
95 case 32:
kono
parents:
diff changeset
96 built_in
kono
parents:
diff changeset
97 = builtin_decl_explicit (BUILT_IN_SYNC_VAL_COMPARE_AND_SWAP_4);
kono
parents:
diff changeset
98 break;
kono
parents:
diff changeset
99 case 64:
kono
parents:
diff changeset
100 built_in
kono
parents:
diff changeset
101 = builtin_decl_explicit (BUILT_IN_SYNC_VAL_COMPARE_AND_SWAP_8);
kono
parents:
diff changeset
102 break;
kono
parents:
diff changeset
103 default:
kono
parents:
diff changeset
104 gcc_unreachable ();
kono
parents:
diff changeset
105 }
kono
parents:
diff changeset
106
kono
parents:
diff changeset
107 tree src1 = operands[first_input + 2];
kono
parents:
diff changeset
108
kono
parents:
diff changeset
109 tree src0_type
kono
parents:
diff changeset
110 = TREE_VALUE (TREE_CHAIN (TYPE_ARG_TYPES (TREE_TYPE (built_in))));
kono
parents:
diff changeset
111
kono
parents:
diff changeset
112 tree src1_type = TREE_VALUE
kono
parents:
diff changeset
113 (TREE_CHAIN (TREE_CHAIN (TYPE_ARG_TYPES (TREE_TYPE (built_in)))));
kono
parents:
diff changeset
114
kono
parents:
diff changeset
115 instr_expr = call_builtin (built_in, 3, instr_type, ptype, ptr,
kono
parents:
diff changeset
116 src0_type, src0, src1_type, src1);
kono
parents:
diff changeset
117 }
kono
parents:
diff changeset
118 else
kono
parents:
diff changeset
119 {
kono
parents:
diff changeset
120 tree built_in = NULL_TREE;
kono
parents:
diff changeset
121 /* The rest of the builtins have the same number of parameters.
kono
parents:
diff changeset
122 Generate a big if..else that finds the correct builtin
kono
parents:
diff changeset
123 automagically from the def file. */
kono
parents:
diff changeset
124 #undef DEF_HSAIL_SAT_BUILTIN
kono
parents:
diff changeset
125 #undef DEF_HSAIL_BUILTIN
kono
parents:
diff changeset
126 #undef DEF_HSAIL_ATOMIC_BUILTIN
kono
parents:
diff changeset
127 #undef DEF_HSAIL_INTR_BUILTIN
kono
parents:
diff changeset
128 #undef DEF_HSAIL_CVT_ZEROI_SAT_BUILTIN
kono
parents:
diff changeset
129
kono
parents:
diff changeset
130 #define DEF_HSAIL_ATOMIC_BUILTIN(ENUM, ATOMIC_OPCODE, HSAIL_TYPE, \
kono
parents:
diff changeset
131 NAME, TYPE, ATTRS) \
kono
parents:
diff changeset
132 if (atomic_opcode == ATOMIC_OPCODE && inst.type == HSAIL_TYPE) \
kono
parents:
diff changeset
133 built_in = builtin_decl_explicit (ENUM); \
kono
parents:
diff changeset
134 else
kono
parents:
diff changeset
135 #include "brig-builtins.def"
kono
parents:
diff changeset
136 switch (atomic_opcode)
kono
parents:
diff changeset
137 {
kono
parents:
diff changeset
138 case BRIG_ATOMIC_ADD:
kono
parents:
diff changeset
139 switch (gccbrig_hsa_type_bit_size (inst.type))
kono
parents:
diff changeset
140 {
kono
parents:
diff changeset
141 case 32:
kono
parents:
diff changeset
142 built_in
kono
parents:
diff changeset
143 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_ADD_4);
kono
parents:
diff changeset
144 break;
kono
parents:
diff changeset
145 case 64:
kono
parents:
diff changeset
146 built_in
kono
parents:
diff changeset
147 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_ADD_8);
kono
parents:
diff changeset
148 break;
kono
parents:
diff changeset
149 default:
kono
parents:
diff changeset
150 gcc_unreachable ();
kono
parents:
diff changeset
151 }
kono
parents:
diff changeset
152 break;
kono
parents:
diff changeset
153 case BRIG_ATOMIC_SUB:
kono
parents:
diff changeset
154 switch (gccbrig_hsa_type_bit_size (inst.type))
kono
parents:
diff changeset
155 {
kono
parents:
diff changeset
156 case 32:
kono
parents:
diff changeset
157 built_in
kono
parents:
diff changeset
158 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_SUB_4);
kono
parents:
diff changeset
159 break;
kono
parents:
diff changeset
160 case 64:
kono
parents:
diff changeset
161 built_in
kono
parents:
diff changeset
162 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_SUB_8);
kono
parents:
diff changeset
163 break;
kono
parents:
diff changeset
164 default:
kono
parents:
diff changeset
165 gcc_unreachable ();
kono
parents:
diff changeset
166 }
kono
parents:
diff changeset
167 break;
kono
parents:
diff changeset
168 case BRIG_ATOMIC_AND:
kono
parents:
diff changeset
169 switch (gccbrig_hsa_type_bit_size (inst.type))
kono
parents:
diff changeset
170 {
kono
parents:
diff changeset
171 case 32:
kono
parents:
diff changeset
172 built_in
kono
parents:
diff changeset
173 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_AND_4);
kono
parents:
diff changeset
174 break;
kono
parents:
diff changeset
175 case 64:
kono
parents:
diff changeset
176 built_in
kono
parents:
diff changeset
177 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_AND_8);
kono
parents:
diff changeset
178 break;
kono
parents:
diff changeset
179 default:
kono
parents:
diff changeset
180 gcc_unreachable ();
kono
parents:
diff changeset
181 }
kono
parents:
diff changeset
182 break;
kono
parents:
diff changeset
183 case BRIG_ATOMIC_XOR:
kono
parents:
diff changeset
184 switch (gccbrig_hsa_type_bit_size (inst.type))
kono
parents:
diff changeset
185 {
kono
parents:
diff changeset
186 case 32:
kono
parents:
diff changeset
187 built_in
kono
parents:
diff changeset
188 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_XOR_4);
kono
parents:
diff changeset
189 break;
kono
parents:
diff changeset
190 case 64:
kono
parents:
diff changeset
191 built_in
kono
parents:
diff changeset
192 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_XOR_8);
kono
parents:
diff changeset
193 break;
kono
parents:
diff changeset
194 default:
kono
parents:
diff changeset
195 gcc_unreachable ();
kono
parents:
diff changeset
196 }
kono
parents:
diff changeset
197 break;
kono
parents:
diff changeset
198 case BRIG_ATOMIC_OR:
kono
parents:
diff changeset
199 switch (gccbrig_hsa_type_bit_size (inst.type))
kono
parents:
diff changeset
200 {
kono
parents:
diff changeset
201 case 32:
kono
parents:
diff changeset
202 built_in
kono
parents:
diff changeset
203 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_OR_4);
kono
parents:
diff changeset
204 break;
kono
parents:
diff changeset
205 case 64:
kono
parents:
diff changeset
206 built_in
kono
parents:
diff changeset
207 = builtin_decl_explicit (BUILT_IN_SYNC_FETCH_AND_OR_8);
kono
parents:
diff changeset
208 break;
kono
parents:
diff changeset
209 default:
kono
parents:
diff changeset
210 gcc_unreachable ();
kono
parents:
diff changeset
211 }
kono
parents:
diff changeset
212 break;
kono
parents:
diff changeset
213 case BRIG_ATOMIC_EXCH:
kono
parents:
diff changeset
214 switch (gccbrig_hsa_type_bit_size (inst.type))
kono
parents:
diff changeset
215 {
kono
parents:
diff changeset
216 case 32:
kono
parents:
diff changeset
217 built_in
kono
parents:
diff changeset
218 = builtin_decl_explicit (BUILT_IN_SYNC_LOCK_TEST_AND_SET_4);
kono
parents:
diff changeset
219 break;
kono
parents:
diff changeset
220 case 64:
kono
parents:
diff changeset
221 built_in
kono
parents:
diff changeset
222 = builtin_decl_explicit (BUILT_IN_SYNC_LOCK_TEST_AND_SET_8);
kono
parents:
diff changeset
223 break;
kono
parents:
diff changeset
224 default:
kono
parents:
diff changeset
225 gcc_unreachable ();
kono
parents:
diff changeset
226 }
kono
parents:
diff changeset
227 break;
kono
parents:
diff changeset
228 default:
kono
parents:
diff changeset
229 gcc_unreachable ();
kono
parents:
diff changeset
230 };
kono
parents:
diff changeset
231
kono
parents:
diff changeset
232 gcc_assert (built_in != NULL_TREE);
kono
parents:
diff changeset
233 tree arg0_type
kono
parents:
diff changeset
234 = TREE_VALUE (TREE_CHAIN (TYPE_ARG_TYPES (TREE_TYPE (built_in))));
kono
parents:
diff changeset
235
kono
parents:
diff changeset
236 instr_expr = call_builtin (built_in, 2, instr_type, ptr_type_node,
kono
parents:
diff changeset
237 ptr, arg0_type, src0);
kono
parents:
diff changeset
238
kono
parents:
diff changeset
239 /* We need a temp variable for the result, because otherwise
kono
parents:
diff changeset
240 the gimplifier drops a necessary (unsigned to signed) cast in
kono
parents:
diff changeset
241 the output assignment and fails a check later. */
kono
parents:
diff changeset
242 tree tmp_var = create_tmp_var (arg0_type, "builtin_out");
kono
parents:
diff changeset
243 tree tmp_assign
kono
parents:
diff changeset
244 = build2 (MODIFY_EXPR, TREE_TYPE (tmp_var), tmp_var, instr_expr);
kono
parents:
diff changeset
245 m_parent.m_cf->append_statement (tmp_assign);
kono
parents:
diff changeset
246 instr_expr = tmp_var;
kono
parents:
diff changeset
247 }
kono
parents:
diff changeset
248
kono
parents:
diff changeset
249 if (first_input > 0)
kono
parents:
diff changeset
250 build_output_assignment (inst, operands[0], instr_expr);
kono
parents:
diff changeset
251 else
kono
parents:
diff changeset
252 m_parent.m_cf->append_statement (instr_expr);
kono
parents:
diff changeset
253
kono
parents:
diff changeset
254 return inst.base.byteCount;
kono
parents:
diff changeset
255 }
kono
parents:
diff changeset
256
kono
parents:
diff changeset
257 size_t
kono
parents:
diff changeset
258 brig_atomic_inst_handler::operator () (const BrigBase *base)
kono
parents:
diff changeset
259 {
kono
parents:
diff changeset
260 const BrigInstAtomic *inst = (const BrigInstAtomic *) base;
kono
parents:
diff changeset
261 BrigAtomicOperation8_t atomic_opcode;
kono
parents:
diff changeset
262 atomic_opcode = inst->atomicOperation;
kono
parents:
diff changeset
263
kono
parents:
diff changeset
264 return generate_tree (inst->base, atomic_opcode);
kono
parents:
diff changeset
265 }