annotate gcc/go/gofrontend/statements.h @ 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 // statements.h -- Go frontend statements. -*- C++ -*-
kono
parents:
diff changeset
2
kono
parents:
diff changeset
3 // Copyright 2009 The Go Authors. All rights reserved.
kono
parents:
diff changeset
4 // Use of this source code is governed by a BSD-style
kono
parents:
diff changeset
5 // license that can be found in the LICENSE file.
kono
parents:
diff changeset
6
kono
parents:
diff changeset
7 #ifndef GO_STATEMENTS_H
kono
parents:
diff changeset
8 #define GO_STATEMENTS_H
kono
parents:
diff changeset
9
kono
parents:
diff changeset
10 #include "operator.h"
kono
parents:
diff changeset
11
kono
parents:
diff changeset
12 class Gogo;
kono
parents:
diff changeset
13 class Traverse;
kono
parents:
diff changeset
14 class Statement_inserter;
kono
parents:
diff changeset
15 class Block;
kono
parents:
diff changeset
16 class Function;
kono
parents:
diff changeset
17 class Unnamed_label;
kono
parents:
diff changeset
18 class Assignment_statement;
kono
parents:
diff changeset
19 class Temporary_statement;
kono
parents:
diff changeset
20 class Variable_declaration_statement;
kono
parents:
diff changeset
21 class Expression_statement;
kono
parents:
diff changeset
22 class Block_statement;
kono
parents:
diff changeset
23 class Return_statement;
kono
parents:
diff changeset
24 class Thunk_statement;
kono
parents:
diff changeset
25 class Goto_statement;
kono
parents:
diff changeset
26 class Goto_unnamed_statement;
kono
parents:
diff changeset
27 class Label_statement;
kono
parents:
diff changeset
28 class Unnamed_label_statement;
kono
parents:
diff changeset
29 class If_statement;
kono
parents:
diff changeset
30 class For_statement;
kono
parents:
diff changeset
31 class For_range_statement;
kono
parents:
diff changeset
32 class Switch_statement;
kono
parents:
diff changeset
33 class Type_switch_statement;
kono
parents:
diff changeset
34 class Send_statement;
kono
parents:
diff changeset
35 class Select_statement;
kono
parents:
diff changeset
36 class Variable;
kono
parents:
diff changeset
37 class Named_object;
kono
parents:
diff changeset
38 class Label;
kono
parents:
diff changeset
39 class Translate_context;
kono
parents:
diff changeset
40 class Expression;
kono
parents:
diff changeset
41 class Expression_list;
kono
parents:
diff changeset
42 class Struct_type;
kono
parents:
diff changeset
43 class Call_expression;
kono
parents:
diff changeset
44 class Map_index_expression;
kono
parents:
diff changeset
45 class Receive_expression;
kono
parents:
diff changeset
46 class Case_clauses;
kono
parents:
diff changeset
47 class Type_case_clauses;
kono
parents:
diff changeset
48 class Select_clauses;
kono
parents:
diff changeset
49 class Typed_identifier_list;
kono
parents:
diff changeset
50 class Bexpression;
kono
parents:
diff changeset
51 class Bstatement;
kono
parents:
diff changeset
52 class Bvariable;
kono
parents:
diff changeset
53 class Ast_dump_context;
kono
parents:
diff changeset
54
kono
parents:
diff changeset
55 // This class is used to traverse assignments made by a statement
kono
parents:
diff changeset
56 // which makes assignments.
kono
parents:
diff changeset
57
kono
parents:
diff changeset
58 class Traverse_assignments
kono
parents:
diff changeset
59 {
kono
parents:
diff changeset
60 public:
kono
parents:
diff changeset
61 Traverse_assignments()
kono
parents:
diff changeset
62 { }
kono
parents:
diff changeset
63
kono
parents:
diff changeset
64 virtual ~Traverse_assignments()
kono
parents:
diff changeset
65 { }
kono
parents:
diff changeset
66
kono
parents:
diff changeset
67 // This is called for a variable initialization.
kono
parents:
diff changeset
68 virtual void
kono
parents:
diff changeset
69 initialize_variable(Named_object*) = 0;
kono
parents:
diff changeset
70
kono
parents:
diff changeset
71 // This is called for each assignment made by the statement. PLHS
kono
parents:
diff changeset
72 // points to the left hand side, and PRHS points to the right hand
kono
parents:
diff changeset
73 // side. PRHS may be NULL if there is no associated expression, as
kono
parents:
diff changeset
74 // in the bool set by a non-blocking receive.
kono
parents:
diff changeset
75 virtual void
kono
parents:
diff changeset
76 assignment(Expression** plhs, Expression** prhs) = 0;
kono
parents:
diff changeset
77
kono
parents:
diff changeset
78 // This is called for each expression which is not passed to the
kono
parents:
diff changeset
79 // assignment function. This is used for some of the statements
kono
parents:
diff changeset
80 // which assign two values, for which there is no expression which
kono
parents:
diff changeset
81 // describes the value. For ++ and -- the value is passed to both
kono
parents:
diff changeset
82 // the assignment method and the rhs method. IS_STORED is true if
kono
parents:
diff changeset
83 // this value is being stored directly. It is false if the value is
kono
parents:
diff changeset
84 // computed but not stored. IS_LOCAL is true if the value is being
kono
parents:
diff changeset
85 // stored in a local variable or this is being called by a return
kono
parents:
diff changeset
86 // statement.
kono
parents:
diff changeset
87 virtual void
kono
parents:
diff changeset
88 value(Expression**, bool is_stored, bool is_local) = 0;
kono
parents:
diff changeset
89 };
kono
parents:
diff changeset
90
kono
parents:
diff changeset
91 // A single statement.
kono
parents:
diff changeset
92
kono
parents:
diff changeset
93 class Statement
kono
parents:
diff changeset
94 {
kono
parents:
diff changeset
95 public:
kono
parents:
diff changeset
96 // The types of statements.
kono
parents:
diff changeset
97 enum Statement_classification
kono
parents:
diff changeset
98 {
kono
parents:
diff changeset
99 STATEMENT_ERROR,
kono
parents:
diff changeset
100 STATEMENT_VARIABLE_DECLARATION,
kono
parents:
diff changeset
101 STATEMENT_TEMPORARY,
kono
parents:
diff changeset
102 STATEMENT_ASSIGNMENT,
kono
parents:
diff changeset
103 STATEMENT_EXPRESSION,
kono
parents:
diff changeset
104 STATEMENT_BLOCK,
kono
parents:
diff changeset
105 STATEMENT_GO,
kono
parents:
diff changeset
106 STATEMENT_DEFER,
kono
parents:
diff changeset
107 STATEMENT_RETURN,
kono
parents:
diff changeset
108 STATEMENT_BREAK_OR_CONTINUE,
kono
parents:
diff changeset
109 STATEMENT_GOTO,
kono
parents:
diff changeset
110 STATEMENT_GOTO_UNNAMED,
kono
parents:
diff changeset
111 STATEMENT_LABEL,
kono
parents:
diff changeset
112 STATEMENT_UNNAMED_LABEL,
kono
parents:
diff changeset
113 STATEMENT_IF,
kono
parents:
diff changeset
114 STATEMENT_CONSTANT_SWITCH,
kono
parents:
diff changeset
115 STATEMENT_SEND,
kono
parents:
diff changeset
116 STATEMENT_SELECT,
kono
parents:
diff changeset
117
kono
parents:
diff changeset
118 // These statements types are created by the parser, but they
kono
parents:
diff changeset
119 // disappear during the lowering pass.
kono
parents:
diff changeset
120 STATEMENT_ASSIGNMENT_OPERATION,
kono
parents:
diff changeset
121 STATEMENT_TUPLE_ASSIGNMENT,
kono
parents:
diff changeset
122 STATEMENT_TUPLE_MAP_ASSIGNMENT,
kono
parents:
diff changeset
123 STATEMENT_TUPLE_RECEIVE_ASSIGNMENT,
kono
parents:
diff changeset
124 STATEMENT_TUPLE_TYPE_GUARD_ASSIGNMENT,
kono
parents:
diff changeset
125 STATEMENT_INCDEC,
kono
parents:
diff changeset
126 STATEMENT_FOR,
kono
parents:
diff changeset
127 STATEMENT_FOR_RANGE,
kono
parents:
diff changeset
128 STATEMENT_SWITCH,
kono
parents:
diff changeset
129 STATEMENT_TYPE_SWITCH
kono
parents:
diff changeset
130 };
kono
parents:
diff changeset
131
kono
parents:
diff changeset
132 Statement(Statement_classification, Location);
kono
parents:
diff changeset
133
kono
parents:
diff changeset
134 virtual ~Statement();
kono
parents:
diff changeset
135
kono
parents:
diff changeset
136 // Make a variable declaration.
kono
parents:
diff changeset
137 static Statement*
kono
parents:
diff changeset
138 make_variable_declaration(Named_object*);
kono
parents:
diff changeset
139
kono
parents:
diff changeset
140 // Make a statement which creates a temporary variable and
kono
parents:
diff changeset
141 // initializes it to an expression. The block is used if the
kono
parents:
diff changeset
142 // temporary variable has to be explicitly destroyed; the variable
kono
parents:
diff changeset
143 // must still be added to the block. References to the temporary
kono
parents:
diff changeset
144 // variable may be constructed using make_temporary_reference.
kono
parents:
diff changeset
145 // Either the type or the initialization expression may be NULL, but
kono
parents:
diff changeset
146 // not both.
kono
parents:
diff changeset
147 static Temporary_statement*
kono
parents:
diff changeset
148 make_temporary(Type*, Expression*, Location);
kono
parents:
diff changeset
149
kono
parents:
diff changeset
150 // Make an assignment statement.
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
151 static Assignment_statement*
111
kono
parents:
diff changeset
152 make_assignment(Expression*, Expression*, Location);
kono
parents:
diff changeset
153
kono
parents:
diff changeset
154 // Make an assignment operation (+=, etc.).
kono
parents:
diff changeset
155 static Statement*
kono
parents:
diff changeset
156 make_assignment_operation(Operator, Expression*, Expression*,
kono
parents:
diff changeset
157 Location);
kono
parents:
diff changeset
158
kono
parents:
diff changeset
159 // Make a tuple assignment statement.
kono
parents:
diff changeset
160 static Statement*
kono
parents:
diff changeset
161 make_tuple_assignment(Expression_list*, Expression_list*, Location);
kono
parents:
diff changeset
162
kono
parents:
diff changeset
163 // Make an assignment from a map index to a pair of variables.
kono
parents:
diff changeset
164 static Statement*
kono
parents:
diff changeset
165 make_tuple_map_assignment(Expression* val, Expression* present,
kono
parents:
diff changeset
166 Expression*, Location);
kono
parents:
diff changeset
167
kono
parents:
diff changeset
168 // Make an assignment from a nonblocking receive to a pair of
kono
parents:
diff changeset
169 // variables.
kono
parents:
diff changeset
170 static Statement*
kono
parents:
diff changeset
171 make_tuple_receive_assignment(Expression* val, Expression* closed,
kono
parents:
diff changeset
172 Expression* channel, Location);
kono
parents:
diff changeset
173
kono
parents:
diff changeset
174 // Make an assignment from a type guard to a pair of variables.
kono
parents:
diff changeset
175 static Statement*
kono
parents:
diff changeset
176 make_tuple_type_guard_assignment(Expression* val, Expression* ok,
kono
parents:
diff changeset
177 Expression* expr, Type* type,
kono
parents:
diff changeset
178 Location);
kono
parents:
diff changeset
179
kono
parents:
diff changeset
180 // Make an expression statement from an Expression. IS_IGNORED is
kono
parents:
diff changeset
181 // true if the value is being explicitly ignored, as in an
kono
parents:
diff changeset
182 // assignment to _.
kono
parents:
diff changeset
183 static Statement*
kono
parents:
diff changeset
184 make_statement(Expression*, bool is_ignored);
kono
parents:
diff changeset
185
kono
parents:
diff changeset
186 // Make a block statement from a Block. This is an embedded list of
kono
parents:
diff changeset
187 // statements which may also include variable definitions.
kono
parents:
diff changeset
188 static Statement*
kono
parents:
diff changeset
189 make_block_statement(Block*, Location);
kono
parents:
diff changeset
190
kono
parents:
diff changeset
191 // Make an increment statement.
kono
parents:
diff changeset
192 static Statement*
kono
parents:
diff changeset
193 make_inc_statement(Expression*);
kono
parents:
diff changeset
194
kono
parents:
diff changeset
195 // Make a decrement statement.
kono
parents:
diff changeset
196 static Statement*
kono
parents:
diff changeset
197 make_dec_statement(Expression*);
kono
parents:
diff changeset
198
kono
parents:
diff changeset
199 // Make a go statement.
kono
parents:
diff changeset
200 static Statement*
kono
parents:
diff changeset
201 make_go_statement(Call_expression* call, Location);
kono
parents:
diff changeset
202
kono
parents:
diff changeset
203 // Make a defer statement.
kono
parents:
diff changeset
204 static Statement*
kono
parents:
diff changeset
205 make_defer_statement(Call_expression* call, Location);
kono
parents:
diff changeset
206
kono
parents:
diff changeset
207 // Make a return statement.
kono
parents:
diff changeset
208 static Return_statement*
kono
parents:
diff changeset
209 make_return_statement(Expression_list*, Location);
kono
parents:
diff changeset
210
kono
parents:
diff changeset
211 // Make a statement that returns the result of a call expression.
kono
parents:
diff changeset
212 // If the call does not return any results, this just returns the
kono
parents:
diff changeset
213 // call expression as a statement, assuming that the function will
kono
parents:
diff changeset
214 // end immediately afterward.
kono
parents:
diff changeset
215 static Statement*
kono
parents:
diff changeset
216 make_return_from_call(Call_expression*, Location);
kono
parents:
diff changeset
217
kono
parents:
diff changeset
218 // Make a break statement.
kono
parents:
diff changeset
219 static Statement*
kono
parents:
diff changeset
220 make_break_statement(Unnamed_label* label, Location);
kono
parents:
diff changeset
221
kono
parents:
diff changeset
222 // Make a continue statement.
kono
parents:
diff changeset
223 static Statement*
kono
parents:
diff changeset
224 make_continue_statement(Unnamed_label* label, Location);
kono
parents:
diff changeset
225
kono
parents:
diff changeset
226 // Make a goto statement.
kono
parents:
diff changeset
227 static Statement*
kono
parents:
diff changeset
228 make_goto_statement(Label* label, Location);
kono
parents:
diff changeset
229
kono
parents:
diff changeset
230 // Make a goto statement to an unnamed label.
kono
parents:
diff changeset
231 static Statement*
kono
parents:
diff changeset
232 make_goto_unnamed_statement(Unnamed_label* label, Location);
kono
parents:
diff changeset
233
kono
parents:
diff changeset
234 // Make a label statement--where the label is defined.
kono
parents:
diff changeset
235 static Statement*
kono
parents:
diff changeset
236 make_label_statement(Label* label, Location);
kono
parents:
diff changeset
237
kono
parents:
diff changeset
238 // Make an unnamed label statement--where the label is defined.
kono
parents:
diff changeset
239 static Statement*
kono
parents:
diff changeset
240 make_unnamed_label_statement(Unnamed_label* label);
kono
parents:
diff changeset
241
kono
parents:
diff changeset
242 // Make an if statement.
kono
parents:
diff changeset
243 static Statement*
kono
parents:
diff changeset
244 make_if_statement(Expression* cond, Block* then_block, Block* else_block,
kono
parents:
diff changeset
245 Location);
kono
parents:
diff changeset
246
kono
parents:
diff changeset
247 // Make a switch statement.
kono
parents:
diff changeset
248 static Switch_statement*
kono
parents:
diff changeset
249 make_switch_statement(Expression* switch_val, Location);
kono
parents:
diff changeset
250
kono
parents:
diff changeset
251 // Make a type switch statement.
kono
parents:
diff changeset
252 static Type_switch_statement*
kono
parents:
diff changeset
253 make_type_switch_statement(const std::string&, Expression*, Location);
kono
parents:
diff changeset
254
kono
parents:
diff changeset
255 // Make a send statement.
kono
parents:
diff changeset
256 static Send_statement*
kono
parents:
diff changeset
257 make_send_statement(Expression* channel, Expression* val, Location);
kono
parents:
diff changeset
258
kono
parents:
diff changeset
259 // Make a select statement.
kono
parents:
diff changeset
260 static Select_statement*
kono
parents:
diff changeset
261 make_select_statement(Location);
kono
parents:
diff changeset
262
kono
parents:
diff changeset
263 // Make a for statement.
kono
parents:
diff changeset
264 static For_statement*
kono
parents:
diff changeset
265 make_for_statement(Block* init, Expression* cond, Block* post,
kono
parents:
diff changeset
266 Location location);
kono
parents:
diff changeset
267
kono
parents:
diff changeset
268 // Make a for statement with a range clause.
kono
parents:
diff changeset
269 static For_range_statement*
kono
parents:
diff changeset
270 make_for_range_statement(Expression* index_var, Expression* value_var,
kono
parents:
diff changeset
271 Expression* range, Location);
kono
parents:
diff changeset
272
kono
parents:
diff changeset
273 // Return the statement classification.
kono
parents:
diff changeset
274 Statement_classification
kono
parents:
diff changeset
275 classification() const
kono
parents:
diff changeset
276 { return this->classification_; }
kono
parents:
diff changeset
277
kono
parents:
diff changeset
278 // Get the statement location.
kono
parents:
diff changeset
279 Location
kono
parents:
diff changeset
280 location() const
kono
parents:
diff changeset
281 { return this->location_; }
kono
parents:
diff changeset
282
kono
parents:
diff changeset
283 // Traverse the tree.
kono
parents:
diff changeset
284 int
kono
parents:
diff changeset
285 traverse(Block*, size_t* index, Traverse*);
kono
parents:
diff changeset
286
kono
parents:
diff changeset
287 // Traverse the contents of this statement--the expressions and
kono
parents:
diff changeset
288 // statements which it contains.
kono
parents:
diff changeset
289 int
kono
parents:
diff changeset
290 traverse_contents(Traverse*);
kono
parents:
diff changeset
291
kono
parents:
diff changeset
292 // If this statement assigns some values, it calls a function for
kono
parents:
diff changeset
293 // each value to which this statement assigns a value, and returns
kono
parents:
diff changeset
294 // true. If this statement does not assign any values, it returns
kono
parents:
diff changeset
295 // false.
kono
parents:
diff changeset
296 bool
kono
parents:
diff changeset
297 traverse_assignments(Traverse_assignments* tassign);
kono
parents:
diff changeset
298
kono
parents:
diff changeset
299 // Lower a statement. This is called immediately after parsing to
kono
parents:
diff changeset
300 // simplify statements for further processing. It returns the same
kono
parents:
diff changeset
301 // Statement or a new one. FUNCTION is the function containing this
kono
parents:
diff changeset
302 // statement. BLOCK is the block containing this statement.
kono
parents:
diff changeset
303 // INSERTER can be used to insert new statements before this one.
kono
parents:
diff changeset
304 Statement*
kono
parents:
diff changeset
305 lower(Gogo* gogo, Named_object* function, Block* block,
kono
parents:
diff changeset
306 Statement_inserter* inserter)
kono
parents:
diff changeset
307 { return this->do_lower(gogo, function, block, inserter); }
kono
parents:
diff changeset
308
kono
parents:
diff changeset
309 // Flatten a statement. This is called immediately after the order of
kono
parents:
diff changeset
310 // evaluation rules are applied to statements. It returns the same
kono
parents:
diff changeset
311 // Statement or a new one. FUNCTION is the function containing this
kono
parents:
diff changeset
312 // statement. BLOCK is the block containing this statement.
kono
parents:
diff changeset
313 // INSERTER can be used to insert new statements before this one.
kono
parents:
diff changeset
314 Statement*
kono
parents:
diff changeset
315 flatten(Gogo* gogo, Named_object* function, Block* block,
kono
parents:
diff changeset
316 Statement_inserter* inserter)
kono
parents:
diff changeset
317 { return this->do_flatten(gogo, function, block, inserter); }
kono
parents:
diff changeset
318
kono
parents:
diff changeset
319 // Set type information for unnamed constants.
kono
parents:
diff changeset
320 void
kono
parents:
diff changeset
321 determine_types();
kono
parents:
diff changeset
322
kono
parents:
diff changeset
323 // Check types in a statement. This simply checks that any
kono
parents:
diff changeset
324 // expressions used by the statement have the right type.
kono
parents:
diff changeset
325 void
kono
parents:
diff changeset
326 check_types(Gogo* gogo)
kono
parents:
diff changeset
327 { this->do_check_types(gogo); }
kono
parents:
diff changeset
328
kono
parents:
diff changeset
329 // Return whether this is a block statement.
kono
parents:
diff changeset
330 bool
kono
parents:
diff changeset
331 is_block_statement() const
kono
parents:
diff changeset
332 { return this->classification_ == STATEMENT_BLOCK; }
kono
parents:
diff changeset
333
kono
parents:
diff changeset
334 // If this is an assignment statement, return it. Otherwise return
kono
parents:
diff changeset
335 // NULL.
kono
parents:
diff changeset
336 Assignment_statement*
kono
parents:
diff changeset
337 assignment_statement()
kono
parents:
diff changeset
338 {
kono
parents:
diff changeset
339 return this->convert<Assignment_statement, STATEMENT_ASSIGNMENT>();
kono
parents:
diff changeset
340 }
kono
parents:
diff changeset
341
kono
parents:
diff changeset
342 // If this is an temporary statement, return it. Otherwise return
kono
parents:
diff changeset
343 // NULL.
kono
parents:
diff changeset
344 Temporary_statement*
kono
parents:
diff changeset
345 temporary_statement()
kono
parents:
diff changeset
346 {
kono
parents:
diff changeset
347 return this->convert<Temporary_statement, STATEMENT_TEMPORARY>();
kono
parents:
diff changeset
348 }
kono
parents:
diff changeset
349
kono
parents:
diff changeset
350 // If this is a variable declaration statement, return it.
kono
parents:
diff changeset
351 // Otherwise return NULL.
kono
parents:
diff changeset
352 Variable_declaration_statement*
kono
parents:
diff changeset
353 variable_declaration_statement()
kono
parents:
diff changeset
354 {
kono
parents:
diff changeset
355 return this->convert<Variable_declaration_statement,
kono
parents:
diff changeset
356 STATEMENT_VARIABLE_DECLARATION>();
kono
parents:
diff changeset
357 }
kono
parents:
diff changeset
358
kono
parents:
diff changeset
359 // If this is an expression statement, return it. Otherwise return
kono
parents:
diff changeset
360 // NULL.
kono
parents:
diff changeset
361 Expression_statement*
kono
parents:
diff changeset
362 expression_statement()
kono
parents:
diff changeset
363 {
kono
parents:
diff changeset
364 return this->convert<Expression_statement, STATEMENT_EXPRESSION>();
kono
parents:
diff changeset
365 }
kono
parents:
diff changeset
366
kono
parents:
diff changeset
367 // If this is an block statement, return it. Otherwise return
kono
parents:
diff changeset
368 // NULL.
kono
parents:
diff changeset
369 Block_statement*
kono
parents:
diff changeset
370 block_statement()
kono
parents:
diff changeset
371 { return this->convert<Block_statement, STATEMENT_BLOCK>(); }
kono
parents:
diff changeset
372
kono
parents:
diff changeset
373 // If this is a return statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
374 Return_statement*
kono
parents:
diff changeset
375 return_statement()
kono
parents:
diff changeset
376 { return this->convert<Return_statement, STATEMENT_RETURN>(); }
kono
parents:
diff changeset
377
kono
parents:
diff changeset
378 // If this is a thunk statement (a go or defer statement), return
kono
parents:
diff changeset
379 // it. Otherwise return NULL.
kono
parents:
diff changeset
380 Thunk_statement*
kono
parents:
diff changeset
381 thunk_statement();
kono
parents:
diff changeset
382
kono
parents:
diff changeset
383 // If this is a goto statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
384 Goto_statement*
kono
parents:
diff changeset
385 goto_statement()
kono
parents:
diff changeset
386 { return this->convert<Goto_statement, STATEMENT_GOTO>(); }
kono
parents:
diff changeset
387
kono
parents:
diff changeset
388 // If this is a goto_unnamed statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
389 Goto_unnamed_statement*
kono
parents:
diff changeset
390 goto_unnamed_statement()
kono
parents:
diff changeset
391 { return this->convert<Goto_unnamed_statement, STATEMENT_GOTO_UNNAMED>(); }
kono
parents:
diff changeset
392
kono
parents:
diff changeset
393 // If this is a label statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
394 Label_statement*
kono
parents:
diff changeset
395 label_statement()
kono
parents:
diff changeset
396 { return this->convert<Label_statement, STATEMENT_LABEL>(); }
kono
parents:
diff changeset
397
kono
parents:
diff changeset
398 // If this is an unnamed_label statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
399 Unnamed_label_statement*
kono
parents:
diff changeset
400 unnamed_label_statement()
kono
parents:
diff changeset
401 { return this->convert<Unnamed_label_statement, STATEMENT_UNNAMED_LABEL>(); }
kono
parents:
diff changeset
402
kono
parents:
diff changeset
403 // If this is an if statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
404 If_statement*
kono
parents:
diff changeset
405 if_statement()
kono
parents:
diff changeset
406 { return this->convert<If_statement, STATEMENT_IF>(); }
kono
parents:
diff changeset
407
kono
parents:
diff changeset
408 // If this is a for statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
409 For_statement*
kono
parents:
diff changeset
410 for_statement()
kono
parents:
diff changeset
411 { return this->convert<For_statement, STATEMENT_FOR>(); }
kono
parents:
diff changeset
412
kono
parents:
diff changeset
413 // If this is a for statement over a range clause, return it.
kono
parents:
diff changeset
414 // Otherwise return NULL.
kono
parents:
diff changeset
415 For_range_statement*
kono
parents:
diff changeset
416 for_range_statement()
kono
parents:
diff changeset
417 { return this->convert<For_range_statement, STATEMENT_FOR_RANGE>(); }
kono
parents:
diff changeset
418
kono
parents:
diff changeset
419 // If this is a switch statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
420 Switch_statement*
kono
parents:
diff changeset
421 switch_statement()
kono
parents:
diff changeset
422 { return this->convert<Switch_statement, STATEMENT_SWITCH>(); }
kono
parents:
diff changeset
423
kono
parents:
diff changeset
424 // If this is a type switch statement, return it. Otherwise return
kono
parents:
diff changeset
425 // NULL.
kono
parents:
diff changeset
426 Type_switch_statement*
kono
parents:
diff changeset
427 type_switch_statement()
kono
parents:
diff changeset
428 { return this->convert<Type_switch_statement, STATEMENT_TYPE_SWITCH>(); }
kono
parents:
diff changeset
429
kono
parents:
diff changeset
430 // If this is a send statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
431 Send_statement*
kono
parents:
diff changeset
432 send_statement()
kono
parents:
diff changeset
433 { return this->convert<Send_statement, STATEMENT_SEND>(); }
kono
parents:
diff changeset
434
kono
parents:
diff changeset
435 // If this is a select statement, return it. Otherwise return NULL.
kono
parents:
diff changeset
436 Select_statement*
kono
parents:
diff changeset
437 select_statement()
kono
parents:
diff changeset
438 { return this->convert<Select_statement, STATEMENT_SELECT>(); }
kono
parents:
diff changeset
439
kono
parents:
diff changeset
440 // Return true if this statement may fall through--if after
kono
parents:
diff changeset
441 // executing this statement we may go on to execute the following
kono
parents:
diff changeset
442 // statement, if any.
kono
parents:
diff changeset
443 bool
kono
parents:
diff changeset
444 may_fall_through() const
kono
parents:
diff changeset
445 { return this->do_may_fall_through(); }
kono
parents:
diff changeset
446
kono
parents:
diff changeset
447 // Convert the statement to the backend representation.
kono
parents:
diff changeset
448 Bstatement*
kono
parents:
diff changeset
449 get_backend(Translate_context*);
kono
parents:
diff changeset
450
kono
parents:
diff changeset
451 // Dump AST representation of a statement to a dump context.
kono
parents:
diff changeset
452 void
kono
parents:
diff changeset
453 dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
454
kono
parents:
diff changeset
455 protected:
kono
parents:
diff changeset
456 // Implemented by child class: traverse the tree.
kono
parents:
diff changeset
457 virtual int
kono
parents:
diff changeset
458 do_traverse(Traverse*) = 0;
kono
parents:
diff changeset
459
kono
parents:
diff changeset
460 // Implemented by child class: traverse assignments. Any statement
kono
parents:
diff changeset
461 // which includes an assignment should implement this.
kono
parents:
diff changeset
462 virtual bool
kono
parents:
diff changeset
463 do_traverse_assignments(Traverse_assignments*)
kono
parents:
diff changeset
464 { return false; }
kono
parents:
diff changeset
465
kono
parents:
diff changeset
466 // Implemented by the child class: lower this statement to a simpler
kono
parents:
diff changeset
467 // one.
kono
parents:
diff changeset
468 virtual Statement*
kono
parents:
diff changeset
469 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*)
kono
parents:
diff changeset
470 { return this; }
kono
parents:
diff changeset
471
kono
parents:
diff changeset
472 // Implemented by the child class: lower this statement to a simpler
kono
parents:
diff changeset
473 // one.
kono
parents:
diff changeset
474 virtual Statement*
kono
parents:
diff changeset
475 do_flatten(Gogo*, Named_object*, Block*, Statement_inserter*)
kono
parents:
diff changeset
476 { return this; }
kono
parents:
diff changeset
477
kono
parents:
diff changeset
478 // Implemented by child class: set type information for unnamed
kono
parents:
diff changeset
479 // constants. Any statement which includes an expression needs to
kono
parents:
diff changeset
480 // implement this.
kono
parents:
diff changeset
481 virtual void
kono
parents:
diff changeset
482 do_determine_types()
kono
parents:
diff changeset
483 { }
kono
parents:
diff changeset
484
kono
parents:
diff changeset
485 // Implemented by child class: check types of expressions used in a
kono
parents:
diff changeset
486 // statement.
kono
parents:
diff changeset
487 virtual void
kono
parents:
diff changeset
488 do_check_types(Gogo*)
kono
parents:
diff changeset
489 { }
kono
parents:
diff changeset
490
kono
parents:
diff changeset
491 // Implemented by child class: return true if this statement may
kono
parents:
diff changeset
492 // fall through.
kono
parents:
diff changeset
493 virtual bool
kono
parents:
diff changeset
494 do_may_fall_through() const
kono
parents:
diff changeset
495 { return true; }
kono
parents:
diff changeset
496
kono
parents:
diff changeset
497 // Implemented by child class: convert to backend representation.
kono
parents:
diff changeset
498 virtual Bstatement*
kono
parents:
diff changeset
499 do_get_backend(Translate_context*) = 0;
kono
parents:
diff changeset
500
kono
parents:
diff changeset
501 // Implemented by child class: dump ast representation.
kono
parents:
diff changeset
502 virtual void
kono
parents:
diff changeset
503 do_dump_statement(Ast_dump_context*) const = 0;
kono
parents:
diff changeset
504
kono
parents:
diff changeset
505 // Traverse an expression in a statement.
kono
parents:
diff changeset
506 int
kono
parents:
diff changeset
507 traverse_expression(Traverse*, Expression**);
kono
parents:
diff changeset
508
kono
parents:
diff changeset
509 // Traverse an expression list in a statement. The Expression_list
kono
parents:
diff changeset
510 // may be NULL.
kono
parents:
diff changeset
511 int
kono
parents:
diff changeset
512 traverse_expression_list(Traverse*, Expression_list*);
kono
parents:
diff changeset
513
kono
parents:
diff changeset
514 // Traverse a type in a statement.
kono
parents:
diff changeset
515 int
kono
parents:
diff changeset
516 traverse_type(Traverse*, Type*);
kono
parents:
diff changeset
517
kono
parents:
diff changeset
518 // For children to call when they detect that they are in error.
kono
parents:
diff changeset
519 void
kono
parents:
diff changeset
520 set_is_error();
kono
parents:
diff changeset
521
kono
parents:
diff changeset
522 // For children to call to report an error conveniently.
kono
parents:
diff changeset
523 void
kono
parents:
diff changeset
524 report_error(const char*);
kono
parents:
diff changeset
525
kono
parents:
diff changeset
526 // For children to return an error statement from lower().
kono
parents:
diff changeset
527 static Statement*
kono
parents:
diff changeset
528 make_error_statement(Location);
kono
parents:
diff changeset
529
kono
parents:
diff changeset
530 private:
kono
parents:
diff changeset
531 // Convert to the desired statement classification, or return NULL.
kono
parents:
diff changeset
532 // This is a controlled dynamic cast.
kono
parents:
diff changeset
533 template<typename Statement_class, Statement_classification sc>
kono
parents:
diff changeset
534 Statement_class*
kono
parents:
diff changeset
535 convert()
kono
parents:
diff changeset
536 {
kono
parents:
diff changeset
537 return (this->classification_ == sc
kono
parents:
diff changeset
538 ? static_cast<Statement_class*>(this)
kono
parents:
diff changeset
539 : NULL);
kono
parents:
diff changeset
540 }
kono
parents:
diff changeset
541
kono
parents:
diff changeset
542 template<typename Statement_class, Statement_classification sc>
kono
parents:
diff changeset
543 const Statement_class*
kono
parents:
diff changeset
544 convert() const
kono
parents:
diff changeset
545 {
kono
parents:
diff changeset
546 return (this->classification_ == sc
kono
parents:
diff changeset
547 ? static_cast<const Statement_class*>(this)
kono
parents:
diff changeset
548 : NULL);
kono
parents:
diff changeset
549 }
kono
parents:
diff changeset
550
kono
parents:
diff changeset
551 // The statement classification.
kono
parents:
diff changeset
552 Statement_classification classification_;
kono
parents:
diff changeset
553 // The location in the input file of the start of this statement.
kono
parents:
diff changeset
554 Location location_;
kono
parents:
diff changeset
555 };
kono
parents:
diff changeset
556
kono
parents:
diff changeset
557 // An assignment statement.
kono
parents:
diff changeset
558
kono
parents:
diff changeset
559 class Assignment_statement : public Statement
kono
parents:
diff changeset
560 {
kono
parents:
diff changeset
561 public:
kono
parents:
diff changeset
562 Assignment_statement(Expression* lhs, Expression* rhs,
kono
parents:
diff changeset
563 Location location)
kono
parents:
diff changeset
564 : Statement(STATEMENT_ASSIGNMENT, location),
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
565 lhs_(lhs), rhs_(rhs), omit_write_barrier_(false)
111
kono
parents:
diff changeset
566 { }
kono
parents:
diff changeset
567
kono
parents:
diff changeset
568 Expression*
kono
parents:
diff changeset
569 lhs() const
kono
parents:
diff changeset
570 { return this->lhs_; }
kono
parents:
diff changeset
571
kono
parents:
diff changeset
572 Expression*
kono
parents:
diff changeset
573 rhs() const
kono
parents:
diff changeset
574 { return this->rhs_; }
kono
parents:
diff changeset
575
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
576 bool
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
577 omit_write_barrier() const
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
578 { return this->omit_write_barrier_; }
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
579
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
580 void
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
581 set_omit_write_barrier()
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
582 { this->omit_write_barrier_ = true; }
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
583
111
kono
parents:
diff changeset
584 protected:
kono
parents:
diff changeset
585 int
kono
parents:
diff changeset
586 do_traverse(Traverse* traverse);
kono
parents:
diff changeset
587
kono
parents:
diff changeset
588 bool
kono
parents:
diff changeset
589 do_traverse_assignments(Traverse_assignments*);
kono
parents:
diff changeset
590
kono
parents:
diff changeset
591 virtual Statement*
kono
parents:
diff changeset
592 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
593
kono
parents:
diff changeset
594 void
kono
parents:
diff changeset
595 do_determine_types();
kono
parents:
diff changeset
596
kono
parents:
diff changeset
597 void
kono
parents:
diff changeset
598 do_check_types(Gogo*);
kono
parents:
diff changeset
599
kono
parents:
diff changeset
600 Statement*
kono
parents:
diff changeset
601 do_flatten(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
602
kono
parents:
diff changeset
603 Bstatement*
kono
parents:
diff changeset
604 do_get_backend(Translate_context*);
kono
parents:
diff changeset
605
kono
parents:
diff changeset
606 void
kono
parents:
diff changeset
607 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
608
kono
parents:
diff changeset
609 private:
kono
parents:
diff changeset
610 // Left hand side--the lvalue.
kono
parents:
diff changeset
611 Expression* lhs_;
kono
parents:
diff changeset
612 // Right hand side--the rvalue.
kono
parents:
diff changeset
613 Expression* rhs_;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
614 // True if we can omit a write barrier from this assignment.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
615 bool omit_write_barrier_;
111
kono
parents:
diff changeset
616 };
kono
parents:
diff changeset
617
kono
parents:
diff changeset
618 // A statement which creates and initializes a temporary variable.
kono
parents:
diff changeset
619
kono
parents:
diff changeset
620 class Temporary_statement : public Statement
kono
parents:
diff changeset
621 {
kono
parents:
diff changeset
622 public:
kono
parents:
diff changeset
623 Temporary_statement(Type* type, Expression* init, Location location)
kono
parents:
diff changeset
624 : Statement(STATEMENT_TEMPORARY, location),
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
625 type_(type), init_(init), bvariable_(NULL), is_address_taken_(false),
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
626 value_escapes_(false)
111
kono
parents:
diff changeset
627 { }
kono
parents:
diff changeset
628
kono
parents:
diff changeset
629 // Return the type of the temporary variable.
kono
parents:
diff changeset
630 Type*
kono
parents:
diff changeset
631 type() const;
kono
parents:
diff changeset
632
kono
parents:
diff changeset
633 // Return the initializer if there is one.
kono
parents:
diff changeset
634 Expression*
kono
parents:
diff changeset
635 init() const
kono
parents:
diff changeset
636 { return this->init_; }
kono
parents:
diff changeset
637
kono
parents:
diff changeset
638 // Record that something takes the address of this temporary
kono
parents:
diff changeset
639 // variable.
kono
parents:
diff changeset
640 void
kono
parents:
diff changeset
641 set_is_address_taken()
kono
parents:
diff changeset
642 { this->is_address_taken_ = true; }
kono
parents:
diff changeset
643
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
644 // Whether the value escapes.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
645 bool
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
646 value_escapes() const
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
647 { return this->value_escapes_; }
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
648
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
649 // Record that the value escapes.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
650 void
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
651 set_value_escapes()
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
652 { this->value_escapes_ = true; }
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
653
111
kono
parents:
diff changeset
654 // Return the temporary variable. This should not be called until
kono
parents:
diff changeset
655 // after the statement itself has been converted.
kono
parents:
diff changeset
656 Bvariable*
kono
parents:
diff changeset
657 get_backend_variable(Translate_context*) const;
kono
parents:
diff changeset
658
kono
parents:
diff changeset
659 protected:
kono
parents:
diff changeset
660 int
kono
parents:
diff changeset
661 do_traverse(Traverse*);
kono
parents:
diff changeset
662
kono
parents:
diff changeset
663 bool
kono
parents:
diff changeset
664 do_traverse_assignments(Traverse_assignments*);
kono
parents:
diff changeset
665
kono
parents:
diff changeset
666 void
kono
parents:
diff changeset
667 do_determine_types();
kono
parents:
diff changeset
668
kono
parents:
diff changeset
669 void
kono
parents:
diff changeset
670 do_check_types(Gogo*);
kono
parents:
diff changeset
671
kono
parents:
diff changeset
672 Statement*
kono
parents:
diff changeset
673 do_flatten(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
674
kono
parents:
diff changeset
675 Bstatement*
kono
parents:
diff changeset
676 do_get_backend(Translate_context*);
kono
parents:
diff changeset
677
kono
parents:
diff changeset
678 void
kono
parents:
diff changeset
679 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
680
kono
parents:
diff changeset
681 private:
kono
parents:
diff changeset
682 // The type of the temporary variable.
kono
parents:
diff changeset
683 Type* type_;
kono
parents:
diff changeset
684 // The initial value of the temporary variable. This may be NULL.
kono
parents:
diff changeset
685 Expression* init_;
kono
parents:
diff changeset
686 // The backend representation of the temporary variable.
kono
parents:
diff changeset
687 Bvariable* bvariable_;
kono
parents:
diff changeset
688 // True if something takes the address of this temporary variable.
kono
parents:
diff changeset
689 bool is_address_taken_;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
690 // True if the value assigned to this temporary variable escapes.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
691 // This is used for select statements.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
692 bool value_escapes_;
111
kono
parents:
diff changeset
693 };
kono
parents:
diff changeset
694
kono
parents:
diff changeset
695 // A variable declaration. This marks the point in the code where a
kono
parents:
diff changeset
696 // variable is declared. The Variable is also attached to a Block.
kono
parents:
diff changeset
697
kono
parents:
diff changeset
698 class Variable_declaration_statement : public Statement
kono
parents:
diff changeset
699 {
kono
parents:
diff changeset
700 public:
kono
parents:
diff changeset
701 Variable_declaration_statement(Named_object* var);
kono
parents:
diff changeset
702
kono
parents:
diff changeset
703 // The variable being declared.
kono
parents:
diff changeset
704 Named_object*
kono
parents:
diff changeset
705 var()
kono
parents:
diff changeset
706 { return this->var_; }
kono
parents:
diff changeset
707
kono
parents:
diff changeset
708 protected:
kono
parents:
diff changeset
709 int
kono
parents:
diff changeset
710 do_traverse(Traverse*);
kono
parents:
diff changeset
711
kono
parents:
diff changeset
712 bool
kono
parents:
diff changeset
713 do_traverse_assignments(Traverse_assignments*);
kono
parents:
diff changeset
714
kono
parents:
diff changeset
715 Statement*
kono
parents:
diff changeset
716 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
717
kono
parents:
diff changeset
718 Statement*
kono
parents:
diff changeset
719 do_flatten(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
720
kono
parents:
diff changeset
721 Bstatement*
kono
parents:
diff changeset
722 do_get_backend(Translate_context*);
kono
parents:
diff changeset
723
kono
parents:
diff changeset
724 void
kono
parents:
diff changeset
725 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
726
kono
parents:
diff changeset
727 private:
kono
parents:
diff changeset
728 Named_object* var_;
kono
parents:
diff changeset
729 };
kono
parents:
diff changeset
730
kono
parents:
diff changeset
731 // A return statement.
kono
parents:
diff changeset
732
kono
parents:
diff changeset
733 class Return_statement : public Statement
kono
parents:
diff changeset
734 {
kono
parents:
diff changeset
735 public:
kono
parents:
diff changeset
736 Return_statement(Expression_list* vals, Location location)
kono
parents:
diff changeset
737 : Statement(STATEMENT_RETURN, location),
kono
parents:
diff changeset
738 vals_(vals), is_lowered_(false)
kono
parents:
diff changeset
739 { }
kono
parents:
diff changeset
740
kono
parents:
diff changeset
741 // The list of values being returned. This may be NULL.
kono
parents:
diff changeset
742 const Expression_list*
kono
parents:
diff changeset
743 vals() const
kono
parents:
diff changeset
744 { return this->vals_; }
kono
parents:
diff changeset
745
kono
parents:
diff changeset
746 protected:
kono
parents:
diff changeset
747 int
kono
parents:
diff changeset
748 do_traverse(Traverse* traverse)
kono
parents:
diff changeset
749 { return this->traverse_expression_list(traverse, this->vals_); }
kono
parents:
diff changeset
750
kono
parents:
diff changeset
751 bool
kono
parents:
diff changeset
752 do_traverse_assignments(Traverse_assignments*);
kono
parents:
diff changeset
753
kono
parents:
diff changeset
754 Statement*
kono
parents:
diff changeset
755 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
756
kono
parents:
diff changeset
757 bool
kono
parents:
diff changeset
758 do_may_fall_through() const
kono
parents:
diff changeset
759 { return false; }
kono
parents:
diff changeset
760
kono
parents:
diff changeset
761 Bstatement*
kono
parents:
diff changeset
762 do_get_backend(Translate_context*);
kono
parents:
diff changeset
763
kono
parents:
diff changeset
764 void
kono
parents:
diff changeset
765 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
766
kono
parents:
diff changeset
767 private:
kono
parents:
diff changeset
768 // Return values. This may be NULL.
kono
parents:
diff changeset
769 Expression_list* vals_;
kono
parents:
diff changeset
770 // True if this statement has been lowered.
kono
parents:
diff changeset
771 bool is_lowered_;
kono
parents:
diff changeset
772 };
kono
parents:
diff changeset
773
kono
parents:
diff changeset
774 // An expression statement.
kono
parents:
diff changeset
775
kono
parents:
diff changeset
776 class Expression_statement : public Statement
kono
parents:
diff changeset
777 {
kono
parents:
diff changeset
778 public:
kono
parents:
diff changeset
779 Expression_statement(Expression* expr, bool is_ignored);
kono
parents:
diff changeset
780
kono
parents:
diff changeset
781 Expression*
kono
parents:
diff changeset
782 expr()
kono
parents:
diff changeset
783 { return this->expr_; }
kono
parents:
diff changeset
784
kono
parents:
diff changeset
785 protected:
kono
parents:
diff changeset
786 int
kono
parents:
diff changeset
787 do_traverse(Traverse* traverse)
kono
parents:
diff changeset
788 { return this->traverse_expression(traverse, &this->expr_); }
kono
parents:
diff changeset
789
kono
parents:
diff changeset
790 void
kono
parents:
diff changeset
791 do_determine_types();
kono
parents:
diff changeset
792
kono
parents:
diff changeset
793 void
kono
parents:
diff changeset
794 do_check_types(Gogo*);
kono
parents:
diff changeset
795
kono
parents:
diff changeset
796 bool
kono
parents:
diff changeset
797 do_may_fall_through() const;
kono
parents:
diff changeset
798
kono
parents:
diff changeset
799 Bstatement*
kono
parents:
diff changeset
800 do_get_backend(Translate_context* context);
kono
parents:
diff changeset
801
kono
parents:
diff changeset
802 void
kono
parents:
diff changeset
803 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
804
kono
parents:
diff changeset
805 private:
kono
parents:
diff changeset
806 Expression* expr_;
kono
parents:
diff changeset
807 // Whether the value of this expression is being explicitly ignored.
kono
parents:
diff changeset
808 bool is_ignored_;
kono
parents:
diff changeset
809 };
kono
parents:
diff changeset
810
kono
parents:
diff changeset
811 // A block statement--a list of statements which may include variable
kono
parents:
diff changeset
812 // definitions.
kono
parents:
diff changeset
813
kono
parents:
diff changeset
814 class Block_statement : public Statement
kono
parents:
diff changeset
815 {
kono
parents:
diff changeset
816 public:
kono
parents:
diff changeset
817 Block_statement(Block* block, Location location)
kono
parents:
diff changeset
818 : Statement(STATEMENT_BLOCK, location),
kono
parents:
diff changeset
819 block_(block), is_lowered_for_statement_(false)
kono
parents:
diff changeset
820 { }
kono
parents:
diff changeset
821
kono
parents:
diff changeset
822 void
kono
parents:
diff changeset
823 set_is_lowered_for_statement()
kono
parents:
diff changeset
824 { this->is_lowered_for_statement_ = true; }
kono
parents:
diff changeset
825
kono
parents:
diff changeset
826 bool
kono
parents:
diff changeset
827 is_lowered_for_statement()
kono
parents:
diff changeset
828 { return this->is_lowered_for_statement_; }
kono
parents:
diff changeset
829
kono
parents:
diff changeset
830 protected:
kono
parents:
diff changeset
831 int
kono
parents:
diff changeset
832 do_traverse(Traverse* traverse)
kono
parents:
diff changeset
833 { return this->block_->traverse(traverse); }
kono
parents:
diff changeset
834
kono
parents:
diff changeset
835 void
kono
parents:
diff changeset
836 do_determine_types()
kono
parents:
diff changeset
837 { this->block_->determine_types(); }
kono
parents:
diff changeset
838
kono
parents:
diff changeset
839 bool
kono
parents:
diff changeset
840 do_may_fall_through() const
kono
parents:
diff changeset
841 { return this->block_->may_fall_through(); }
kono
parents:
diff changeset
842
kono
parents:
diff changeset
843 Bstatement*
kono
parents:
diff changeset
844 do_get_backend(Translate_context* context);
kono
parents:
diff changeset
845
kono
parents:
diff changeset
846 void
kono
parents:
diff changeset
847 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
848
kono
parents:
diff changeset
849 private:
kono
parents:
diff changeset
850 Block* block_;
kono
parents:
diff changeset
851 // True if this block statement represents a lowered for statement.
kono
parents:
diff changeset
852 bool is_lowered_for_statement_;
kono
parents:
diff changeset
853 };
kono
parents:
diff changeset
854
kono
parents:
diff changeset
855 // A send statement.
kono
parents:
diff changeset
856
kono
parents:
diff changeset
857 class Send_statement : public Statement
kono
parents:
diff changeset
858 {
kono
parents:
diff changeset
859 public:
kono
parents:
diff changeset
860 Send_statement(Expression* channel, Expression* val,
kono
parents:
diff changeset
861 Location location)
kono
parents:
diff changeset
862 : Statement(STATEMENT_SEND, location),
kono
parents:
diff changeset
863 channel_(channel), val_(val)
kono
parents:
diff changeset
864 { }
kono
parents:
diff changeset
865
kono
parents:
diff changeset
866 Expression*
kono
parents:
diff changeset
867 channel()
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
868 { return this->channel_; }
111
kono
parents:
diff changeset
869
kono
parents:
diff changeset
870 Expression*
kono
parents:
diff changeset
871 val()
kono
parents:
diff changeset
872 { return this->val_; }
kono
parents:
diff changeset
873
kono
parents:
diff changeset
874 protected:
kono
parents:
diff changeset
875 int
kono
parents:
diff changeset
876 do_traverse(Traverse* traverse);
kono
parents:
diff changeset
877
kono
parents:
diff changeset
878 void
kono
parents:
diff changeset
879 do_determine_types();
kono
parents:
diff changeset
880
kono
parents:
diff changeset
881 void
kono
parents:
diff changeset
882 do_check_types(Gogo*);
kono
parents:
diff changeset
883
kono
parents:
diff changeset
884 Statement*
kono
parents:
diff changeset
885 do_flatten(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
886
kono
parents:
diff changeset
887 Bstatement*
kono
parents:
diff changeset
888 do_get_backend(Translate_context*);
kono
parents:
diff changeset
889
kono
parents:
diff changeset
890 void
kono
parents:
diff changeset
891 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
892
kono
parents:
diff changeset
893 private:
kono
parents:
diff changeset
894 // The channel on which to send the value.
kono
parents:
diff changeset
895 Expression* channel_;
kono
parents:
diff changeset
896 // The value to send.
kono
parents:
diff changeset
897 Expression* val_;
kono
parents:
diff changeset
898 };
kono
parents:
diff changeset
899
kono
parents:
diff changeset
900 // Select_clauses holds the clauses of a select statement. This is
kono
parents:
diff changeset
901 // built by the parser.
kono
parents:
diff changeset
902
kono
parents:
diff changeset
903 class Select_clauses
kono
parents:
diff changeset
904 {
kono
parents:
diff changeset
905 public:
kono
parents:
diff changeset
906 Select_clauses()
kono
parents:
diff changeset
907 : clauses_()
kono
parents:
diff changeset
908 { }
kono
parents:
diff changeset
909
kono
parents:
diff changeset
910 // Add a new clause. IS_SEND is true if this is a send clause,
kono
parents:
diff changeset
911 // false for a receive clause. For a send clause CHANNEL is the
kono
parents:
diff changeset
912 // channel and VAL is the value to send. For a receive clause
kono
parents:
diff changeset
913 // CHANNEL is the channel, VAL is either NULL or a Var_expression
kono
parents:
diff changeset
914 // for the variable to set, and CLOSED is either NULL or a
kono
parents:
diff changeset
915 // Var_expression to set to whether the channel is closed. If VAL
kono
parents:
diff changeset
916 // is NULL, VAR may be a variable to be initialized with the
kono
parents:
diff changeset
917 // received value, and CLOSEDVAR ma be a variable to be initialized
kono
parents:
diff changeset
918 // with whether the channel is closed. IS_DEFAULT is true if this
kono
parents:
diff changeset
919 // is the default clause. STATEMENTS is the list of statements to
kono
parents:
diff changeset
920 // execute.
kono
parents:
diff changeset
921 void
kono
parents:
diff changeset
922 add(bool is_send, Expression* channel, Expression* val, Expression* closed,
kono
parents:
diff changeset
923 Named_object* var, Named_object* closedvar, bool is_default,
kono
parents:
diff changeset
924 Block* statements, Location location)
kono
parents:
diff changeset
925 {
kono
parents:
diff changeset
926 this->clauses_.push_back(Select_clause(is_send, channel, val, closed, var,
kono
parents:
diff changeset
927 closedvar, is_default, statements,
kono
parents:
diff changeset
928 location));
kono
parents:
diff changeset
929 }
kono
parents:
diff changeset
930
kono
parents:
diff changeset
931 size_t
kono
parents:
diff changeset
932 size() const
kono
parents:
diff changeset
933 { return this->clauses_.size(); }
kono
parents:
diff changeset
934
kono
parents:
diff changeset
935 // Traverse the select clauses.
kono
parents:
diff changeset
936 int
kono
parents:
diff changeset
937 traverse(Traverse*);
kono
parents:
diff changeset
938
kono
parents:
diff changeset
939 // Lower statements.
kono
parents:
diff changeset
940 void
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
941 lower(Gogo*, Named_object*, Block*, Temporary_statement*,
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
942 Temporary_statement*);
111
kono
parents:
diff changeset
943
kono
parents:
diff changeset
944 // Determine types.
kono
parents:
diff changeset
945 void
kono
parents:
diff changeset
946 determine_types();
kono
parents:
diff changeset
947
kono
parents:
diff changeset
948 // Check types.
kono
parents:
diff changeset
949 void
kono
parents:
diff changeset
950 check_types();
kono
parents:
diff changeset
951
kono
parents:
diff changeset
952 // Whether the select clauses may fall through to the statement
kono
parents:
diff changeset
953 // which follows the overall select statement.
kono
parents:
diff changeset
954 bool
kono
parents:
diff changeset
955 may_fall_through() const;
kono
parents:
diff changeset
956
kono
parents:
diff changeset
957 // Convert to the backend representation.
kono
parents:
diff changeset
958 Bstatement*
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
959 get_backend(Translate_context*, Temporary_statement* index,
111
kono
parents:
diff changeset
960 Unnamed_label* break_label, Location);
kono
parents:
diff changeset
961
kono
parents:
diff changeset
962 // Dump AST representation.
kono
parents:
diff changeset
963 void
kono
parents:
diff changeset
964 dump_clauses(Ast_dump_context*) const;
kono
parents:
diff changeset
965
kono
parents:
diff changeset
966 private:
kono
parents:
diff changeset
967 // A single clause.
kono
parents:
diff changeset
968 class Select_clause
kono
parents:
diff changeset
969 {
kono
parents:
diff changeset
970 public:
kono
parents:
diff changeset
971 Select_clause()
kono
parents:
diff changeset
972 : channel_(NULL), val_(NULL), closed_(NULL), var_(NULL),
kono
parents:
diff changeset
973 closedvar_(NULL), statements_(NULL), is_send_(false),
kono
parents:
diff changeset
974 is_default_(false)
kono
parents:
diff changeset
975 { }
kono
parents:
diff changeset
976
kono
parents:
diff changeset
977 Select_clause(bool is_send, Expression* channel, Expression* val,
kono
parents:
diff changeset
978 Expression* closed, Named_object* var,
kono
parents:
diff changeset
979 Named_object* closedvar, bool is_default, Block* statements,
kono
parents:
diff changeset
980 Location location)
kono
parents:
diff changeset
981 : channel_(channel), val_(val), closed_(closed), var_(var),
kono
parents:
diff changeset
982 closedvar_(closedvar), statements_(statements), location_(location),
kono
parents:
diff changeset
983 is_send_(is_send), is_default_(is_default), is_lowered_(false)
kono
parents:
diff changeset
984 { go_assert(is_default ? channel == NULL : channel != NULL); }
kono
parents:
diff changeset
985
kono
parents:
diff changeset
986 // Traverse the select clause.
kono
parents:
diff changeset
987 int
kono
parents:
diff changeset
988 traverse(Traverse*);
kono
parents:
diff changeset
989
kono
parents:
diff changeset
990 // Lower statements.
kono
parents:
diff changeset
991 void
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
992 lower(Gogo*, Named_object*, Block*, Temporary_statement*, size_t,
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
993 Temporary_statement*);
111
kono
parents:
diff changeset
994
kono
parents:
diff changeset
995 // Determine types.
kono
parents:
diff changeset
996 void
kono
parents:
diff changeset
997 determine_types();
kono
parents:
diff changeset
998
kono
parents:
diff changeset
999 // Check types.
kono
parents:
diff changeset
1000 void
kono
parents:
diff changeset
1001 check_types();
kono
parents:
diff changeset
1002
kono
parents:
diff changeset
1003 // Return true if this is the default clause.
kono
parents:
diff changeset
1004 bool
kono
parents:
diff changeset
1005 is_default() const
kono
parents:
diff changeset
1006 { return this->is_default_; }
kono
parents:
diff changeset
1007
kono
parents:
diff changeset
1008 // Return the channel. This will return NULL for the default
kono
parents:
diff changeset
1009 // clause.
kono
parents:
diff changeset
1010 Expression*
kono
parents:
diff changeset
1011 channel() const
kono
parents:
diff changeset
1012 { return this->channel_; }
kono
parents:
diff changeset
1013
kono
parents:
diff changeset
1014 // Return true for a send, false for a receive.
kono
parents:
diff changeset
1015 bool
kono
parents:
diff changeset
1016 is_send() const
kono
parents:
diff changeset
1017 {
kono
parents:
diff changeset
1018 go_assert(!this->is_default_);
kono
parents:
diff changeset
1019 return this->is_send_;
kono
parents:
diff changeset
1020 }
kono
parents:
diff changeset
1021
kono
parents:
diff changeset
1022 // Return the statements.
kono
parents:
diff changeset
1023 const Block*
kono
parents:
diff changeset
1024 statements() const
kono
parents:
diff changeset
1025 { return this->statements_; }
kono
parents:
diff changeset
1026
kono
parents:
diff changeset
1027 // Return the location.
kono
parents:
diff changeset
1028 Location
kono
parents:
diff changeset
1029 location() const
kono
parents:
diff changeset
1030 { return this->location_; }
kono
parents:
diff changeset
1031
kono
parents:
diff changeset
1032 // Whether this clause may fall through to the statement which
kono
parents:
diff changeset
1033 // follows the overall select statement.
kono
parents:
diff changeset
1034 bool
kono
parents:
diff changeset
1035 may_fall_through() const;
kono
parents:
diff changeset
1036
kono
parents:
diff changeset
1037 // Convert the statements to the backend representation.
kono
parents:
diff changeset
1038 Bstatement*
kono
parents:
diff changeset
1039 get_statements_backend(Translate_context*);
kono
parents:
diff changeset
1040
kono
parents:
diff changeset
1041 // Dump AST representation.
kono
parents:
diff changeset
1042 void
kono
parents:
diff changeset
1043 dump_clause(Ast_dump_context*) const;
kono
parents:
diff changeset
1044
kono
parents:
diff changeset
1045 private:
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1046 // These values must match the values in libgo/go/runtime/select.go.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1047 enum
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1048 {
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1049 caseRecv = 1,
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1050 caseSend = 2,
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1051 caseDefault = 3,
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1052 };
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1053
111
kono
parents:
diff changeset
1054 void
kono
parents:
diff changeset
1055 lower_default(Block*, Expression*);
kono
parents:
diff changeset
1056
kono
parents:
diff changeset
1057 void
kono
parents:
diff changeset
1058 lower_send(Block*, Expression*, Expression*);
kono
parents:
diff changeset
1059
kono
parents:
diff changeset
1060 void
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1061 lower_recv(Gogo*, Named_object*, Block*, Expression*, Expression*,
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1062 Temporary_statement*);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1063
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1064 void
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1065 set_case(Block*, Expression*, Expression*, Expression*, int);
111
kono
parents:
diff changeset
1066
kono
parents:
diff changeset
1067 // The channel.
kono
parents:
diff changeset
1068 Expression* channel_;
kono
parents:
diff changeset
1069 // The value to send or the lvalue to receive into.
kono
parents:
diff changeset
1070 Expression* val_;
kono
parents:
diff changeset
1071 // The lvalue to set to whether the channel is closed on a
kono
parents:
diff changeset
1072 // receive.
kono
parents:
diff changeset
1073 Expression* closed_;
kono
parents:
diff changeset
1074 // The variable to initialize, for "case a := <-ch".
kono
parents:
diff changeset
1075 Named_object* var_;
kono
parents:
diff changeset
1076 // The variable to initialize to whether the channel is closed,
kono
parents:
diff changeset
1077 // for "case a, c := <-ch".
kono
parents:
diff changeset
1078 Named_object* closedvar_;
kono
parents:
diff changeset
1079 // The statements to execute.
kono
parents:
diff changeset
1080 Block* statements_;
kono
parents:
diff changeset
1081 // The location of this clause.
kono
parents:
diff changeset
1082 Location location_;
kono
parents:
diff changeset
1083 // Whether this is a send or a receive.
kono
parents:
diff changeset
1084 bool is_send_;
kono
parents:
diff changeset
1085 // Whether this is the default.
kono
parents:
diff changeset
1086 bool is_default_;
kono
parents:
diff changeset
1087 // Whether this has been lowered.
kono
parents:
diff changeset
1088 bool is_lowered_;
kono
parents:
diff changeset
1089 };
kono
parents:
diff changeset
1090
kono
parents:
diff changeset
1091 typedef std::vector<Select_clause> Clauses;
kono
parents:
diff changeset
1092
kono
parents:
diff changeset
1093 Clauses clauses_;
kono
parents:
diff changeset
1094 };
kono
parents:
diff changeset
1095
kono
parents:
diff changeset
1096 // A select statement.
kono
parents:
diff changeset
1097
kono
parents:
diff changeset
1098 class Select_statement : public Statement
kono
parents:
diff changeset
1099 {
kono
parents:
diff changeset
1100 public:
kono
parents:
diff changeset
1101 Select_statement(Location location)
kono
parents:
diff changeset
1102 : Statement(STATEMENT_SELECT, location),
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1103 clauses_(NULL), index_(NULL), break_label_(NULL), is_lowered_(false)
111
kono
parents:
diff changeset
1104 { }
kono
parents:
diff changeset
1105
kono
parents:
diff changeset
1106 // Add the clauses.
kono
parents:
diff changeset
1107 void
kono
parents:
diff changeset
1108 add_clauses(Select_clauses* clauses)
kono
parents:
diff changeset
1109 {
kono
parents:
diff changeset
1110 go_assert(this->clauses_ == NULL);
kono
parents:
diff changeset
1111 this->clauses_ = clauses;
kono
parents:
diff changeset
1112 }
kono
parents:
diff changeset
1113
kono
parents:
diff changeset
1114 // Return the break label for this select statement.
kono
parents:
diff changeset
1115 Unnamed_label*
kono
parents:
diff changeset
1116 break_label();
kono
parents:
diff changeset
1117
kono
parents:
diff changeset
1118 protected:
kono
parents:
diff changeset
1119 int
kono
parents:
diff changeset
1120 do_traverse(Traverse* traverse)
kono
parents:
diff changeset
1121 { return this->clauses_->traverse(traverse); }
kono
parents:
diff changeset
1122
kono
parents:
diff changeset
1123 Statement*
kono
parents:
diff changeset
1124 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
1125
kono
parents:
diff changeset
1126 void
kono
parents:
diff changeset
1127 do_determine_types()
kono
parents:
diff changeset
1128 { this->clauses_->determine_types(); }
kono
parents:
diff changeset
1129
kono
parents:
diff changeset
1130 void
kono
parents:
diff changeset
1131 do_check_types(Gogo*)
kono
parents:
diff changeset
1132 { this->clauses_->check_types(); }
kono
parents:
diff changeset
1133
kono
parents:
diff changeset
1134 bool
kono
parents:
diff changeset
1135 do_may_fall_through() const;
kono
parents:
diff changeset
1136
kono
parents:
diff changeset
1137 Bstatement*
kono
parents:
diff changeset
1138 do_get_backend(Translate_context*);
kono
parents:
diff changeset
1139
kono
parents:
diff changeset
1140 void
kono
parents:
diff changeset
1141 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1142
kono
parents:
diff changeset
1143 private:
kono
parents:
diff changeset
1144 // The select clauses.
kono
parents:
diff changeset
1145 Select_clauses* clauses_;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1146 // A temporary that holds the index value returned by selectgo.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1147 Temporary_statement* index_;
111
kono
parents:
diff changeset
1148 // The break label.
kono
parents:
diff changeset
1149 Unnamed_label* break_label_;
kono
parents:
diff changeset
1150 // Whether this statement has been lowered.
kono
parents:
diff changeset
1151 bool is_lowered_;
kono
parents:
diff changeset
1152 };
kono
parents:
diff changeset
1153
kono
parents:
diff changeset
1154 // A statement which requires a thunk: go or defer.
kono
parents:
diff changeset
1155
kono
parents:
diff changeset
1156 class Thunk_statement : public Statement
kono
parents:
diff changeset
1157 {
kono
parents:
diff changeset
1158 public:
kono
parents:
diff changeset
1159 Thunk_statement(Statement_classification, Call_expression*,
kono
parents:
diff changeset
1160 Location);
kono
parents:
diff changeset
1161
kono
parents:
diff changeset
1162 // Return the call expression.
kono
parents:
diff changeset
1163 Expression*
kono
parents:
diff changeset
1164 call() const
kono
parents:
diff changeset
1165 { return this->call_; }
kono
parents:
diff changeset
1166
kono
parents:
diff changeset
1167 // Simplify a go or defer statement so that it only uses a single
kono
parents:
diff changeset
1168 // parameter.
kono
parents:
diff changeset
1169 bool
kono
parents:
diff changeset
1170 simplify_statement(Gogo*, Named_object*, Block*);
kono
parents:
diff changeset
1171
kono
parents:
diff changeset
1172 protected:
kono
parents:
diff changeset
1173 int
kono
parents:
diff changeset
1174 do_traverse(Traverse* traverse);
kono
parents:
diff changeset
1175
kono
parents:
diff changeset
1176 bool
kono
parents:
diff changeset
1177 do_traverse_assignments(Traverse_assignments*);
kono
parents:
diff changeset
1178
kono
parents:
diff changeset
1179 void
kono
parents:
diff changeset
1180 do_determine_types();
kono
parents:
diff changeset
1181
kono
parents:
diff changeset
1182 void
kono
parents:
diff changeset
1183 do_check_types(Gogo*);
kono
parents:
diff changeset
1184
kono
parents:
diff changeset
1185 // Return the function and argument for the call.
kono
parents:
diff changeset
1186 bool
kono
parents:
diff changeset
1187 get_fn_and_arg(Expression** pfn, Expression** parg);
kono
parents:
diff changeset
1188
kono
parents:
diff changeset
1189 private:
kono
parents:
diff changeset
1190 // Return whether this is a simple go statement.
kono
parents:
diff changeset
1191 bool
kono
parents:
diff changeset
1192 is_simple(Function_type*) const;
kono
parents:
diff changeset
1193
kono
parents:
diff changeset
1194 // Return whether the thunk function is a constant.
kono
parents:
diff changeset
1195 bool
kono
parents:
diff changeset
1196 is_constant_function() const;
kono
parents:
diff changeset
1197
kono
parents:
diff changeset
1198 // Build the struct to use for a complex case.
kono
parents:
diff changeset
1199 Struct_type*
kono
parents:
diff changeset
1200 build_struct(Function_type* fntype);
kono
parents:
diff changeset
1201
kono
parents:
diff changeset
1202 // Build the thunk.
kono
parents:
diff changeset
1203 void
kono
parents:
diff changeset
1204 build_thunk(Gogo*, const std::string&);
kono
parents:
diff changeset
1205
kono
parents:
diff changeset
1206 // Set the name to use for thunk field N.
kono
parents:
diff changeset
1207 void
kono
parents:
diff changeset
1208 thunk_field_param(int n, char* buf, size_t buflen);
kono
parents:
diff changeset
1209
kono
parents:
diff changeset
1210 // The function call to be executed in a separate thread (go) or
kono
parents:
diff changeset
1211 // later (defer).
kono
parents:
diff changeset
1212 Expression* call_;
kono
parents:
diff changeset
1213 // The type used for a struct to pass to a thunk, if this is not a
kono
parents:
diff changeset
1214 // simple call.
kono
parents:
diff changeset
1215 Struct_type* struct_type_;
kono
parents:
diff changeset
1216 };
kono
parents:
diff changeset
1217
kono
parents:
diff changeset
1218 // A go statement.
kono
parents:
diff changeset
1219
kono
parents:
diff changeset
1220 class Go_statement : public Thunk_statement
kono
parents:
diff changeset
1221 {
kono
parents:
diff changeset
1222 public:
kono
parents:
diff changeset
1223 Go_statement(Call_expression* call, Location location)
kono
parents:
diff changeset
1224 : Thunk_statement(STATEMENT_GO, call, location)
kono
parents:
diff changeset
1225 { }
kono
parents:
diff changeset
1226
kono
parents:
diff changeset
1227 protected:
kono
parents:
diff changeset
1228 Bstatement*
kono
parents:
diff changeset
1229 do_get_backend(Translate_context*);
kono
parents:
diff changeset
1230
kono
parents:
diff changeset
1231 void
kono
parents:
diff changeset
1232 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1233 };
kono
parents:
diff changeset
1234
kono
parents:
diff changeset
1235 // A defer statement.
kono
parents:
diff changeset
1236
kono
parents:
diff changeset
1237 class Defer_statement : public Thunk_statement
kono
parents:
diff changeset
1238 {
kono
parents:
diff changeset
1239 public:
kono
parents:
diff changeset
1240 Defer_statement(Call_expression* call, Location location)
kono
parents:
diff changeset
1241 : Thunk_statement(STATEMENT_DEFER, call, location)
kono
parents:
diff changeset
1242 { }
kono
parents:
diff changeset
1243
kono
parents:
diff changeset
1244 protected:
kono
parents:
diff changeset
1245 Bstatement*
kono
parents:
diff changeset
1246 do_get_backend(Translate_context*);
kono
parents:
diff changeset
1247
kono
parents:
diff changeset
1248 void
kono
parents:
diff changeset
1249 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1250 };
kono
parents:
diff changeset
1251
kono
parents:
diff changeset
1252 // A goto statement.
kono
parents:
diff changeset
1253
kono
parents:
diff changeset
1254 class Goto_statement : public Statement
kono
parents:
diff changeset
1255 {
kono
parents:
diff changeset
1256 public:
kono
parents:
diff changeset
1257 Goto_statement(Label* label, Location location)
kono
parents:
diff changeset
1258 : Statement(STATEMENT_GOTO, location),
kono
parents:
diff changeset
1259 label_(label)
kono
parents:
diff changeset
1260 { }
kono
parents:
diff changeset
1261
kono
parents:
diff changeset
1262 // Return the label being jumped to.
kono
parents:
diff changeset
1263 Label*
kono
parents:
diff changeset
1264 label() const
kono
parents:
diff changeset
1265 { return this->label_; }
kono
parents:
diff changeset
1266
kono
parents:
diff changeset
1267 protected:
kono
parents:
diff changeset
1268 int
kono
parents:
diff changeset
1269 do_traverse(Traverse*);
kono
parents:
diff changeset
1270
kono
parents:
diff changeset
1271 void
kono
parents:
diff changeset
1272 do_check_types(Gogo*);
kono
parents:
diff changeset
1273
kono
parents:
diff changeset
1274 bool
kono
parents:
diff changeset
1275 do_may_fall_through() const
kono
parents:
diff changeset
1276 { return false; }
kono
parents:
diff changeset
1277
kono
parents:
diff changeset
1278 Bstatement*
kono
parents:
diff changeset
1279 do_get_backend(Translate_context*);
kono
parents:
diff changeset
1280
kono
parents:
diff changeset
1281 void
kono
parents:
diff changeset
1282 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1283
kono
parents:
diff changeset
1284 private:
kono
parents:
diff changeset
1285 Label* label_;
kono
parents:
diff changeset
1286 };
kono
parents:
diff changeset
1287
kono
parents:
diff changeset
1288 // A goto statement to an unnamed label.
kono
parents:
diff changeset
1289
kono
parents:
diff changeset
1290 class Goto_unnamed_statement : public Statement
kono
parents:
diff changeset
1291 {
kono
parents:
diff changeset
1292 public:
kono
parents:
diff changeset
1293 Goto_unnamed_statement(Unnamed_label* label, Location location)
kono
parents:
diff changeset
1294 : Statement(STATEMENT_GOTO_UNNAMED, location),
kono
parents:
diff changeset
1295 label_(label)
kono
parents:
diff changeset
1296 { }
kono
parents:
diff changeset
1297
kono
parents:
diff changeset
1298 Unnamed_label*
kono
parents:
diff changeset
1299 unnamed_label() const
kono
parents:
diff changeset
1300 { return this->label_; }
kono
parents:
diff changeset
1301
kono
parents:
diff changeset
1302 protected:
kono
parents:
diff changeset
1303 int
kono
parents:
diff changeset
1304 do_traverse(Traverse*);
kono
parents:
diff changeset
1305
kono
parents:
diff changeset
1306 bool
kono
parents:
diff changeset
1307 do_may_fall_through() const
kono
parents:
diff changeset
1308 { return false; }
kono
parents:
diff changeset
1309
kono
parents:
diff changeset
1310 Bstatement*
kono
parents:
diff changeset
1311 do_get_backend(Translate_context* context);
kono
parents:
diff changeset
1312
kono
parents:
diff changeset
1313 void
kono
parents:
diff changeset
1314 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1315
kono
parents:
diff changeset
1316 private:
kono
parents:
diff changeset
1317 Unnamed_label* label_;
kono
parents:
diff changeset
1318 };
kono
parents:
diff changeset
1319
kono
parents:
diff changeset
1320 // A label statement.
kono
parents:
diff changeset
1321
kono
parents:
diff changeset
1322 class Label_statement : public Statement
kono
parents:
diff changeset
1323 {
kono
parents:
diff changeset
1324 public:
kono
parents:
diff changeset
1325 Label_statement(Label* label, Location location)
kono
parents:
diff changeset
1326 : Statement(STATEMENT_LABEL, location),
kono
parents:
diff changeset
1327 label_(label)
kono
parents:
diff changeset
1328 { }
kono
parents:
diff changeset
1329
kono
parents:
diff changeset
1330 // Return the label itself.
kono
parents:
diff changeset
1331 Label*
kono
parents:
diff changeset
1332 label() const
kono
parents:
diff changeset
1333 { return this->label_; }
kono
parents:
diff changeset
1334
kono
parents:
diff changeset
1335 protected:
kono
parents:
diff changeset
1336 int
kono
parents:
diff changeset
1337 do_traverse(Traverse*);
kono
parents:
diff changeset
1338
kono
parents:
diff changeset
1339 Bstatement*
kono
parents:
diff changeset
1340 do_get_backend(Translate_context*);
kono
parents:
diff changeset
1341
kono
parents:
diff changeset
1342 void
kono
parents:
diff changeset
1343 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1344
kono
parents:
diff changeset
1345 private:
kono
parents:
diff changeset
1346 // The label.
kono
parents:
diff changeset
1347 Label* label_;
kono
parents:
diff changeset
1348 };
kono
parents:
diff changeset
1349
kono
parents:
diff changeset
1350 // An unnamed label statement.
kono
parents:
diff changeset
1351
kono
parents:
diff changeset
1352 class Unnamed_label_statement : public Statement
kono
parents:
diff changeset
1353 {
kono
parents:
diff changeset
1354 public:
kono
parents:
diff changeset
1355 Unnamed_label_statement(Unnamed_label* label);
kono
parents:
diff changeset
1356
kono
parents:
diff changeset
1357 protected:
kono
parents:
diff changeset
1358 int
kono
parents:
diff changeset
1359 do_traverse(Traverse*);
kono
parents:
diff changeset
1360
kono
parents:
diff changeset
1361 Bstatement*
kono
parents:
diff changeset
1362 do_get_backend(Translate_context* context);
kono
parents:
diff changeset
1363
kono
parents:
diff changeset
1364 void
kono
parents:
diff changeset
1365 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1366
kono
parents:
diff changeset
1367 private:
kono
parents:
diff changeset
1368 // The label.
kono
parents:
diff changeset
1369 Unnamed_label* label_;
kono
parents:
diff changeset
1370 };
kono
parents:
diff changeset
1371
kono
parents:
diff changeset
1372 // An if statement.
kono
parents:
diff changeset
1373
kono
parents:
diff changeset
1374 class If_statement : public Statement
kono
parents:
diff changeset
1375 {
kono
parents:
diff changeset
1376 public:
kono
parents:
diff changeset
1377 If_statement(Expression* cond, Block* then_block, Block* else_block,
kono
parents:
diff changeset
1378 Location location)
kono
parents:
diff changeset
1379 : Statement(STATEMENT_IF, location),
kono
parents:
diff changeset
1380 cond_(cond), then_block_(then_block), else_block_(else_block)
kono
parents:
diff changeset
1381 { }
kono
parents:
diff changeset
1382
kono
parents:
diff changeset
1383 Expression*
kono
parents:
diff changeset
1384 condition() const
kono
parents:
diff changeset
1385 { return this->cond_; }
kono
parents:
diff changeset
1386
kono
parents:
diff changeset
1387 protected:
kono
parents:
diff changeset
1388 int
kono
parents:
diff changeset
1389 do_traverse(Traverse*);
kono
parents:
diff changeset
1390
kono
parents:
diff changeset
1391 void
kono
parents:
diff changeset
1392 do_determine_types();
kono
parents:
diff changeset
1393
kono
parents:
diff changeset
1394 void
kono
parents:
diff changeset
1395 do_check_types(Gogo*);
kono
parents:
diff changeset
1396
kono
parents:
diff changeset
1397 bool
kono
parents:
diff changeset
1398 do_may_fall_through() const;
kono
parents:
diff changeset
1399
kono
parents:
diff changeset
1400 Bstatement*
kono
parents:
diff changeset
1401 do_get_backend(Translate_context*);
kono
parents:
diff changeset
1402
kono
parents:
diff changeset
1403 void
kono
parents:
diff changeset
1404 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1405
kono
parents:
diff changeset
1406 private:
kono
parents:
diff changeset
1407 Expression* cond_;
kono
parents:
diff changeset
1408 Block* then_block_;
kono
parents:
diff changeset
1409 Block* else_block_;
kono
parents:
diff changeset
1410 };
kono
parents:
diff changeset
1411
kono
parents:
diff changeset
1412 // A for statement.
kono
parents:
diff changeset
1413
kono
parents:
diff changeset
1414 class For_statement : public Statement
kono
parents:
diff changeset
1415 {
kono
parents:
diff changeset
1416 public:
kono
parents:
diff changeset
1417 For_statement(Block* init, Expression* cond, Block* post,
kono
parents:
diff changeset
1418 Location location)
kono
parents:
diff changeset
1419 : Statement(STATEMENT_FOR, location),
kono
parents:
diff changeset
1420 init_(init), cond_(cond), post_(post), statements_(NULL),
kono
parents:
diff changeset
1421 break_label_(NULL), continue_label_(NULL)
kono
parents:
diff changeset
1422 { }
kono
parents:
diff changeset
1423
kono
parents:
diff changeset
1424 // Add the statements.
kono
parents:
diff changeset
1425 void
kono
parents:
diff changeset
1426 add_statements(Block* statements)
kono
parents:
diff changeset
1427 {
kono
parents:
diff changeset
1428 go_assert(this->statements_ == NULL);
kono
parents:
diff changeset
1429 this->statements_ = statements;
kono
parents:
diff changeset
1430 }
kono
parents:
diff changeset
1431
kono
parents:
diff changeset
1432 // Return the break label for this for statement.
kono
parents:
diff changeset
1433 Unnamed_label*
kono
parents:
diff changeset
1434 break_label();
kono
parents:
diff changeset
1435
kono
parents:
diff changeset
1436 // Return the continue label for this for statement.
kono
parents:
diff changeset
1437 Unnamed_label*
kono
parents:
diff changeset
1438 continue_label();
kono
parents:
diff changeset
1439
kono
parents:
diff changeset
1440 // Set the break and continue labels for this statement.
kono
parents:
diff changeset
1441 void
kono
parents:
diff changeset
1442 set_break_continue_labels(Unnamed_label* break_label,
kono
parents:
diff changeset
1443 Unnamed_label* continue_label);
kono
parents:
diff changeset
1444
kono
parents:
diff changeset
1445 protected:
kono
parents:
diff changeset
1446 int
kono
parents:
diff changeset
1447 do_traverse(Traverse*);
kono
parents:
diff changeset
1448
kono
parents:
diff changeset
1449 bool
kono
parents:
diff changeset
1450 do_traverse_assignments(Traverse_assignments*)
kono
parents:
diff changeset
1451 { go_unreachable(); }
kono
parents:
diff changeset
1452
kono
parents:
diff changeset
1453 Statement*
kono
parents:
diff changeset
1454 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
1455
kono
parents:
diff changeset
1456 bool
kono
parents:
diff changeset
1457 do_may_fall_through() const;
kono
parents:
diff changeset
1458
kono
parents:
diff changeset
1459 Bstatement*
kono
parents:
diff changeset
1460 do_get_backend(Translate_context*)
kono
parents:
diff changeset
1461 { go_unreachable(); }
kono
parents:
diff changeset
1462
kono
parents:
diff changeset
1463 void
kono
parents:
diff changeset
1464 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1465
kono
parents:
diff changeset
1466 private:
kono
parents:
diff changeset
1467 // The initialization statements. This may be NULL.
kono
parents:
diff changeset
1468 Block* init_;
kono
parents:
diff changeset
1469 // The condition. This may be NULL.
kono
parents:
diff changeset
1470 Expression* cond_;
kono
parents:
diff changeset
1471 // The statements to run after each iteration. This may be NULL.
kono
parents:
diff changeset
1472 Block* post_;
kono
parents:
diff changeset
1473 // The statements in the loop itself.
kono
parents:
diff changeset
1474 Block* statements_;
kono
parents:
diff changeset
1475 // The break label, if needed.
kono
parents:
diff changeset
1476 Unnamed_label* break_label_;
kono
parents:
diff changeset
1477 // The continue label, if needed.
kono
parents:
diff changeset
1478 Unnamed_label* continue_label_;
kono
parents:
diff changeset
1479 };
kono
parents:
diff changeset
1480
kono
parents:
diff changeset
1481 // A for statement over a range clause.
kono
parents:
diff changeset
1482
kono
parents:
diff changeset
1483 class For_range_statement : public Statement
kono
parents:
diff changeset
1484 {
kono
parents:
diff changeset
1485 public:
kono
parents:
diff changeset
1486 For_range_statement(Expression* index_var, Expression* value_var,
kono
parents:
diff changeset
1487 Expression* range, Location location)
kono
parents:
diff changeset
1488 : Statement(STATEMENT_FOR_RANGE, location),
kono
parents:
diff changeset
1489 index_var_(index_var), value_var_(value_var), range_(range),
kono
parents:
diff changeset
1490 statements_(NULL), break_label_(NULL), continue_label_(NULL)
kono
parents:
diff changeset
1491 { }
kono
parents:
diff changeset
1492
kono
parents:
diff changeset
1493 // Add the statements.
kono
parents:
diff changeset
1494 void
kono
parents:
diff changeset
1495 add_statements(Block* statements)
kono
parents:
diff changeset
1496 {
kono
parents:
diff changeset
1497 go_assert(this->statements_ == NULL);
kono
parents:
diff changeset
1498 this->statements_ = statements;
kono
parents:
diff changeset
1499 }
kono
parents:
diff changeset
1500
kono
parents:
diff changeset
1501 // Return the break label for this for statement.
kono
parents:
diff changeset
1502 Unnamed_label*
kono
parents:
diff changeset
1503 break_label();
kono
parents:
diff changeset
1504
kono
parents:
diff changeset
1505 // Return the continue label for this for statement.
kono
parents:
diff changeset
1506 Unnamed_label*
kono
parents:
diff changeset
1507 continue_label();
kono
parents:
diff changeset
1508
kono
parents:
diff changeset
1509 protected:
kono
parents:
diff changeset
1510 int
kono
parents:
diff changeset
1511 do_traverse(Traverse*);
kono
parents:
diff changeset
1512
kono
parents:
diff changeset
1513 bool
kono
parents:
diff changeset
1514 do_traverse_assignments(Traverse_assignments*)
kono
parents:
diff changeset
1515 { go_unreachable(); }
kono
parents:
diff changeset
1516
kono
parents:
diff changeset
1517 Statement*
kono
parents:
diff changeset
1518 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
1519
kono
parents:
diff changeset
1520 Bstatement*
kono
parents:
diff changeset
1521 do_get_backend(Translate_context*)
kono
parents:
diff changeset
1522 { go_unreachable(); }
kono
parents:
diff changeset
1523
kono
parents:
diff changeset
1524 void
kono
parents:
diff changeset
1525 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1526
kono
parents:
diff changeset
1527 private:
kono
parents:
diff changeset
1528 Expression*
kono
parents:
diff changeset
1529 make_range_ref(Named_object*, Temporary_statement*, Location);
kono
parents:
diff changeset
1530
kono
parents:
diff changeset
1531 Call_expression*
kono
parents:
diff changeset
1532 call_builtin(Gogo*, const char* funcname, Expression* arg, Location);
kono
parents:
diff changeset
1533
kono
parents:
diff changeset
1534 void
kono
parents:
diff changeset
1535 lower_range_array(Gogo*, Block*, Block*, Named_object*, Temporary_statement*,
kono
parents:
diff changeset
1536 Temporary_statement*, Temporary_statement*,
kono
parents:
diff changeset
1537 Block**, Expression**, Block**, Block**);
kono
parents:
diff changeset
1538
kono
parents:
diff changeset
1539 void
kono
parents:
diff changeset
1540 lower_range_slice(Gogo*, Block*, Block*, Named_object*, Temporary_statement*,
kono
parents:
diff changeset
1541 Temporary_statement*, Temporary_statement*,
kono
parents:
diff changeset
1542 Block**, Expression**, Block**, Block**);
kono
parents:
diff changeset
1543
kono
parents:
diff changeset
1544 void
kono
parents:
diff changeset
1545 lower_range_string(Gogo*, Block*, Block*, Named_object*, Temporary_statement*,
kono
parents:
diff changeset
1546 Temporary_statement*, Temporary_statement*,
kono
parents:
diff changeset
1547 Block**, Expression**, Block**, Block**);
kono
parents:
diff changeset
1548
kono
parents:
diff changeset
1549 void
kono
parents:
diff changeset
1550 lower_range_map(Gogo*, Map_type*, Block*, Block*, Named_object*,
kono
parents:
diff changeset
1551 Temporary_statement*, Temporary_statement*,
kono
parents:
diff changeset
1552 Temporary_statement*, Block**, Expression**, Block**,
kono
parents:
diff changeset
1553 Block**);
kono
parents:
diff changeset
1554
kono
parents:
diff changeset
1555 void
kono
parents:
diff changeset
1556 lower_range_channel(Gogo*, Block*, Block*, Named_object*,
kono
parents:
diff changeset
1557 Temporary_statement*, Temporary_statement*,
kono
parents:
diff changeset
1558 Temporary_statement*, Block**, Expression**, Block**,
kono
parents:
diff changeset
1559 Block**);
kono
parents:
diff changeset
1560
kono
parents:
diff changeset
1561 // The variable which is set to the index value.
kono
parents:
diff changeset
1562 Expression* index_var_;
kono
parents:
diff changeset
1563 // The variable which is set to the element value. This may be
kono
parents:
diff changeset
1564 // NULL.
kono
parents:
diff changeset
1565 Expression* value_var_;
kono
parents:
diff changeset
1566 // The expression we are ranging over.
kono
parents:
diff changeset
1567 Expression* range_;
kono
parents:
diff changeset
1568 // The statements in the block.
kono
parents:
diff changeset
1569 Block* statements_;
kono
parents:
diff changeset
1570 // The break label, if needed.
kono
parents:
diff changeset
1571 Unnamed_label* break_label_;
kono
parents:
diff changeset
1572 // The continue label, if needed.
kono
parents:
diff changeset
1573 Unnamed_label* continue_label_;
kono
parents:
diff changeset
1574 };
kono
parents:
diff changeset
1575
kono
parents:
diff changeset
1576 // Class Case_clauses holds the clauses of a switch statement. This
kono
parents:
diff changeset
1577 // is built by the parser.
kono
parents:
diff changeset
1578
kono
parents:
diff changeset
1579 class Case_clauses
kono
parents:
diff changeset
1580 {
kono
parents:
diff changeset
1581 public:
kono
parents:
diff changeset
1582 Case_clauses()
kono
parents:
diff changeset
1583 : clauses_()
kono
parents:
diff changeset
1584 { }
kono
parents:
diff changeset
1585
kono
parents:
diff changeset
1586 // Add a new clause. CASES is a list of case expressions; it may be
kono
parents:
diff changeset
1587 // NULL. IS_DEFAULT is true if this is the default case.
kono
parents:
diff changeset
1588 // STATEMENTS is a block of statements. IS_FALLTHROUGH is true if
kono
parents:
diff changeset
1589 // after the statements the case clause should fall through to the
kono
parents:
diff changeset
1590 // next clause.
kono
parents:
diff changeset
1591 void
kono
parents:
diff changeset
1592 add(Expression_list* cases, bool is_default, Block* statements,
kono
parents:
diff changeset
1593 bool is_fallthrough, Location location)
kono
parents:
diff changeset
1594 {
kono
parents:
diff changeset
1595 this->clauses_.push_back(Case_clause(cases, is_default, statements,
kono
parents:
diff changeset
1596 is_fallthrough, location));
kono
parents:
diff changeset
1597 }
kono
parents:
diff changeset
1598
kono
parents:
diff changeset
1599 // Return whether there are no clauses.
kono
parents:
diff changeset
1600 bool
kono
parents:
diff changeset
1601 empty() const
kono
parents:
diff changeset
1602 { return this->clauses_.empty(); }
kono
parents:
diff changeset
1603
kono
parents:
diff changeset
1604 // Traverse the case clauses.
kono
parents:
diff changeset
1605 int
kono
parents:
diff changeset
1606 traverse(Traverse*);
kono
parents:
diff changeset
1607
kono
parents:
diff changeset
1608 // Lower for a nonconstant switch.
kono
parents:
diff changeset
1609 void
kono
parents:
diff changeset
1610 lower(Block*, Temporary_statement*, Unnamed_label*) const;
kono
parents:
diff changeset
1611
kono
parents:
diff changeset
1612 // Determine types of expressions. The Type parameter is the type
kono
parents:
diff changeset
1613 // of the switch value.
kono
parents:
diff changeset
1614 void
kono
parents:
diff changeset
1615 determine_types(Type*);
kono
parents:
diff changeset
1616
kono
parents:
diff changeset
1617 // Check types. The Type parameter is the type of the switch value.
kono
parents:
diff changeset
1618 bool
kono
parents:
diff changeset
1619 check_types(Type*);
kono
parents:
diff changeset
1620
kono
parents:
diff changeset
1621 // Return true if all the clauses are constant values.
kono
parents:
diff changeset
1622 bool
kono
parents:
diff changeset
1623 is_constant() const;
kono
parents:
diff changeset
1624
kono
parents:
diff changeset
1625 // Return true if these clauses may fall through to the statements
kono
parents:
diff changeset
1626 // following the switch statement.
kono
parents:
diff changeset
1627 bool
kono
parents:
diff changeset
1628 may_fall_through() const;
kono
parents:
diff changeset
1629
kono
parents:
diff changeset
1630 // Return the body of a SWITCH_EXPR when all the clauses are
kono
parents:
diff changeset
1631 // constants.
kono
parents:
diff changeset
1632 void
kono
parents:
diff changeset
1633 get_backend(Translate_context*, Unnamed_label* break_label,
kono
parents:
diff changeset
1634 std::vector<std::vector<Bexpression*> >* all_cases,
kono
parents:
diff changeset
1635 std::vector<Bstatement*>* all_statements) const;
kono
parents:
diff changeset
1636
kono
parents:
diff changeset
1637 // Dump the AST representation to a dump context.
kono
parents:
diff changeset
1638 void
kono
parents:
diff changeset
1639 dump_clauses(Ast_dump_context*) const;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1640
111
kono
parents:
diff changeset
1641 private:
kono
parents:
diff changeset
1642 // For a constant switch we need to keep a record of constants we
kono
parents:
diff changeset
1643 // have already seen.
kono
parents:
diff changeset
1644 class Hash_integer_value;
kono
parents:
diff changeset
1645 class Eq_integer_value;
kono
parents:
diff changeset
1646 typedef Unordered_set_hash(Expression*, Hash_integer_value,
kono
parents:
diff changeset
1647 Eq_integer_value) Case_constants;
kono
parents:
diff changeset
1648
kono
parents:
diff changeset
1649 // One case clause.
kono
parents:
diff changeset
1650 class Case_clause
kono
parents:
diff changeset
1651 {
kono
parents:
diff changeset
1652 public:
kono
parents:
diff changeset
1653 Case_clause()
kono
parents:
diff changeset
1654 : cases_(NULL), statements_(NULL), is_default_(false),
kono
parents:
diff changeset
1655 is_fallthrough_(false), location_(Linemap::unknown_location())
kono
parents:
diff changeset
1656 { }
kono
parents:
diff changeset
1657
kono
parents:
diff changeset
1658 Case_clause(Expression_list* cases, bool is_default, Block* statements,
kono
parents:
diff changeset
1659 bool is_fallthrough, Location location)
kono
parents:
diff changeset
1660 : cases_(cases), statements_(statements), is_default_(is_default),
kono
parents:
diff changeset
1661 is_fallthrough_(is_fallthrough), location_(location)
kono
parents:
diff changeset
1662 { }
kono
parents:
diff changeset
1663
kono
parents:
diff changeset
1664 // Whether this clause falls through to the next clause.
kono
parents:
diff changeset
1665 bool
kono
parents:
diff changeset
1666 is_fallthrough() const
kono
parents:
diff changeset
1667 { return this->is_fallthrough_; }
kono
parents:
diff changeset
1668
kono
parents:
diff changeset
1669 // Whether this is the default.
kono
parents:
diff changeset
1670 bool
kono
parents:
diff changeset
1671 is_default() const
kono
parents:
diff changeset
1672 { return this->is_default_; }
kono
parents:
diff changeset
1673
kono
parents:
diff changeset
1674 // The location of this clause.
kono
parents:
diff changeset
1675 Location
kono
parents:
diff changeset
1676 location() const
kono
parents:
diff changeset
1677 { return this->location_; }
kono
parents:
diff changeset
1678
kono
parents:
diff changeset
1679 // Traversal.
kono
parents:
diff changeset
1680 int
kono
parents:
diff changeset
1681 traverse(Traverse*);
kono
parents:
diff changeset
1682
kono
parents:
diff changeset
1683 // Lower for a nonconstant switch.
kono
parents:
diff changeset
1684 void
kono
parents:
diff changeset
1685 lower(Block*, Temporary_statement*, Unnamed_label*, Unnamed_label*) const;
kono
parents:
diff changeset
1686
kono
parents:
diff changeset
1687 // Determine types.
kono
parents:
diff changeset
1688 void
kono
parents:
diff changeset
1689 determine_types(Type*);
kono
parents:
diff changeset
1690
kono
parents:
diff changeset
1691 // Check types.
kono
parents:
diff changeset
1692 bool
kono
parents:
diff changeset
1693 check_types(Type*);
kono
parents:
diff changeset
1694
kono
parents:
diff changeset
1695 // Return true if all the case expressions are constant.
kono
parents:
diff changeset
1696 bool
kono
parents:
diff changeset
1697 is_constant() const;
kono
parents:
diff changeset
1698
kono
parents:
diff changeset
1699 // Return true if this clause may fall through to execute the
kono
parents:
diff changeset
1700 // statements following the switch statement. This is not the
kono
parents:
diff changeset
1701 // same as whether this clause falls through to the next clause.
kono
parents:
diff changeset
1702 bool
kono
parents:
diff changeset
1703 may_fall_through() const;
kono
parents:
diff changeset
1704
kono
parents:
diff changeset
1705 // Convert the case values and statements to the backend
kono
parents:
diff changeset
1706 // representation.
kono
parents:
diff changeset
1707 Bstatement*
kono
parents:
diff changeset
1708 get_backend(Translate_context*, Unnamed_label* break_label,
kono
parents:
diff changeset
1709 Case_constants*, std::vector<Bexpression*>* cases) const;
kono
parents:
diff changeset
1710
kono
parents:
diff changeset
1711 // Dump the AST representation to a dump context.
kono
parents:
diff changeset
1712 void
kono
parents:
diff changeset
1713 dump_clause(Ast_dump_context*) const;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1714
111
kono
parents:
diff changeset
1715 private:
kono
parents:
diff changeset
1716 // The list of case expressions.
kono
parents:
diff changeset
1717 Expression_list* cases_;
kono
parents:
diff changeset
1718 // The statements to execute.
kono
parents:
diff changeset
1719 Block* statements_;
kono
parents:
diff changeset
1720 // Whether this is the default case.
kono
parents:
diff changeset
1721 bool is_default_;
kono
parents:
diff changeset
1722 // Whether this falls through after the statements.
kono
parents:
diff changeset
1723 bool is_fallthrough_;
kono
parents:
diff changeset
1724 // The location of this case clause.
kono
parents:
diff changeset
1725 Location location_;
kono
parents:
diff changeset
1726 };
kono
parents:
diff changeset
1727
kono
parents:
diff changeset
1728 friend class Case_clause;
kono
parents:
diff changeset
1729
kono
parents:
diff changeset
1730 // The type of the list of clauses.
kono
parents:
diff changeset
1731 typedef std::vector<Case_clause> Clauses;
kono
parents:
diff changeset
1732
kono
parents:
diff changeset
1733 // All the case clauses.
kono
parents:
diff changeset
1734 Clauses clauses_;
kono
parents:
diff changeset
1735 };
kono
parents:
diff changeset
1736
kono
parents:
diff changeset
1737 // A switch statement.
kono
parents:
diff changeset
1738
kono
parents:
diff changeset
1739 class Switch_statement : public Statement
kono
parents:
diff changeset
1740 {
kono
parents:
diff changeset
1741 public:
kono
parents:
diff changeset
1742 Switch_statement(Expression* val, Location location)
kono
parents:
diff changeset
1743 : Statement(STATEMENT_SWITCH, location),
kono
parents:
diff changeset
1744 val_(val), clauses_(NULL), break_label_(NULL)
kono
parents:
diff changeset
1745 { }
kono
parents:
diff changeset
1746
kono
parents:
diff changeset
1747 // Add the clauses.
kono
parents:
diff changeset
1748 void
kono
parents:
diff changeset
1749 add_clauses(Case_clauses* clauses)
kono
parents:
diff changeset
1750 {
kono
parents:
diff changeset
1751 go_assert(this->clauses_ == NULL);
kono
parents:
diff changeset
1752 this->clauses_ = clauses;
kono
parents:
diff changeset
1753 }
kono
parents:
diff changeset
1754
kono
parents:
diff changeset
1755 // Return the break label for this switch statement.
kono
parents:
diff changeset
1756 Unnamed_label*
kono
parents:
diff changeset
1757 break_label();
kono
parents:
diff changeset
1758
kono
parents:
diff changeset
1759 protected:
kono
parents:
diff changeset
1760 int
kono
parents:
diff changeset
1761 do_traverse(Traverse*);
kono
parents:
diff changeset
1762
kono
parents:
diff changeset
1763 Statement*
kono
parents:
diff changeset
1764 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
1765
kono
parents:
diff changeset
1766 Bstatement*
kono
parents:
diff changeset
1767 do_get_backend(Translate_context*)
kono
parents:
diff changeset
1768 { go_unreachable(); }
kono
parents:
diff changeset
1769
kono
parents:
diff changeset
1770 void
kono
parents:
diff changeset
1771 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1772
kono
parents:
diff changeset
1773 bool
kono
parents:
diff changeset
1774 do_may_fall_through() const;
kono
parents:
diff changeset
1775
kono
parents:
diff changeset
1776 private:
kono
parents:
diff changeset
1777 // The value to switch on. This may be NULL.
kono
parents:
diff changeset
1778 Expression* val_;
kono
parents:
diff changeset
1779 // The case clauses.
kono
parents:
diff changeset
1780 Case_clauses* clauses_;
kono
parents:
diff changeset
1781 // The break label, if needed.
kono
parents:
diff changeset
1782 Unnamed_label* break_label_;
kono
parents:
diff changeset
1783 };
kono
parents:
diff changeset
1784
kono
parents:
diff changeset
1785 // Class Type_case_clauses holds the clauses of a type switch
kono
parents:
diff changeset
1786 // statement. This is built by the parser.
kono
parents:
diff changeset
1787
kono
parents:
diff changeset
1788 class Type_case_clauses
kono
parents:
diff changeset
1789 {
kono
parents:
diff changeset
1790 public:
kono
parents:
diff changeset
1791 Type_case_clauses()
kono
parents:
diff changeset
1792 : clauses_()
kono
parents:
diff changeset
1793 { }
kono
parents:
diff changeset
1794
kono
parents:
diff changeset
1795 // Add a new clause. TYPE is the type for this clause; it may be
kono
parents:
diff changeset
1796 // NULL. IS_FALLTHROUGH is true if this falls through to the next
kono
parents:
diff changeset
1797 // clause; in this case STATEMENTS will be NULL. IS_DEFAULT is true
kono
parents:
diff changeset
1798 // if this is the default case. STATEMENTS is a block of
kono
parents:
diff changeset
1799 // statements; it may be NULL.
kono
parents:
diff changeset
1800 void
kono
parents:
diff changeset
1801 add(Type* type, bool is_fallthrough, bool is_default, Block* statements,
kono
parents:
diff changeset
1802 Location location)
kono
parents:
diff changeset
1803 {
kono
parents:
diff changeset
1804 this->clauses_.push_back(Type_case_clause(type, is_fallthrough, is_default,
kono
parents:
diff changeset
1805 statements, location));
kono
parents:
diff changeset
1806 }
kono
parents:
diff changeset
1807
kono
parents:
diff changeset
1808 // Return whether there are no clauses.
kono
parents:
diff changeset
1809 bool
kono
parents:
diff changeset
1810 empty() const
kono
parents:
diff changeset
1811 { return this->clauses_.empty(); }
kono
parents:
diff changeset
1812
kono
parents:
diff changeset
1813 // Traverse the type case clauses.
kono
parents:
diff changeset
1814 int
kono
parents:
diff changeset
1815 traverse(Traverse*);
kono
parents:
diff changeset
1816
kono
parents:
diff changeset
1817 // Check for duplicates.
kono
parents:
diff changeset
1818 void
kono
parents:
diff changeset
1819 check_duplicates() const;
kono
parents:
diff changeset
1820
kono
parents:
diff changeset
1821 // Lower to if and goto statements.
kono
parents:
diff changeset
1822 void
kono
parents:
diff changeset
1823 lower(Type*, Block*, Temporary_statement* descriptor_temp,
kono
parents:
diff changeset
1824 Unnamed_label* break_label) const;
kono
parents:
diff changeset
1825
kono
parents:
diff changeset
1826 // Return true if these clauses may fall through to the statements
kono
parents:
diff changeset
1827 // following the switch statement.
kono
parents:
diff changeset
1828 bool
kono
parents:
diff changeset
1829 may_fall_through() const;
kono
parents:
diff changeset
1830
kono
parents:
diff changeset
1831 // Dump the AST representation to a dump context.
kono
parents:
diff changeset
1832 void
kono
parents:
diff changeset
1833 dump_clauses(Ast_dump_context*) const;
kono
parents:
diff changeset
1834
kono
parents:
diff changeset
1835 private:
kono
parents:
diff changeset
1836 // One type case clause.
kono
parents:
diff changeset
1837 class Type_case_clause
kono
parents:
diff changeset
1838 {
kono
parents:
diff changeset
1839 public:
kono
parents:
diff changeset
1840 Type_case_clause()
kono
parents:
diff changeset
1841 : type_(NULL), statements_(NULL), is_default_(false),
kono
parents:
diff changeset
1842 location_(Linemap::unknown_location())
kono
parents:
diff changeset
1843 { }
kono
parents:
diff changeset
1844
kono
parents:
diff changeset
1845 Type_case_clause(Type* type, bool is_fallthrough, bool is_default,
kono
parents:
diff changeset
1846 Block* statements, Location location)
kono
parents:
diff changeset
1847 : type_(type), statements_(statements), is_fallthrough_(is_fallthrough),
kono
parents:
diff changeset
1848 is_default_(is_default), location_(location)
kono
parents:
diff changeset
1849 { }
kono
parents:
diff changeset
1850
kono
parents:
diff changeset
1851 // The type.
kono
parents:
diff changeset
1852 Type*
kono
parents:
diff changeset
1853 type() const
kono
parents:
diff changeset
1854 { return this->type_; }
kono
parents:
diff changeset
1855
kono
parents:
diff changeset
1856 // Whether this is the default.
kono
parents:
diff changeset
1857 bool
kono
parents:
diff changeset
1858 is_default() const
kono
parents:
diff changeset
1859 { return this->is_default_; }
kono
parents:
diff changeset
1860
kono
parents:
diff changeset
1861 // The location of this type clause.
kono
parents:
diff changeset
1862 Location
kono
parents:
diff changeset
1863 location() const
kono
parents:
diff changeset
1864 { return this->location_; }
kono
parents:
diff changeset
1865
kono
parents:
diff changeset
1866 // Traversal.
kono
parents:
diff changeset
1867 int
kono
parents:
diff changeset
1868 traverse(Traverse*);
kono
parents:
diff changeset
1869
kono
parents:
diff changeset
1870 // Lower to if and goto statements.
kono
parents:
diff changeset
1871 void
kono
parents:
diff changeset
1872 lower(Type*, Block*, Temporary_statement* descriptor_temp,
kono
parents:
diff changeset
1873 Unnamed_label* break_label, Unnamed_label** stmts_label) const;
kono
parents:
diff changeset
1874
kono
parents:
diff changeset
1875 // Return true if this clause may fall through to execute the
kono
parents:
diff changeset
1876 // statements following the switch statement. This is not the
kono
parents:
diff changeset
1877 // same as whether this clause falls through to the next clause.
kono
parents:
diff changeset
1878 bool
kono
parents:
diff changeset
1879 may_fall_through() const;
kono
parents:
diff changeset
1880
kono
parents:
diff changeset
1881 // Dump the AST representation to a dump context.
kono
parents:
diff changeset
1882 void
kono
parents:
diff changeset
1883 dump_clause(Ast_dump_context*) const;
kono
parents:
diff changeset
1884
kono
parents:
diff changeset
1885 private:
kono
parents:
diff changeset
1886 // The type for this type clause.
kono
parents:
diff changeset
1887 Type* type_;
kono
parents:
diff changeset
1888 // The statements to execute.
kono
parents:
diff changeset
1889 Block* statements_;
kono
parents:
diff changeset
1890 // Whether this falls through--this is true for "case T1, T2".
kono
parents:
diff changeset
1891 bool is_fallthrough_;
kono
parents:
diff changeset
1892 // Whether this is the default case.
kono
parents:
diff changeset
1893 bool is_default_;
kono
parents:
diff changeset
1894 // The location of this type case clause.
kono
parents:
diff changeset
1895 Location location_;
kono
parents:
diff changeset
1896 };
kono
parents:
diff changeset
1897
kono
parents:
diff changeset
1898 friend class Type_case_clause;
kono
parents:
diff changeset
1899
kono
parents:
diff changeset
1900 // The type of the list of type clauses.
kono
parents:
diff changeset
1901 typedef std::vector<Type_case_clause> Type_clauses;
kono
parents:
diff changeset
1902
kono
parents:
diff changeset
1903 // All the type case clauses.
kono
parents:
diff changeset
1904 Type_clauses clauses_;
kono
parents:
diff changeset
1905 };
kono
parents:
diff changeset
1906
kono
parents:
diff changeset
1907 // A type switch statement.
kono
parents:
diff changeset
1908
kono
parents:
diff changeset
1909 class Type_switch_statement : public Statement
kono
parents:
diff changeset
1910 {
kono
parents:
diff changeset
1911 public:
kono
parents:
diff changeset
1912 Type_switch_statement(const std::string& name, Expression* expr,
kono
parents:
diff changeset
1913 Location location)
kono
parents:
diff changeset
1914 : Statement(STATEMENT_TYPE_SWITCH, location),
kono
parents:
diff changeset
1915 name_(name), expr_(expr), clauses_(NULL), break_label_(NULL)
kono
parents:
diff changeset
1916 { }
kono
parents:
diff changeset
1917
kono
parents:
diff changeset
1918 // Add the clauses.
kono
parents:
diff changeset
1919 void
kono
parents:
diff changeset
1920 add_clauses(Type_case_clauses* clauses)
kono
parents:
diff changeset
1921 {
kono
parents:
diff changeset
1922 go_assert(this->clauses_ == NULL);
kono
parents:
diff changeset
1923 this->clauses_ = clauses;
kono
parents:
diff changeset
1924 }
kono
parents:
diff changeset
1925
kono
parents:
diff changeset
1926 // Return the break label for this type switch statement.
kono
parents:
diff changeset
1927 Unnamed_label*
kono
parents:
diff changeset
1928 break_label();
kono
parents:
diff changeset
1929
kono
parents:
diff changeset
1930 protected:
kono
parents:
diff changeset
1931 int
kono
parents:
diff changeset
1932 do_traverse(Traverse*);
kono
parents:
diff changeset
1933
kono
parents:
diff changeset
1934 Statement*
kono
parents:
diff changeset
1935 do_lower(Gogo*, Named_object*, Block*, Statement_inserter*);
kono
parents:
diff changeset
1936
kono
parents:
diff changeset
1937 Bstatement*
kono
parents:
diff changeset
1938 do_get_backend(Translate_context*)
kono
parents:
diff changeset
1939 { go_unreachable(); }
kono
parents:
diff changeset
1940
kono
parents:
diff changeset
1941 void
kono
parents:
diff changeset
1942 do_dump_statement(Ast_dump_context*) const;
kono
parents:
diff changeset
1943
kono
parents:
diff changeset
1944 bool
kono
parents:
diff changeset
1945 do_may_fall_through() const;
kono
parents:
diff changeset
1946
kono
parents:
diff changeset
1947 private:
kono
parents:
diff changeset
1948 // The name of the variable declared in the type switch guard. Empty if there
kono
parents:
diff changeset
1949 // is no variable declared.
kono
parents:
diff changeset
1950 std::string name_;
kono
parents:
diff changeset
1951 // The expression we are switching on if there is no variable.
kono
parents:
diff changeset
1952 Expression* expr_;
kono
parents:
diff changeset
1953 // The type case clauses.
kono
parents:
diff changeset
1954 Type_case_clauses* clauses_;
kono
parents:
diff changeset
1955 // The break label, if needed.
kono
parents:
diff changeset
1956 Unnamed_label* break_label_;
kono
parents:
diff changeset
1957 };
kono
parents:
diff changeset
1958
kono
parents:
diff changeset
1959 #endif // !defined(GO_STATEMENTS_H)