File: | jdk/src/hotspot/share/adlc/adlparse.cpp |
Warning: | line 3159, column 15 Value stored to 'encode_class' during its initialization is never read |
Press '?' to see keyboard shortcuts
Keyboard shortcuts:
1 | /* |
2 | * Copyright (c) 1997, 2020, Oracle and/or its affiliates. All rights reserved. |
3 | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
4 | * |
5 | * This code is free software; you can redistribute it and/or modify it |
6 | * under the terms of the GNU General Public License version 2 only, as |
7 | * published by the Free Software Foundation. |
8 | * |
9 | * This code is distributed in the hope that it will be useful, but WITHOUT |
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
12 | * version 2 for more details (a copy is included in the LICENSE file that |
13 | * accompanied this code). |
14 | * |
15 | * You should have received a copy of the GNU General Public License version |
16 | * 2 along with this work; if not, write to the Free Software Foundation, |
17 | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
18 | * |
19 | * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
20 | * or visit www.oracle.com if you need additional information or have any |
21 | * questions. |
22 | * |
23 | */ |
24 | |
25 | // ADLPARSE.CPP - Architecture Description Language Parser |
26 | // Authors: Chris Vick and Mike Paleczny |
27 | #include "adlc.hpp" |
28 | |
29 | //----------------------------ADLParser---------------------------------------- |
30 | // Create a new ADL parser |
31 | ADLParser::ADLParser(FileBuff& buffer, ArchDesc& archDesc) |
32 | : _buf(buffer), _AD(archDesc), |
33 | _globalNames(archDesc.globalNames()) { |
34 | _AD._syntax_errs = _AD._semantic_errs = 0; // No errors so far this file |
35 | _AD._warnings = 0; // No warnings either |
36 | _curline = _ptr = NULL__null; // No pointers into buffer yet |
37 | |
38 | _preproc_depth = 0; |
39 | _preproc_not_taken = 0; |
40 | |
41 | // Delimit command-line definitions from in-file definitions: |
42 | _AD._preproc_list.add_signal(); |
43 | } |
44 | |
45 | //------------------------------~ADLParser------------------------------------- |
46 | // Delete an ADL parser. |
47 | ADLParser::~ADLParser() { |
48 | if (!_AD._quiet_mode) |
49 | fprintf(stderrstderr,"---------------------------- Errors and Warnings ----------------------------\n"); |
50 | #ifndef ASSERT1 |
51 | if (!_AD._quiet_mode) { |
52 | fprintf(stderrstderr, "**************************************************************\n"); |
53 | fprintf(stderrstderr, "***** WARNING: ASSERT is undefined, assertions disabled. *****\n"); |
54 | fprintf(stderrstderr, "**************************************************************\n"); |
55 | } |
56 | #endif |
57 | if( _AD._syntax_errs + _AD._semantic_errs + _AD._warnings == 0 ) { |
58 | if (!_AD._quiet_mode) |
59 | fprintf(stderrstderr,"No errors or warnings to report from phase-1 parse.\n" ); |
60 | } |
61 | else { |
62 | if( _AD._syntax_errs ) { // Any syntax errors? |
63 | fprintf(stderrstderr,"%s: Found %d syntax error", _buf._fp->_name, _AD._syntax_errs); |
64 | if( _AD._syntax_errs > 1 ) fprintf(stderrstderr,"s.\n\n"); |
65 | else fprintf(stderrstderr,".\n\n"); |
66 | } |
67 | if( _AD._semantic_errs ) { // Any semantic errors? |
68 | fprintf(stderrstderr,"%s: Found %d semantic error", _buf._fp->_name, _AD._semantic_errs); |
69 | if( _AD._semantic_errs > 1 ) fprintf(stderrstderr,"s.\n\n"); |
70 | else fprintf(stderrstderr,".\n\n"); |
71 | } |
72 | if( _AD._warnings ) { // Any warnings? |
73 | fprintf(stderrstderr,"%s: Found %d warning", _buf._fp->_name, _AD._warnings); |
74 | if( _AD._warnings > 1 ) fprintf(stderrstderr,"s.\n\n"); |
75 | else fprintf(stderrstderr,".\n\n"); |
76 | } |
77 | } |
78 | if (!_AD._quiet_mode) |
79 | fprintf(stderrstderr,"-----------------------------------------------------------------------------\n"); |
80 | _AD._TotalLines += linenum()-1; // -1 for overshoot in "nextline" routine |
81 | |
82 | // Write out information we have stored |
83 | // // UNIXism == fsync(stderr); |
84 | } |
85 | |
86 | //------------------------------parse------------------------------------------ |
87 | // Each top-level keyword should appear as the first non-whitespace on a line. |
88 | // |
89 | void ADLParser::parse() { |
90 | char *ident; |
91 | |
92 | // Iterate over the lines in the file buffer parsing Level 1 objects |
93 | for( next_line(); _curline != NULL__null; next_line()) { |
94 | _ptr = _curline; // Reset ptr to start of new line |
95 | skipws(); // Skip any leading whitespace |
96 | ident = get_ident(); // Get first token |
97 | if (ident == NULL__null) { // Empty line |
98 | continue; // Get the next line |
99 | } |
100 | if (!strcmp(ident, "instruct")) instr_parse(); |
101 | else if (!strcmp(ident, "operand")) oper_parse(); |
102 | else if (!strcmp(ident, "opclass")) opclass_parse(); |
103 | else if (!strcmp(ident, "ins_attrib")) ins_attr_parse(); |
104 | else if (!strcmp(ident, "op_attrib")) op_attr_parse(); |
105 | else if (!strcmp(ident, "source")) source_parse(); |
106 | else if (!strcmp(ident, "source_hpp")) source_hpp_parse(); |
107 | else if (!strcmp(ident, "register")) reg_parse(); |
108 | else if (!strcmp(ident, "frame")) frame_parse(); |
109 | else if (!strcmp(ident, "encode")) encode_parse(); |
110 | else if (!strcmp(ident, "pipeline")) pipe_parse(); |
111 | else if (!strcmp(ident, "definitions")) definitions_parse(); |
112 | else if (!strcmp(ident, "peephole")) peep_parse(); |
113 | else if (!strcmp(ident, "#line")) preproc_line(); |
114 | else if (!strcmp(ident, "#define")) preproc_define(); |
115 | else if (!strcmp(ident, "#undef")) preproc_undef(); |
116 | else { |
117 | parse_err(SYNERR1, "expected one of - instruct, operand, ins_attrib, op_attrib, source, register, pipeline, encode\n Found %s",ident); |
118 | } |
119 | } |
120 | // Add reg_class spill_regs after parsing. |
121 | RegisterForm *regBlock = _AD.get_registers(); |
122 | if (regBlock == NULL__null) { |
123 | parse_err(SEMERR2, "Did not declare 'register' definitions"); |
124 | } |
125 | regBlock->addSpillRegClass(); |
126 | regBlock->addDynamicRegClass(); |
127 | |
128 | // Done with parsing, check consistency. |
129 | |
130 | if (_preproc_depth != 0) { |
131 | parse_err(SYNERR1, "End of file inside #ifdef"); |
132 | } |
133 | |
134 | // AttributeForms ins_cost and op_cost must be defined for default behaviour |
135 | if (_globalNames[AttributeForm::_ins_cost] == NULL__null) { |
136 | parse_err(SEMERR2, "Did not declare 'ins_cost' attribute"); |
137 | } |
138 | if (_globalNames[AttributeForm::_op_cost] == NULL__null) { |
139 | parse_err(SEMERR2, "Did not declare 'op_cost' attribute"); |
140 | } |
141 | } |
142 | |
143 | // ******************** Private Level 1 Parse Functions ******************** |
144 | //------------------------------instr_parse------------------------------------ |
145 | // Parse the contents of an instruction definition, build the InstructForm to |
146 | // represent that instruction, and add it to the InstructForm list. |
147 | void ADLParser::instr_parse(void) { |
148 | char *ident; |
149 | InstructForm *instr; |
150 | MatchRule *rule; |
151 | int match_rules_cnt = 0; |
152 | |
153 | // First get the name of the instruction |
154 | if( (ident = get_unique_ident(_globalNames,"instruction")) == NULL__null ) |
155 | return; |
156 | instr = new InstructForm(ident); // Create new instruction form |
157 | instr->_linenum = linenum(); |
158 | _globalNames.Insert(ident, instr); // Add name to the name table |
159 | // Debugging Stuff |
160 | if (_AD._adl_debug > 1) |
161 | fprintf(stderrstderr,"Parsing Instruction Form %s\n", ident); |
162 | |
163 | // Then get the operands |
164 | skipws(); |
165 | if (_curchar != '(') { |
166 | parse_err(SYNERR1, "missing '(' in instruct definition\n"); |
167 | } |
168 | // Parse the operand list |
169 | else get_oplist(instr->_parameters, instr->_localNames); |
170 | skipws(); // Skip leading whitespace |
171 | // Check for block delimiter |
172 | if ( (_curchar != '%') |
173 | || ( next_char(), (_curchar != '{')) ) { |
174 | parse_err(SYNERR1, "missing '%%{' in instruction definition\n"); |
175 | return; |
176 | } |
177 | next_char(); // Maintain the invariant |
178 | do { |
179 | ident = get_ident(); // Grab next identifier |
180 | if (ident == NULL__null) { |
181 | parse_err(SYNERR1, "keyword identifier expected at %c\n", _curchar); |
182 | continue; |
183 | } |
184 | if (!strcmp(ident, "predicate")) instr->_predicate = pred_parse(); |
185 | else if (!strcmp(ident, "match")) { |
186 | // Allow one instruction have several match rules. |
187 | rule = instr->_matrule; |
188 | if (rule == NULL__null) { |
189 | // This is first match rule encountered |
190 | rule = match_parse(instr->_localNames); |
191 | if (rule) { |
192 | instr->_matrule = rule; |
193 | // Special case the treatment of Control instructions. |
194 | if( instr->is_ideal_control() ) { |
195 | // Control instructions return a special result, 'Universe' |
196 | rule->_result = "Universe"; |
197 | } |
198 | // Check for commutative operations with tree operands. |
199 | matchrule_clone_and_swap(rule, instr->_ident, match_rules_cnt); |
200 | } |
201 | } else { |
202 | // Find the end of the match rule list |
203 | while (rule->_next != NULL__null) |
204 | rule = rule->_next; |
205 | // Add the new match rule to the list |
206 | rule->_next = match_parse(instr->_localNames); |
207 | if (rule->_next) { |
208 | rule = rule->_next; |
209 | if( instr->is_ideal_control() ) { |
210 | parse_err(SYNERR1, "unique match rule expected for %s\n", rule->_name); |
211 | return; |
212 | } |
213 | assert(match_rules_cnt < 100," too many match rule clones"){ if (!(match_rules_cnt < 100)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 213, " too many match rule clones"); abort(); }}; |
214 | char* buf = (char*) AllocateHeap(strlen(instr->_ident) + 4); |
215 | sprintf(buf, "%s_%d", instr->_ident, match_rules_cnt++); |
216 | rule->_result = buf; |
217 | // Check for commutative operations with tree operands. |
218 | matchrule_clone_and_swap(rule, instr->_ident, match_rules_cnt); |
219 | } |
220 | } |
221 | } |
222 | else if (!strcmp(ident, "encode")) { |
223 | parse_err(SYNERR1, "Instructions specify ins_encode, not encode\n"); |
224 | } |
225 | else if (!strcmp(ident, "ins_encode")) ins_encode_parse(*instr); |
226 | // Parse late expand keyword. |
227 | else if (!strcmp(ident, "postalloc_expand")) postalloc_expand_parse(*instr); |
228 | else if (!strcmp(ident, "opcode")) instr->_opcode = opcode_parse(instr); |
229 | else if (!strcmp(ident, "size")) instr->_size = size_parse(instr); |
230 | else if (!strcmp(ident, "effect")) effect_parse(instr); |
231 | else if (!strcmp(ident, "expand")) instr->_exprule = expand_parse(instr); |
232 | else if (!strcmp(ident, "rewrite")) instr->_rewrule = rewrite_parse(); |
233 | else if (!strcmp(ident, "constraint")) { |
234 | parse_err(SYNERR1, "Instructions do not specify a constraint\n"); |
235 | } |
236 | else if (!strcmp(ident, "construct")) { |
237 | parse_err(SYNERR1, "Instructions do not specify a construct\n"); |
238 | } |
239 | else if (!strcmp(ident, "format")) instr->_format = format_parse(); |
240 | else if (!strcmp(ident, "interface")) { |
241 | parse_err(SYNERR1, "Instructions do not specify an interface\n"); |
242 | } |
243 | else if (!strcmp(ident, "ins_pipe")) ins_pipe_parse(*instr); |
244 | else { // Done with staticly defined parts of instruction definition |
245 | // Check identifier to see if it is the name of an attribute |
246 | const Form *form = _globalNames[ident]; |
247 | AttributeForm *attr = form ? form->is_attribute() : NULL__null; |
248 | if (attr && (attr->_atype == INS_ATTR0)) { |
249 | // Insert the new attribute into the linked list. |
250 | Attribute *temp = attr_parse(ident); |
251 | temp->_next = instr->_attribs; |
252 | instr->_attribs = temp; |
253 | } else { |
254 | parse_err(SYNERR1, "expected one of:\n predicate, match, encode, or the name of" |
255 | " an instruction attribute at %s\n", ident); |
256 | } |
257 | } |
258 | skipws(); |
259 | } while(_curchar != '%'); |
260 | next_char(); |
261 | if (_curchar != '}') { |
262 | parse_err(SYNERR1, "missing '%%}' in instruction definition\n"); |
263 | return; |
264 | } |
265 | // Check for "Set" form of chain rule |
266 | adjust_set_rule(instr); |
267 | if (_AD._pipeline) { |
268 | // No pipe required for late expand. |
269 | if (instr->expands() || instr->postalloc_expands()) { |
270 | if (instr->_ins_pipe) { |
271 | parse_err(WARN0, "ins_pipe and expand rule both specified for instruction \"%s\";" |
272 | " ins_pipe will be unused\n", instr->_ident); |
273 | } |
274 | } else { |
275 | if (!instr->_ins_pipe) { |
276 | parse_err(WARN0, "No ins_pipe specified for instruction \"%s\"\n", instr->_ident); |
277 | } |
278 | } |
279 | } |
280 | // Add instruction to tail of instruction list |
281 | _AD.addForm(instr); |
282 | |
283 | // Create instruction form for each additional match rule |
284 | rule = instr->_matrule; |
285 | if (rule != NULL__null) { |
286 | rule = rule->_next; |
287 | while (rule != NULL__null) { |
288 | ident = (char*)rule->_result; |
289 | InstructForm *clone = new InstructForm(ident, instr, rule); // Create new instruction form |
290 | _globalNames.Insert(ident, clone); // Add name to the name table |
291 | // Debugging Stuff |
292 | if (_AD._adl_debug > 1) |
293 | fprintf(stderrstderr,"Parsing Instruction Form %s\n", ident); |
294 | // Check for "Set" form of chain rule |
295 | adjust_set_rule(clone); |
296 | // Add instruction to tail of instruction list |
297 | _AD.addForm(clone); |
298 | rule = rule->_next; |
299 | clone->_matrule->_next = NULL__null; // One match rule per clone |
300 | } |
301 | } |
302 | } |
303 | |
304 | //------------------------------matchrule_clone_and_swap----------------------- |
305 | // Check for commutative operations with subtree operands, |
306 | // create clones and swap operands. |
307 | void ADLParser::matchrule_clone_and_swap(MatchRule* rule, const char* instr_ident, int& match_rules_cnt) { |
308 | // Check for commutative operations with tree operands. |
309 | int count = 0; |
310 | rule->count_commutative_op(count); |
311 | if (count > 0) { |
312 | // Clone match rule and swap commutative operation's operands. |
313 | rule->matchrule_swap_commutative_op(instr_ident, count, match_rules_cnt); |
314 | } |
315 | } |
316 | |
317 | //------------------------------adjust_set_rule-------------------------------- |
318 | // Check for "Set" form of chain rule |
319 | void ADLParser::adjust_set_rule(InstructForm *instr) { |
320 | if (instr->_matrule == NULL__null || instr->_matrule->_rChild == NULL__null) return; |
321 | const char *rch = instr->_matrule->_rChild->_opType; |
322 | const Form *frm = _globalNames[rch]; |
323 | if( (! strcmp(instr->_matrule->_opType,"Set")) && |
324 | frm && frm->is_operand() && (! frm->ideal_only()) ) { |
325 | // Previous implementation, which missed leaP*, but worked for loadCon* |
326 | unsigned position = 0; |
327 | const char *result = NULL__null; |
328 | const char *name = NULL__null; |
329 | const char *optype = NULL__null; |
330 | MatchNode *right = instr->_matrule->_rChild; |
331 | if (right->base_operand(position, _globalNames, result, name, optype)) { |
332 | position = 1; |
333 | const char *result2 = NULL__null; |
334 | const char *name2 = NULL__null; |
335 | const char *optype2 = NULL__null; |
336 | // Can not have additional base operands in right side of match! |
337 | if ( ! right->base_operand( position, _globalNames, result2, name2, optype2) ) { |
338 | if (instr->_predicate != NULL__null) |
339 | parse_err(SYNERR1, "ADLC does not support instruction chain rules with predicates"); |
340 | // Chain from input _ideal_operand_type_, |
341 | // Needed for shared roots of match-trees |
342 | ChainList *lst = (ChainList *)_AD._chainRules[optype]; |
343 | if (lst == NULL__null) { |
344 | lst = new ChainList(); |
345 | _AD._chainRules.Insert(optype, lst); |
346 | } |
347 | if (!lst->search(instr->_matrule->_lChild->_opType)) { |
348 | const char *cost = instr->cost(); |
349 | if (cost == NULL__null) { |
350 | cost = ((AttributeForm*)_globalNames[AttributeForm::_ins_cost])->_attrdef; |
351 | } |
352 | // The ADLC does not support chaining from the ideal operand type |
353 | // of a predicated user-defined operand |
354 | if( frm->is_operand() == NULL__null || frm->is_operand()->_predicate == NULL__null ) { |
355 | lst->insert(instr->_matrule->_lChild->_opType,cost,instr->_ident); |
356 | } |
357 | } |
358 | // Chain from input _user_defined_operand_type_, |
359 | lst = (ChainList *)_AD._chainRules[result]; |
360 | if (lst == NULL__null) { |
361 | lst = new ChainList(); |
362 | _AD._chainRules.Insert(result, lst); |
363 | } |
364 | if (!lst->search(instr->_matrule->_lChild->_opType)) { |
365 | const char *cost = instr->cost(); |
366 | if (cost == NULL__null) { |
367 | cost = ((AttributeForm*)_globalNames[AttributeForm::_ins_cost])->_attrdef; |
368 | } |
369 | // It is safe to chain from the top-level user-defined operand even |
370 | // if it has a predicate, since the predicate is checked before |
371 | // the user-defined type is available. |
372 | lst->insert(instr->_matrule->_lChild->_opType,cost,instr->_ident); |
373 | } |
374 | } else { |
375 | // May have instruction chain rule if root of right-tree is an ideal |
376 | OperandForm *rightOp = _globalNames[right->_opType]->is_operand(); |
377 | if( rightOp ) { |
378 | const Form *rightRoot = _globalNames[rightOp->_matrule->_opType]; |
379 | if( rightRoot && rightRoot->ideal_only() ) { |
380 | const char *chain_op = NULL__null; |
381 | if( rightRoot->is_instruction() ) |
382 | chain_op = rightOp->_ident; |
383 | if( chain_op ) { |
384 | // Look-up the operation in chain rule table |
385 | ChainList *lst = (ChainList *)_AD._chainRules[chain_op]; |
386 | if (lst == NULL__null) { |
387 | lst = new ChainList(); |
388 | _AD._chainRules.Insert(chain_op, lst); |
389 | } |
390 | // if (!lst->search(instr->_matrule->_lChild->_opType)) { |
391 | const char *cost = instr->cost(); |
392 | if (cost == NULL__null) { |
393 | cost = ((AttributeForm*)_globalNames[AttributeForm::_ins_cost])->_attrdef; |
394 | } |
395 | // This chains from a top-level operand whose predicate, if any, |
396 | // has been checked. |
397 | lst->insert(instr->_matrule->_lChild->_opType,cost,instr->_ident); |
398 | // } |
399 | } |
400 | } |
401 | } |
402 | } // end chain rule from right-tree's ideal root |
403 | } |
404 | } |
405 | } |
406 | |
407 | |
408 | //------------------------------oper_parse------------------------------------- |
409 | void ADLParser::oper_parse(void) { |
410 | char *ident; |
411 | OperandForm *oper; |
412 | AttributeForm *attr; |
413 | MatchRule *rule; |
414 | |
415 | // First get the name of the operand |
416 | skipws(); |
417 | if( (ident = get_unique_ident(_globalNames,"operand")) == NULL__null ) |
418 | return; |
419 | oper = new OperandForm(ident); // Create new operand form |
420 | oper->_linenum = linenum(); |
421 | _globalNames.Insert(ident, oper); // Add name to the name table |
422 | |
423 | // Debugging Stuff |
424 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Parsing Operand Form %s\n", ident); |
425 | |
426 | // Get the component operands |
427 | skipws(); |
428 | if (_curchar != '(') { |
429 | parse_err(SYNERR1, "missing '(' in operand definition\n"); |
430 | return; |
431 | } |
432 | else get_oplist(oper->_parameters, oper->_localNames); // Parse the component operand list |
433 | skipws(); |
434 | // Check for block delimiter |
435 | if ((_curchar != '%') || (*(_ptr+1) != '{')) { // If not open block |
436 | parse_err(SYNERR1, "missing '%%{' in operand definition\n"); |
437 | return; |
438 | } |
439 | next_char(); next_char(); // Skip over "%{" symbol |
440 | do { |
441 | ident = get_ident(); // Grab next identifier |
442 | if (ident == NULL__null) { |
443 | parse_err(SYNERR1, "keyword identifier expected at %c\n", _curchar); |
444 | continue; |
445 | } |
446 | if (!strcmp(ident, "predicate")) oper->_predicate = pred_parse(); |
447 | else if (!strcmp(ident, "match")) { |
448 | // Find the end of the match rule list |
449 | rule = oper->_matrule; |
450 | if (rule) { |
451 | while (rule->_next) rule = rule->_next; |
452 | // Add the new match rule to the list |
453 | rule->_next = match_parse(oper->_localNames); |
454 | if (rule->_next) { |
455 | rule->_next->_result = oper->_ident; |
456 | } |
457 | } |
458 | else { |
459 | // This is first match rule encountered |
460 | oper->_matrule = match_parse(oper->_localNames); |
461 | if (oper->_matrule) { |
462 | oper->_matrule->_result = oper->_ident; |
463 | } |
464 | } |
465 | } |
466 | else if (!strcmp(ident, "encode")) oper->_interface = interface_parse(); |
467 | else if (!strcmp(ident, "ins_encode")) { |
468 | parse_err(SYNERR1, "Operands specify 'encode', not 'ins_encode'\n"); |
469 | } |
470 | else if (!strcmp(ident, "opcode")) { |
471 | parse_err(SYNERR1, "Operands do not specify an opcode\n"); |
472 | } |
473 | else if (!strcmp(ident, "effect")) { |
474 | parse_err(SYNERR1, "Operands do not specify an effect\n"); |
475 | } |
476 | else if (!strcmp(ident, "expand")) { |
477 | parse_err(SYNERR1, "Operands do not specify an expand\n"); |
478 | } |
479 | else if (!strcmp(ident, "rewrite")) { |
480 | parse_err(SYNERR1, "Operands do not specify a rewrite\n"); |
481 | } |
482 | else if (!strcmp(ident, "constraint"))oper->_constraint= constraint_parse(); |
483 | else if (!strcmp(ident, "construct")) oper->_construct = construct_parse(); |
484 | else if (!strcmp(ident, "format")) oper->_format = format_parse(); |
485 | else if (!strcmp(ident, "interface")) oper->_interface = interface_parse(); |
486 | // Check identifier to see if it is the name of an attribute |
487 | else if (((attr = _globalNames[ident]->is_attribute()) != NULL__null) && |
488 | (attr->_atype == OP_ATTR1)) oper->_attribs = attr_parse(ident); |
489 | else { |
490 | parse_err(SYNERR1, "expected one of - constraint, predicate, match, encode, format, construct, or the name of a defined operand attribute at %s\n", ident); |
491 | } |
492 | skipws(); |
493 | } while(_curchar != '%'); |
494 | next_char(); |
495 | if (_curchar != '}') { |
496 | parse_err(SYNERR1, "missing '%%}' in operand definition\n"); |
497 | return; |
498 | } |
499 | // Add operand to tail of operand list |
500 | _AD.addForm(oper); |
501 | } |
502 | |
503 | //------------------------------opclass_parse---------------------------------- |
504 | // Operand Classes are a block with a comma delimited list of operand names |
505 | void ADLParser::opclass_parse(void) { |
506 | char *ident; |
507 | OpClassForm *opc; |
508 | OperandForm *opForm; |
509 | |
510 | // First get the name of the operand class |
511 | skipws(); |
512 | if( (ident = get_unique_ident(_globalNames,"opclass")) == NULL__null ) |
513 | return; |
514 | opc = new OpClassForm(ident); // Create new operand class form |
515 | _globalNames.Insert(ident, opc); // Add name to the name table |
516 | |
517 | // Debugging Stuff |
518 | if (_AD._adl_debug > 1) |
519 | fprintf(stderrstderr,"Parsing Operand Class Form %s\n", ident); |
520 | |
521 | // Get the list of operands |
522 | skipws(); |
523 | if (_curchar != '(') { |
524 | parse_err(SYNERR1, "missing '(' in operand definition\n"); |
525 | return; |
526 | } |
527 | do { |
528 | next_char(); // Skip past open paren or comma |
529 | ident = get_ident(); // Grab next identifier |
530 | if (ident == NULL__null) { |
531 | parse_err(SYNERR1, "keyword identifier expected at %c\n", _curchar); |
532 | continue; |
533 | } |
534 | // Check identifier to see if it is the name of an operand |
535 | const Form *form = _globalNames[ident]; |
536 | opForm = form ? form->is_operand() : NULL__null; |
537 | if ( opForm ) { |
538 | opc->_oplst.addName(ident); // Add operand to opclass list |
539 | opForm->_classes.addName(opc->_ident);// Add opclass to operand list |
540 | } |
541 | else { |
542 | parse_err(SYNERR1, "expected name of a defined operand at %s\n", ident); |
543 | } |
544 | skipws(); // skip trailing whitespace |
545 | } while (_curchar == ','); // Check for the comma |
546 | // Check for closing ')' |
547 | if (_curchar != ')') { |
548 | parse_err(SYNERR1, "missing ')' or ',' in opclass definition\n"); |
549 | return; |
550 | } |
551 | next_char(); // Consume the ')' |
552 | skipws(); |
553 | // Check for closing ';' |
554 | if (_curchar != ';') { |
555 | parse_err(SYNERR1, "missing ';' in opclass definition\n"); |
556 | return; |
557 | } |
558 | next_char(); // Consume the ';' |
559 | // Add operand to tail of operand list |
560 | _AD.addForm(opc); |
561 | } |
562 | |
563 | //------------------------------ins_attr_parse--------------------------------- |
564 | void ADLParser::ins_attr_parse(void) { |
565 | char *ident; |
566 | char *aexpr; |
567 | AttributeForm *attrib; |
568 | |
569 | // get name for the instruction attribute |
570 | skipws(); // Skip leading whitespace |
571 | if( (ident = get_unique_ident(_globalNames,"inst_attrib")) == NULL__null ) |
572 | return; |
573 | // Debugging Stuff |
574 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Parsing Ins_Attribute Form %s\n", ident); |
575 | |
576 | // Get default value of the instruction attribute |
577 | skipws(); // Skip whitespace |
578 | if ((aexpr = get_paren_expr("attribute default expression string")) == NULL__null) { |
579 | parse_err(SYNERR1, "missing '(' in ins_attrib definition\n"); |
580 | return; |
581 | } |
582 | // Debug Stuff |
583 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Attribute Expression: %s\n", aexpr); |
584 | |
585 | // Check for terminator |
586 | if (_curchar != ';') { |
587 | parse_err(SYNERR1, "missing ';' in ins_attrib definition\n"); |
588 | return; |
589 | } |
590 | next_char(); // Advance past the ';' |
591 | |
592 | // Construct the attribute, record global name, and store in ArchDesc |
593 | attrib = new AttributeForm(ident, INS_ATTR0, aexpr); |
594 | _globalNames.Insert(ident, attrib); // Add name to the name table |
595 | _AD.addForm(attrib); |
596 | } |
597 | |
598 | //------------------------------op_attr_parse---------------------------------- |
599 | void ADLParser::op_attr_parse(void) { |
600 | char *ident; |
601 | char *aexpr; |
602 | AttributeForm *attrib; |
603 | |
604 | // get name for the operand attribute |
605 | skipws(); // Skip leading whitespace |
606 | if( (ident = get_unique_ident(_globalNames,"op_attrib")) == NULL__null ) |
607 | return; |
608 | // Debugging Stuff |
609 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Parsing Op_Attribute Form %s\n", ident); |
610 | |
611 | // Get default value of the instruction attribute |
612 | skipws(); // Skip whitespace |
613 | if ((aexpr = get_paren_expr("attribute default expression string")) == NULL__null) { |
614 | parse_err(SYNERR1, "missing '(' in op_attrib definition\n"); |
615 | return; |
616 | } |
617 | // Debug Stuff |
618 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Attribute Expression: %s\n", aexpr); |
619 | |
620 | // Check for terminator |
621 | if (_curchar != ';') { |
622 | parse_err(SYNERR1, "missing ';' in op_attrib definition\n"); |
623 | return; |
624 | } |
625 | next_char(); // Advance past the ';' |
626 | |
627 | // Construct the attribute, record global name, and store in ArchDesc |
628 | attrib = new AttributeForm(ident, OP_ATTR1, aexpr); |
629 | _globalNames.Insert(ident, attrib); |
630 | _AD.addForm(attrib); |
631 | } |
632 | |
633 | //------------------------------definitions_parse----------------------------------- |
634 | void ADLParser::definitions_parse(void) { |
635 | skipws(); // Skip leading whitespace |
636 | if (_curchar == '%' && *(_ptr+1) == '{') { |
637 | next_char(); next_char(); // Skip "%{" |
638 | skipws(); |
639 | while (_curchar != '%' && *(_ptr+1) != '}') { |
640 | // Process each definition until finding closing string "%}" |
641 | char *token = get_ident(); |
642 | if (token == NULL__null) { |
643 | parse_err(SYNERR1, "missing identifier inside definitions block.\n"); |
644 | return; |
645 | } |
646 | if (strcmp(token,"int_def")==0) { int_def_parse(); } |
647 | // if (strcmp(token,"str_def")==0) { str_def_parse(); } |
648 | skipws(); |
649 | } |
650 | } |
651 | else { |
652 | parse_err(SYNERR1, "Missing %%{ ... %%} block after definitions keyword.\n"); |
653 | return; |
654 | } |
655 | } |
656 | |
657 | //------------------------------int_def_parse---------------------------------- |
658 | // Parse Example: |
659 | // int_def MEMORY_REF_COST ( 200, DEFAULT_COST * 2); |
660 | // <keyword> <name> ( <int_value>, <description> ); |
661 | // |
662 | void ADLParser::int_def_parse(void) { |
663 | char *name = NULL__null; // Name of definition |
664 | char *value = NULL__null; // its value, |
665 | int int_value = -1; // positive values only |
666 | char *description = NULL__null; // textual description |
667 | |
668 | // Get definition name |
669 | skipws(); // Skip whitespace |
670 | name = get_ident(); |
671 | if (name == NULL__null) { |
672 | parse_err(SYNERR1, "missing definition name after int_def\n"); |
673 | return; |
674 | } |
675 | |
676 | // Check for value of int_def dname( integer_value [, string_expression ] ) |
677 | skipws(); |
678 | if (_curchar == '(') { |
679 | |
680 | // Parse the integer value. |
681 | next_char(); |
682 | value = get_ident(); |
683 | if (value == NULL__null) { |
684 | parse_err(SYNERR1, "missing value in int_def\n"); |
685 | return; |
686 | } |
687 | if( !is_int_token(value, int_value) ) { |
688 | parse_err(SYNERR1, "value in int_def is not recognized as integer\n"); |
689 | return; |
690 | } |
691 | skipws(); |
692 | |
693 | // Check for description |
694 | if (_curchar == ',') { |
695 | next_char(); // skip ',' |
696 | |
697 | description = get_expr("int_def description", ")"); |
698 | if (description == NULL__null) { |
699 | parse_err(SYNERR1, "invalid or missing description in int_def\n"); |
700 | return; |
701 | } |
702 | trim(description); |
703 | } |
704 | |
705 | if (_curchar != ')') { |
706 | parse_err(SYNERR1, "missing ')' in register definition statement\n"); |
707 | return; |
708 | } |
709 | next_char(); |
710 | } |
711 | |
712 | // Check for closing ';' |
713 | skipws(); |
714 | if (_curchar != ';') { |
715 | parse_err(SYNERR1, "missing ';' after int_def\n"); |
716 | return; |
717 | } |
718 | next_char(); // move past ';' |
719 | |
720 | // Debug Stuff |
721 | if (_AD._adl_debug > 1) { |
722 | fprintf(stderrstderr,"int_def: %s ( %s, %s )\n", name, |
723 | (value), (description ? description : "")); |
724 | } |
725 | |
726 | // Record new definition. |
727 | Expr *expr = new Expr(name, description, int_value, int_value); |
728 | const Expr *old_expr = _AD.globalDefs().define(name, expr); |
729 | if (old_expr != NULL__null) { |
730 | parse_err(SYNERR1, "Duplicate definition\n"); |
731 | return; |
732 | } |
733 | |
734 | return; |
735 | } |
736 | |
737 | |
738 | //------------------------------source_parse----------------------------------- |
739 | void ADLParser::source_parse(void) { |
740 | SourceForm *source; // Encode class for instruction/operand |
741 | char *rule = NULL__null; // String representation of encode rule |
742 | |
743 | skipws(); // Skip leading whitespace |
744 | if ( (rule = find_cpp_block("source block")) == NULL__null ) { |
745 | parse_err(SYNERR1, "incorrect or missing block for 'source'.\n"); |
746 | return; |
747 | } |
748 | // Debug Stuff |
749 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Source Form: %s\n", rule); |
750 | |
751 | source = new SourceForm(rule); // Build new Source object |
752 | _AD.addForm(source); |
753 | // skipws(); |
754 | } |
755 | |
756 | //------------------------------source_hpp_parse------------------------------- |
757 | // Parse a source_hpp %{ ... %} block. |
758 | // The code gets stuck into the ad_<arch>.hpp file. |
759 | // If the source_hpp block appears before the register block in the AD |
760 | // file, it goes up at the very top of the ad_<arch>.hpp file, so that |
761 | // it can be used by register encodings, etc. Otherwise, it goes towards |
762 | // the bottom, where it's useful as a global definition to *.cpp files. |
763 | void ADLParser::source_hpp_parse(void) { |
764 | char *rule = NULL__null; // String representation of encode rule |
765 | |
766 | skipws(); // Skip leading whitespace |
767 | if ( (rule = find_cpp_block("source_hpp block")) == NULL__null ) { |
768 | parse_err(SYNERR1, "incorrect or missing block for 'source_hpp'.\n"); |
769 | return; |
770 | } |
771 | // Debug Stuff |
772 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Header Form: %s\n", rule); |
773 | |
774 | if (_AD.get_registers() == NULL__null) { |
775 | // Very early in the file, before reg_defs, we collect pre-headers. |
776 | PreHeaderForm* pre_header = new PreHeaderForm(rule); |
777 | _AD.addForm(pre_header); |
778 | } else { |
779 | // Normally, we collect header info, placed at the bottom of the hpp file. |
780 | HeaderForm* header = new HeaderForm(rule); |
781 | _AD.addForm(header); |
782 | } |
783 | } |
784 | |
785 | //------------------------------reg_parse-------------------------------------- |
786 | void ADLParser::reg_parse(void) { |
787 | RegisterForm *regBlock = _AD.get_registers(); // Information about registers encoding |
788 | if (regBlock == NULL__null) { |
789 | // Create the RegisterForm for the architecture description. |
790 | regBlock = new RegisterForm(); // Build new Source object |
791 | _AD.addForm(regBlock); |
792 | } |
793 | |
794 | skipws(); // Skip leading whitespace |
795 | if (_curchar == '%' && *(_ptr+1) == '{') { |
796 | next_char(); next_char(); // Skip "%{" |
797 | skipws(); |
798 | while (_curchar != '%' && *(_ptr+1) != '}') { |
799 | char *token = get_ident(); |
800 | if (token == NULL__null) { |
801 | parse_err(SYNERR1, "missing identifier inside register block.\n"); |
802 | return; |
803 | } |
804 | if (strcmp(token,"reg_def")==0) { reg_def_parse(); } |
805 | else if (strcmp(token,"reg_class")==0) { reg_class_parse(); } |
806 | else if (strcmp(token, "reg_class_dynamic") == 0) { reg_class_dynamic_parse(); } |
807 | else if (strcmp(token,"alloc_class")==0) { alloc_class_parse(); } |
808 | else if (strcmp(token,"#define")==0) { preproc_define(); } |
809 | else { parse_err(SYNERR1, "bad token %s inside register block.\n", token); break; } |
810 | skipws(); |
811 | } |
812 | } |
813 | else { |
814 | parse_err(SYNERR1, "Missing %c{ ... %c} block after register keyword.\n",'%','%'); |
815 | return; |
816 | } |
817 | } |
818 | |
819 | //------------------------------encode_parse----------------------------------- |
820 | void ADLParser::encode_parse(void) { |
821 | EncodeForm *encBlock; // Information about instruction/operand encoding |
822 | |
823 | _AD.getForm(&encBlock); |
824 | if ( encBlock == NULL__null) { |
825 | // Create the EncodeForm for the architecture description. |
826 | encBlock = new EncodeForm(); // Build new Source object |
827 | _AD.addForm(encBlock); |
828 | } |
829 | |
830 | skipws(); // Skip leading whitespace |
831 | if (_curchar == '%' && *(_ptr+1) == '{') { |
832 | next_char(); next_char(); // Skip "%{" |
833 | skipws(); |
834 | while (_curchar != '%' && *(_ptr+1) != '}') { |
835 | char *token = get_ident(); |
836 | if (token == NULL__null) { |
837 | parse_err(SYNERR1, "missing identifier inside encoding block.\n"); |
838 | return; |
839 | } |
840 | if (strcmp(token,"enc_class")==0) { enc_class_parse(); } |
841 | skipws(); |
842 | } |
843 | } |
844 | else { |
845 | parse_err(SYNERR1, "Missing %c{ ... %c} block after encode keyword.\n",'%','%'); |
846 | return; |
847 | } |
848 | } |
849 | |
850 | //------------------------------enc_class_parse-------------------------------- |
851 | void ADLParser::enc_class_parse(void) { |
852 | char *ec_name; // Name of encoding class being defined |
853 | |
854 | // Get encoding class name |
855 | skipws(); // Skip whitespace |
856 | ec_name = get_ident(); |
857 | if (ec_name == NULL__null) { |
858 | parse_err(SYNERR1, "missing encoding class name after encode.\n"); |
859 | return; |
860 | } |
861 | |
862 | EncClass *encoding = _AD._encode->add_EncClass(ec_name); |
863 | encoding->_linenum = linenum(); |
864 | |
865 | skipws(); // Skip leading whitespace |
866 | // Check for optional parameter list |
867 | if (_curchar == '(') { |
868 | do { |
869 | char *pType = NULL__null; // parameter type |
870 | char *pName = NULL__null; // parameter name |
871 | |
872 | next_char(); // skip open paren & comma characters |
873 | skipws(); |
874 | if (_curchar == ')') break; |
875 | |
876 | // Get parameter type |
877 | pType = get_ident(); |
878 | if (pType == NULL__null) { |
879 | parse_err(SYNERR1, "parameter type expected at %c\n", _curchar); |
880 | return; |
881 | } |
882 | |
883 | skipws(); |
884 | // Get parameter name |
885 | pName = get_ident(); |
886 | if (pName == NULL__null) { |
887 | parse_err(SYNERR1, "parameter name expected at %c\n", _curchar); |
888 | return; |
889 | } |
890 | |
891 | // Record parameter type and name |
892 | encoding->add_parameter( pType, pName ); |
893 | |
894 | skipws(); |
895 | } while(_curchar == ','); |
896 | |
897 | if (_curchar != ')') parse_err(SYNERR1, "missing ')'\n"); |
898 | else { |
899 | next_char(); // Skip ')' |
900 | } |
901 | } // Done with parameter list |
902 | |
903 | skipws(); |
904 | // Check for block starting delimiters |
905 | if ((_curchar != '%') || (*(_ptr+1) != '{')) { // If not open block |
906 | parse_err(SYNERR1, "missing '%c{' in enc_class definition\n", '%'); |
907 | return; |
908 | } |
909 | next_char(); // Skip '%' |
910 | next_char(); // Skip '{' |
911 | |
912 | enc_class_parse_block(encoding, ec_name); |
913 | } |
914 | |
915 | |
916 | void ADLParser::enc_class_parse_block(EncClass* encoding, char* ec_name) { |
917 | skipws_no_preproc(); // Skip leading whitespace |
918 | // Prepend location descriptor, for debugging; cf. ADLParser::find_cpp_block |
919 | if (_AD._adlocation_debug) { |
920 | encoding->add_code(get_line_string()); |
921 | } |
922 | |
923 | // Collect the parts of the encode description |
924 | // (1) strings that are passed through to output |
925 | // (2) replacement/substitution variable, preceeded by a '$' |
926 | while ( (_curchar != '%') && (*(_ptr+1) != '}') ) { |
927 | |
928 | // (1) |
929 | // Check if there is a string to pass through to output |
930 | char *start = _ptr; // Record start of the next string |
931 | while ((_curchar != '$') && ((_curchar != '%') || (*(_ptr+1) != '}')) ) { |
932 | // If at the start of a comment, skip past it |
933 | if( (_curchar == '/') && ((*(_ptr+1) == '/') || (*(_ptr+1) == '*')) ) { |
934 | skipws_no_preproc(); |
935 | } else { |
936 | // ELSE advance to the next character, or start of the next line |
937 | next_char_or_line(); |
938 | } |
939 | } |
940 | // If a string was found, terminate it and record in EncClass |
941 | if ( start != _ptr ) { |
942 | *_ptr = '\0'; // Terminate the string |
943 | encoding->add_code(start); |
944 | } |
945 | |
946 | // (2) |
947 | // If we are at a replacement variable, |
948 | // copy it and record in EncClass |
949 | if (_curchar == '$') { |
950 | // Found replacement Variable |
951 | char* rep_var = get_rep_var_ident_dup(); |
952 | // Add flag to _strings list indicating we should check _rep_vars |
953 | encoding->add_rep_var(rep_var); |
954 | } |
955 | } // end while part of format description |
956 | next_char(); // Skip '%' |
957 | next_char(); // Skip '}' |
958 | |
959 | skipws(); |
960 | |
961 | if (_AD._adlocation_debug) { |
962 | encoding->add_code(end_line_marker()); |
963 | } |
964 | |
965 | // Debug Stuff |
966 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"EncodingClass Form: %s\n", ec_name); |
967 | } |
968 | |
969 | //------------------------------frame_parse----------------------------------- |
970 | void ADLParser::frame_parse(void) { |
971 | FrameForm *frame; // Information about stack-frame layout |
972 | char *desc = NULL__null; // String representation of frame |
973 | |
974 | skipws(); // Skip leading whitespace |
975 | |
976 | frame = new FrameForm(); // Build new Frame object |
977 | // Check for open block sequence |
978 | skipws(); // Skip leading whitespace |
979 | if (_curchar == '%' && *(_ptr+1) == '{') { |
980 | next_char(); next_char(); // Skip "%{" |
981 | skipws(); |
982 | while (_curchar != '%' && *(_ptr+1) != '}') { |
983 | char *token = get_ident(); |
984 | if (token == NULL__null) { |
985 | parse_err(SYNERR1, "missing identifier inside frame block.\n"); |
986 | return; |
987 | } |
988 | if (strcmp(token,"sync_stack_slots")==0) { |
989 | sync_stack_slots_parse(frame); |
990 | } |
991 | if (strcmp(token,"frame_pointer")==0) { |
992 | frame_pointer_parse(frame, false); |
993 | } |
994 | if (strcmp(token,"interpreter_frame_pointer")==0) { |
995 | interpreter_frame_pointer_parse(frame, false); |
996 | } |
997 | if (strcmp(token,"inline_cache_reg")==0) { |
998 | inline_cache_parse(frame, false); |
999 | } |
1000 | if (strcmp(token,"compiler_method_oop_reg")==0) { |
1001 | parse_err(WARN0, "Using obsolete Token, compiler_method_oop_reg"); |
1002 | skipws(); |
1003 | } |
1004 | if (strcmp(token,"interpreter_method_oop_reg")==0) { |
1005 | parse_err(WARN0, "Using obsolete Token, interpreter_method_oop_reg"); |
1006 | skipws(); |
1007 | } |
1008 | if (strcmp(token,"interpreter_method_reg")==0) { |
1009 | parse_err(WARN0, "Using obsolete Token, interpreter_method_reg"); |
1010 | skipws(); |
1011 | } |
1012 | if (strcmp(token,"cisc_spilling_operand_name")==0) { |
1013 | cisc_spilling_operand_name_parse(frame, false); |
1014 | } |
1015 | if (strcmp(token,"stack_alignment")==0) { |
1016 | stack_alignment_parse(frame); |
1017 | } |
1018 | if (strcmp(token,"return_addr")==0) { |
1019 | return_addr_parse(frame, false); |
1020 | } |
1021 | if (strcmp(token,"in_preserve_stack_slots")==0) { |
1022 | parse_err(WARN0, "Using obsolete token, in_preserve_stack_slots"); |
1023 | skipws(); |
1024 | } |
1025 | if (strcmp(token,"out_preserve_stack_slots")==0) { |
1026 | parse_err(WARN0, "Using obsolete token, out_preserve_stack_slots"); |
1027 | skipws(); |
1028 | } |
1029 | if (strcmp(token,"varargs_C_out_slots_killed")==0) { |
1030 | frame->_varargs_C_out_slots_killed = parse_one_arg("varargs C out slots killed"); |
1031 | } |
1032 | if (strcmp(token,"calling_convention")==0) { |
1033 | parse_err(WARN0, "Using obsolete token, calling_convention"); |
1034 | skipws(); |
1035 | } |
1036 | if (strcmp(token,"return_value")==0) { |
1037 | frame->_return_value = return_value_parse(); |
1038 | } |
1039 | if (strcmp(token,"c_frame_pointer")==0) { |
1040 | frame_pointer_parse(frame, true); |
1041 | } |
1042 | if (strcmp(token,"c_return_addr")==0) { |
1043 | return_addr_parse(frame, true); |
1044 | } |
1045 | if (strcmp(token,"c_calling_convention")==0) { |
1046 | parse_err(WARN0, "Using obsolete token, c_calling_convention"); |
1047 | skipws(); |
1048 | } |
1049 | if (strcmp(token,"c_return_value")==0) { |
1050 | frame->_c_return_value = return_value_parse(); |
1051 | } |
1052 | |
1053 | skipws(); |
1054 | } |
1055 | } |
1056 | else { |
1057 | parse_err(SYNERR1, "Missing %c{ ... %c} block after encode keyword.\n",'%','%'); |
1058 | return; |
1059 | } |
1060 | // All Java versions are required, native versions are optional |
1061 | if(frame->_frame_pointer == NULL__null) { |
1062 | parse_err(SYNERR1, "missing frame pointer definition in frame section.\n"); |
1063 | return; |
1064 | } |
1065 | // !!!!! !!!!! |
1066 | // if(frame->_interpreter_frame_ptr_reg == NULL) { |
1067 | // parse_err(SYNERR, "missing interpreter frame pointer definition in frame section.\n"); |
1068 | // return; |
1069 | // } |
1070 | if(frame->_alignment == NULL__null) { |
1071 | parse_err(SYNERR1, "missing alignment definition in frame section.\n"); |
1072 | return; |
1073 | } |
1074 | if(frame->_return_addr == NULL__null) { |
1075 | parse_err(SYNERR1, "missing return address location in frame section.\n"); |
1076 | return; |
1077 | } |
1078 | if(frame->_varargs_C_out_slots_killed == NULL__null) { |
1079 | parse_err(SYNERR1, "missing varargs C out slots killed definition in frame section.\n"); |
1080 | return; |
1081 | } |
1082 | if(frame->_return_value == NULL__null) { |
1083 | parse_err(SYNERR1, "missing return value definition in frame section.\n"); |
1084 | return; |
1085 | } |
1086 | // Fill natives in identically with the Java versions if not present. |
1087 | if(frame->_c_frame_pointer == NULL__null) { |
1088 | frame->_c_frame_pointer = frame->_frame_pointer; |
1089 | } |
1090 | if(frame->_c_return_addr == NULL__null) { |
1091 | frame->_c_return_addr = frame->_return_addr; |
1092 | frame->_c_return_addr_loc = frame->_return_addr_loc; |
1093 | } |
1094 | if(frame->_c_return_value == NULL__null) { |
1095 | frame->_c_return_value = frame->_return_value; |
1096 | } |
1097 | |
1098 | // Debug Stuff |
1099 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Frame Form: %s\n", desc); |
1100 | |
1101 | // Create the EncodeForm for the architecture description. |
1102 | _AD.addForm(frame); |
1103 | // skipws(); |
1104 | } |
1105 | |
1106 | //------------------------------sync_stack_slots_parse------------------------- |
1107 | void ADLParser::sync_stack_slots_parse(FrameForm *frame) { |
1108 | // Assign value into frame form |
1109 | frame->_sync_stack_slots = parse_one_arg("sync stack slots entry"); |
1110 | } |
1111 | |
1112 | //------------------------------frame_pointer_parse---------------------------- |
1113 | void ADLParser::frame_pointer_parse(FrameForm *frame, bool native) { |
1114 | char *frame_pointer = parse_one_arg("frame pointer entry"); |
1115 | // Assign value into frame form |
1116 | if (native) { frame->_c_frame_pointer = frame_pointer; } |
1117 | else { frame->_frame_pointer = frame_pointer; } |
1118 | } |
1119 | |
1120 | //------------------------------interpreter_frame_pointer_parse---------------------------- |
1121 | void ADLParser::interpreter_frame_pointer_parse(FrameForm *frame, bool native) { |
1122 | frame->_interpreter_frame_pointer_reg = parse_one_arg("interpreter frame pointer entry"); |
1123 | } |
1124 | |
1125 | //------------------------------inline_cache_parse----------------------------- |
1126 | void ADLParser::inline_cache_parse(FrameForm *frame, bool native) { |
1127 | frame->_inline_cache_reg = parse_one_arg("inline cache reg entry"); |
1128 | } |
1129 | |
1130 | //------------------------------cisc_spilling_operand_parse--------------------- |
1131 | void ADLParser::cisc_spilling_operand_name_parse(FrameForm *frame, bool native) { |
1132 | frame->_cisc_spilling_operand_name = parse_one_arg("cisc spilling operand name"); |
1133 | } |
1134 | |
1135 | //------------------------------stack_alignment_parse-------------------------- |
1136 | void ADLParser::stack_alignment_parse(FrameForm *frame) { |
1137 | char *alignment = parse_one_arg("stack alignment entry"); |
1138 | // Assign value into frame |
1139 | frame->_alignment = alignment; |
1140 | } |
1141 | |
1142 | //------------------------------parse_one_arg------------------------------- |
1143 | char *ADLParser::parse_one_arg(const char *description) { |
1144 | char *token = NULL__null; |
1145 | if(_curchar == '(') { |
1146 | next_char(); |
1147 | skipws(); |
1148 | token = get_expr(description, ")"); |
1149 | if (token == NULL__null) { |
1150 | parse_err(SYNERR1, "missing value inside %s.\n", description); |
1151 | return NULL__null; |
1152 | } |
1153 | next_char(); // skip the close paren |
1154 | if(_curchar != ';') { // check for semi-colon |
1155 | parse_err(SYNERR1, "missing %c in.\n", ';', description); |
1156 | return NULL__null; |
1157 | } |
1158 | next_char(); // skip the semi-colon |
1159 | } |
1160 | else { |
1161 | parse_err(SYNERR1, "Missing %c in.\n", '(', description); |
1162 | return NULL__null; |
1163 | } |
1164 | |
1165 | trim(token); |
1166 | return token; |
1167 | } |
1168 | |
1169 | //------------------------------return_addr_parse------------------------------ |
1170 | void ADLParser::return_addr_parse(FrameForm *frame, bool native) { |
1171 | bool in_register = true; |
1172 | if(_curchar == '(') { |
1173 | next_char(); |
1174 | skipws(); |
1175 | char *token = get_ident(); |
1176 | if (token == NULL__null) { |
1177 | parse_err(SYNERR1, "missing value inside return address entry.\n"); |
1178 | return; |
1179 | } |
1180 | // check for valid values for stack/register |
1181 | if (strcmp(token, "REG") == 0) { |
1182 | in_register = true; |
1183 | } |
1184 | else if (strcmp(token, "STACK") == 0) { |
1185 | in_register = false; |
1186 | } |
1187 | else { |
1188 | parse_err(SYNERR1, "invalid value inside return_address entry.\n"); |
1189 | return; |
1190 | } |
1191 | if (native) { frame->_c_return_addr_loc = in_register; } |
1192 | else { frame->_return_addr_loc = in_register; } |
1193 | |
1194 | // Parse expression that specifies register or stack position |
1195 | skipws(); |
1196 | char *token2 = get_expr("return address entry", ")"); |
1197 | if (token2 == NULL__null) { |
1198 | parse_err(SYNERR1, "missing value inside return address entry.\n"); |
1199 | return; |
1200 | } |
1201 | next_char(); // skip the close paren |
1202 | if (native) { frame->_c_return_addr = token2; } |
1203 | else { frame->_return_addr = token2; } |
1204 | |
1205 | if(_curchar != ';') { // check for semi-colon |
1206 | parse_err(SYNERR1, "missing %c in return address entry.\n", ';'); |
1207 | return; |
1208 | } |
1209 | next_char(); // skip the semi-colon |
1210 | } |
1211 | else { |
1212 | parse_err(SYNERR1, "Missing %c in return_address entry.\n", '('); |
1213 | } |
1214 | } |
1215 | |
1216 | //------------------------------return_value_parse----------------------------- |
1217 | char *ADLParser::return_value_parse() { |
1218 | char *desc = NULL__null; // String representation of return_value |
1219 | |
1220 | skipws(); // Skip leading whitespace |
1221 | if ( (desc = find_cpp_block("return value block")) == NULL__null ) { |
1222 | parse_err(SYNERR1, "incorrect or missing block for 'return_value'.\n"); |
1223 | } |
1224 | return desc; |
1225 | } |
1226 | |
1227 | //------------------------------ins_pipe_parse--------------------------------- |
1228 | void ADLParser::ins_pipe_parse(InstructForm &instr) { |
1229 | char * ident; |
1230 | |
1231 | skipws(); |
1232 | if ( _curchar != '(' ) { // Check for delimiter |
1233 | parse_err(SYNERR1, "missing \"(\" in ins_pipe definition\n"); |
1234 | return; |
1235 | } |
1236 | |
1237 | next_char(); |
1238 | ident = get_ident(); // Grab next identifier |
1239 | |
1240 | if (ident == NULL__null) { |
1241 | parse_err(SYNERR1, "keyword identifier expected at %c\n", _curchar); |
1242 | return; |
1243 | } |
1244 | |
1245 | skipws(); |
1246 | if ( _curchar != ')' ) { // Check for delimiter |
1247 | parse_err(SYNERR1, "missing \")\" in ins_pipe definition\n"); |
1248 | return; |
1249 | } |
1250 | |
1251 | next_char(); // skip the close paren |
1252 | if(_curchar != ';') { // check for semi-colon |
1253 | parse_err(SYNERR1, "missing %c in return value entry.\n", ';'); |
1254 | return; |
1255 | } |
1256 | next_char(); // skip the semi-colon |
1257 | |
1258 | // Check ident for validity |
1259 | if (_AD._pipeline && !_AD._pipeline->_classlist.search(ident)) { |
1260 | parse_err(SYNERR1, "\"%s\" is not a valid pipeline class\n", ident); |
1261 | return; |
1262 | } |
1263 | |
1264 | // Add this instruction to the list in the pipeline class |
1265 | _AD._pipeline->_classdict[ident]->is_pipeclass()->_instructs.addName(instr._ident); |
1266 | |
1267 | // Set the name of the pipeline class in the instruction |
1268 | instr._ins_pipe = ident; |
1269 | return; |
1270 | } |
1271 | |
1272 | //------------------------------pipe_parse------------------------------------- |
1273 | void ADLParser::pipe_parse(void) { |
1274 | PipelineForm *pipeline; // Encode class for instruction/operand |
1275 | char * ident; |
1276 | |
1277 | pipeline = new PipelineForm(); // Build new Source object |
1278 | _AD.addForm(pipeline); |
1279 | |
1280 | skipws(); // Skip leading whitespace |
1281 | // Check for block delimiter |
1282 | if ( (_curchar != '%') |
1283 | || ( next_char(), (_curchar != '{')) ) { |
1284 | parse_err(SYNERR1, "missing '%%{' in pipeline definition\n"); |
1285 | return; |
1286 | } |
1287 | next_char(); // Maintain the invariant |
1288 | do { |
1289 | ident = get_ident(); // Grab next identifier |
1290 | if (ident == NULL__null) { |
1291 | parse_err(SYNERR1, "keyword identifier expected at %c\n", _curchar); |
1292 | continue; |
1293 | } |
1294 | if (!strcmp(ident, "resources" )) resource_parse(*pipeline); |
1295 | else if (!strcmp(ident, "pipe_desc" )) pipe_desc_parse(*pipeline); |
1296 | else if (!strcmp(ident, "pipe_class")) pipe_class_parse(*pipeline); |
1297 | else if (!strcmp(ident, "define")) { |
1298 | skipws(); |
1299 | if ( (_curchar != '%') |
1300 | || ( next_char(), (_curchar != '{')) ) { |
1301 | parse_err(SYNERR1, "expected '%%{'\n"); |
1302 | return; |
1303 | } |
1304 | next_char(); skipws(); |
1305 | |
1306 | char *node_class = get_ident(); |
1307 | if (node_class == NULL__null) { |
1308 | parse_err(SYNERR1, "expected identifier, found \"%c\"\n", _curchar); |
1309 | return; |
1310 | } |
1311 | |
1312 | skipws(); |
1313 | if (_curchar != ',' && _curchar != '=') { |
1314 | parse_err(SYNERR1, "expected `=`, found '%c'\n", _curchar); |
1315 | break; |
1316 | } |
1317 | next_char(); skipws(); |
1318 | |
1319 | char *pipe_class = get_ident(); |
1320 | if (pipe_class == NULL__null) { |
1321 | parse_err(SYNERR1, "expected identifier, found \"%c\"\n", _curchar); |
1322 | return; |
1323 | } |
1324 | if (_curchar != ';' ) { |
1325 | parse_err(SYNERR1, "expected `;`, found '%c'\n", _curchar); |
1326 | break; |
1327 | } |
1328 | next_char(); // Skip over semi-colon |
1329 | |
1330 | skipws(); |
1331 | if ( (_curchar != '%') |
1332 | || ( next_char(), (_curchar != '}')) ) { |
1333 | parse_err(SYNERR1, "expected '%%}', found \"%c\"\n", _curchar); |
1334 | } |
1335 | next_char(); |
1336 | |
1337 | // Check ident for validity |
1338 | if (_AD._pipeline && !_AD._pipeline->_classlist.search(pipe_class)) { |
1339 | parse_err(SYNERR1, "\"%s\" is not a valid pipeline class\n", pipe_class); |
1340 | return; |
1341 | } |
1342 | |
1343 | // Add this machine node to the list in the pipeline class |
1344 | _AD._pipeline->_classdict[pipe_class]->is_pipeclass()->_instructs.addName(node_class); |
1345 | |
1346 | MachNodeForm *machnode = new MachNodeForm(node_class); // Create new machnode form |
1347 | machnode->_machnode_pipe = pipe_class; |
1348 | |
1349 | _AD.addForm(machnode); |
1350 | } |
1351 | else if (!strcmp(ident, "attributes")) { |
1352 | bool vsi_seen = false; |
1353 | |
1354 | skipws(); |
1355 | if ( (_curchar != '%') |
1356 | || ( next_char(), (_curchar != '{')) ) { |
1357 | parse_err(SYNERR1, "expected '%%{'\n"); |
1358 | return; |
1359 | } |
1360 | next_char(); skipws(); |
1361 | |
1362 | while (_curchar != '%') { |
1363 | ident = get_ident(); |
1364 | if (ident == NULL__null) |
1365 | break; |
1366 | |
1367 | if (!strcmp(ident, "variable_size_instructions")) { |
1368 | skipws(); |
1369 | if (_curchar == ';') { |
1370 | next_char(); skipws(); |
1371 | } |
1372 | |
1373 | pipeline->_variableSizeInstrs = true; |
1374 | vsi_seen = true; |
1375 | continue; |
1376 | } |
1377 | |
1378 | if (!strcmp(ident, "fixed_size_instructions")) { |
1379 | skipws(); |
1380 | if (_curchar == ';') { |
1381 | next_char(); skipws(); |
1382 | } |
1383 | |
1384 | pipeline->_variableSizeInstrs = false; |
1385 | vsi_seen = true; |
1386 | continue; |
1387 | } |
1388 | |
1389 | if (!strcmp(ident, "branch_has_delay_slot")) { |
1390 | skipws(); |
1391 | if (_curchar == ';') { |
1392 | next_char(); skipws(); |
1393 | } |
1394 | |
1395 | pipeline->_branchHasDelaySlot = true; |
1396 | continue; |
1397 | } |
1398 | |
1399 | if (!strcmp(ident, "max_instructions_per_bundle")) { |
1400 | skipws(); |
1401 | if (_curchar != '=') { |
1402 | parse_err(SYNERR1, "expected `=`\n"); |
1403 | break; |
1404 | } |
1405 | |
1406 | next_char(); skipws(); |
1407 | pipeline->_maxInstrsPerBundle = get_int(); |
1408 | skipws(); |
1409 | |
1410 | if (_curchar == ';') { |
1411 | next_char(); skipws(); |
1412 | } |
1413 | |
1414 | continue; |
1415 | } |
1416 | |
1417 | if (!strcmp(ident, "max_bundles_per_cycle")) { |
1418 | skipws(); |
1419 | if (_curchar != '=') { |
1420 | parse_err(SYNERR1, "expected `=`\n"); |
1421 | break; |
1422 | } |
1423 | |
1424 | next_char(); skipws(); |
1425 | pipeline->_maxBundlesPerCycle = get_int(); |
1426 | skipws(); |
1427 | |
1428 | if (_curchar == ';') { |
1429 | next_char(); skipws(); |
1430 | } |
1431 | |
1432 | continue; |
1433 | } |
1434 | |
1435 | if (!strcmp(ident, "instruction_unit_size")) { |
1436 | skipws(); |
1437 | if (_curchar != '=') { |
1438 | parse_err(SYNERR1, "expected `=`, found '%c'\n", _curchar); |
1439 | break; |
1440 | } |
1441 | |
1442 | next_char(); skipws(); |
1443 | pipeline->_instrUnitSize = get_int(); |
1444 | skipws(); |
1445 | |
1446 | if (_curchar == ';') { |
1447 | next_char(); skipws(); |
1448 | } |
1449 | |
1450 | continue; |
1451 | } |
1452 | |
1453 | if (!strcmp(ident, "bundle_unit_size")) { |
1454 | skipws(); |
1455 | if (_curchar != '=') { |
1456 | parse_err(SYNERR1, "expected `=`, found '%c'\n", _curchar); |
1457 | break; |
1458 | } |
1459 | |
1460 | next_char(); skipws(); |
1461 | pipeline->_bundleUnitSize = get_int(); |
1462 | skipws(); |
1463 | |
1464 | if (_curchar == ';') { |
1465 | next_char(); skipws(); |
1466 | } |
1467 | |
1468 | continue; |
1469 | } |
1470 | |
1471 | if (!strcmp(ident, "instruction_fetch_unit_size")) { |
1472 | skipws(); |
1473 | if (_curchar != '=') { |
1474 | parse_err(SYNERR1, "expected `=`, found '%c'\n", _curchar); |
1475 | break; |
1476 | } |
1477 | |
1478 | next_char(); skipws(); |
1479 | pipeline->_instrFetchUnitSize = get_int(); |
1480 | skipws(); |
1481 | |
1482 | if (_curchar == ';') { |
1483 | next_char(); skipws(); |
1484 | } |
1485 | |
1486 | continue; |
1487 | } |
1488 | |
1489 | if (!strcmp(ident, "instruction_fetch_units")) { |
1490 | skipws(); |
1491 | if (_curchar != '=') { |
1492 | parse_err(SYNERR1, "expected `=`, found '%c'\n", _curchar); |
1493 | break; |
1494 | } |
1495 | |
1496 | next_char(); skipws(); |
1497 | pipeline->_instrFetchUnits = get_int(); |
1498 | skipws(); |
1499 | |
1500 | if (_curchar == ';') { |
1501 | next_char(); skipws(); |
1502 | } |
1503 | |
1504 | continue; |
1505 | } |
1506 | |
1507 | if (!strcmp(ident, "nops")) { |
1508 | skipws(); |
1509 | if (_curchar != '(') { |
1510 | parse_err(SYNERR1, "expected `(`, found '%c'\n", _curchar); |
1511 | break; |
1512 | } |
1513 | |
1514 | next_char(); skipws(); |
1515 | |
1516 | while (_curchar != ')') { |
1517 | ident = get_ident(); |
1518 | if (ident == NULL__null) { |
1519 | parse_err(SYNERR1, "expected identifier for nop instruction, found '%c'\n", _curchar); |
1520 | break; |
1521 | } |
1522 | |
1523 | pipeline->_noplist.addName(ident); |
1524 | pipeline->_nopcnt++; |
1525 | skipws(); |
1526 | |
1527 | if (_curchar == ',') { |
1528 | next_char(); skipws(); |
1529 | } |
1530 | } |
1531 | |
1532 | next_char(); skipws(); |
1533 | |
1534 | if (_curchar == ';') { |
1535 | next_char(); skipws(); |
1536 | } |
1537 | |
1538 | continue; |
1539 | } |
1540 | |
1541 | parse_err(SYNERR1, "unknown specifier \"%s\"\n", ident); |
1542 | } |
1543 | |
1544 | if ( (_curchar != '%') |
1545 | || ( next_char(), (_curchar != '}')) ) { |
1546 | parse_err(SYNERR1, "expected '%%}', found \"%c\"\n", _curchar); |
1547 | } |
1548 | next_char(); skipws(); |
1549 | |
1550 | if (pipeline->_maxInstrsPerBundle == 0) |
1551 | parse_err(SYNERR1, "\"max_instructions_per_bundle\" unspecified\n"); |
1552 | if (pipeline->_instrUnitSize == 0 && pipeline->_bundleUnitSize == 0) |
1553 | parse_err(SYNERR1, "\"instruction_unit_size\" and \"bundle_unit_size\" unspecified\n"); |
1554 | if (pipeline->_instrFetchUnitSize == 0) |
1555 | parse_err(SYNERR1, "\"instruction_fetch_unit_size\" unspecified\n"); |
1556 | if (pipeline->_instrFetchUnits == 0) |
1557 | parse_err(SYNERR1, "\"instruction_fetch_units\" unspecified\n"); |
1558 | if (!vsi_seen) |
1559 | parse_err(SYNERR1, "\"variable_size_instruction\" or \"fixed_size_instruction\" unspecified\n"); |
1560 | } |
1561 | else { // Done with staticly defined parts of instruction definition |
1562 | parse_err(SYNERR1, "expected one of \"resources\", \"pipe_desc\", \"pipe_class\", found \"%s\"\n", ident); |
1563 | return; |
1564 | } |
1565 | skipws(); |
1566 | if (_curchar == ';') |
1567 | skipws(); |
1568 | } while(_curchar != '%'); |
1569 | |
1570 | next_char(); |
1571 | if (_curchar != '}') { |
1572 | parse_err(SYNERR1, "missing \"%%}\" in pipeline definition\n"); |
1573 | return; |
1574 | } |
1575 | |
1576 | next_char(); |
1577 | } |
1578 | |
1579 | //------------------------------resource_parse---------------------------- |
1580 | void ADLParser::resource_parse(PipelineForm &pipeline) { |
1581 | ResourceForm *resource; |
1582 | char * ident; |
1583 | char * expr; |
1584 | unsigned mask; |
1585 | pipeline._rescount = 0; |
1586 | |
1587 | skipws(); // Skip leading whitespace |
1588 | |
1589 | if (_curchar != '(') { |
1590 | parse_err(SYNERR1, "missing \"(\" in resource definition\n"); |
1591 | return; |
1592 | } |
1593 | |
1594 | do { |
1595 | next_char(); // Skip "(" or "," |
1596 | ident = get_ident(); // Grab next identifier |
1597 | |
1598 | if (_AD._adl_debug > 1) { |
1599 | if (ident != NULL__null) { |
1600 | fprintf(stderrstderr, "resource_parse: identifier: %s\n", ident); |
1601 | } |
1602 | } |
1603 | |
1604 | if (ident == NULL__null) { |
1605 | parse_err(SYNERR1, "keyword identifier expected at \"%c\"\n", _curchar); |
1606 | return; |
1607 | } |
1608 | skipws(); |
1609 | |
1610 | if (_curchar != '=') { |
1611 | mask = (1 << pipeline._rescount++); |
1612 | } |
1613 | else { |
1614 | next_char(); skipws(); |
1615 | expr = get_ident(); // Grab next identifier |
1616 | if (expr == NULL__null) { |
1617 | parse_err(SYNERR1, "keyword identifier expected at \"%c\"\n", _curchar); |
1618 | return; |
1619 | } |
1620 | resource = (ResourceForm *) pipeline._resdict[expr]; |
1621 | if (resource == NULL__null) { |
1622 | parse_err(SYNERR1, "resource \"%s\" is not defined\n", expr); |
1623 | return; |
1624 | } |
1625 | mask = resource->mask(); |
1626 | |
1627 | skipws(); |
1628 | while (_curchar == '|') { |
1629 | next_char(); skipws(); |
1630 | |
1631 | expr = get_ident(); // Grab next identifier |
1632 | if (expr == NULL__null) { |
1633 | parse_err(SYNERR1, "keyword identifier expected at \"%c\"\n", _curchar); |
1634 | return; |
1635 | } |
1636 | |
1637 | resource = (ResourceForm *) pipeline._resdict[expr]; // Look up the value |
1638 | if (resource == NULL__null) { |
1639 | parse_err(SYNERR1, "resource \"%s\" is not defined\n", expr); |
1640 | return; |
1641 | } |
1642 | |
1643 | mask |= resource->mask(); |
1644 | skipws(); |
1645 | } |
1646 | } |
1647 | |
1648 | resource = new ResourceForm(mask); |
1649 | |
1650 | pipeline._resdict.Insert(ident, resource); |
1651 | pipeline._reslist.addName(ident); |
1652 | } while (_curchar == ','); |
1653 | |
1654 | if (_curchar != ')') { |
1655 | parse_err(SYNERR1, "\")\" expected at \"%c\"\n", _curchar); |
1656 | return; |
1657 | } |
1658 | |
1659 | next_char(); // Skip ")" |
1660 | if (_curchar == ';') |
1661 | next_char(); // Skip ";" |
1662 | } |
1663 | |
1664 | //------------------------------resource_parse---------------------------- |
1665 | void ADLParser::pipe_desc_parse(PipelineForm &pipeline) { |
1666 | char * ident; |
1667 | |
1668 | skipws(); // Skip leading whitespace |
1669 | |
1670 | if (_curchar != '(') { |
1671 | parse_err(SYNERR1, "missing \"(\" in pipe_desc definition\n"); |
1672 | return; |
1673 | } |
1674 | |
1675 | do { |
1676 | next_char(); // Skip "(" or "," |
1677 | ident = get_ident(); // Grab next identifier |
1678 | if (ident == NULL__null) { |
1679 | parse_err(SYNERR1, "keyword identifier expected at \"%c\"\n", _curchar); |
1680 | return; |
1681 | } |
1682 | |
1683 | // Add the name to the list |
1684 | pipeline._stages.addName(ident); |
1685 | pipeline._stagecnt++; |
1686 | |
1687 | skipws(); |
1688 | } while (_curchar == ','); |
1689 | |
1690 | if (_curchar != ')') { |
1691 | parse_err(SYNERR1, "\")\" expected at \"%c\"\n", _curchar); |
1692 | return; |
1693 | } |
1694 | |
1695 | next_char(); // Skip ")" |
1696 | if (_curchar == ';') |
1697 | next_char(); // Skip ";" |
1698 | } |
1699 | |
1700 | //------------------------------pipe_class_parse-------------------------- |
1701 | void ADLParser::pipe_class_parse(PipelineForm &pipeline) { |
1702 | PipeClassForm *pipe_class; |
1703 | char * ident; |
1704 | char * stage; |
1705 | char * read_or_write; |
1706 | int is_write; |
1707 | int is_read; |
1708 | OperandForm *oper; |
1709 | |
1710 | skipws(); // Skip leading whitespace |
1711 | |
1712 | ident = get_ident(); // Grab next identifier |
1713 | |
1714 | if (ident == NULL__null) { |
1715 | parse_err(SYNERR1, "keyword identifier expected at \"%c\"\n", _curchar); |
1716 | return; |
1717 | } |
1718 | |
1719 | // Create a record for the pipe_class |
1720 | pipe_class = new PipeClassForm(ident, ++pipeline._classcnt); |
1721 | pipeline._classdict.Insert(ident, pipe_class); |
1722 | pipeline._classlist.addName(ident); |
1723 | |
1724 | // Then get the operands |
1725 | skipws(); |
1726 | if (_curchar != '(') { |
1727 | parse_err(SYNERR1, "missing \"(\" in pipe_class definition\n"); |
1728 | } |
1729 | // Parse the operand list |
1730 | else get_oplist(pipe_class->_parameters, pipe_class->_localNames); |
1731 | skipws(); // Skip leading whitespace |
1732 | // Check for block delimiter |
1733 | if ( (_curchar != '%') |
1734 | || ( next_char(), (_curchar != '{')) ) { |
1735 | parse_err(SYNERR1, "missing \"%%{\" in pipe_class definition\n"); |
1736 | return; |
1737 | } |
1738 | next_char(); |
1739 | |
1740 | do { |
1741 | ident = get_ident(); // Grab next identifier |
1742 | if (ident == NULL__null) { |
1743 | parse_err(SYNERR1, "keyword identifier expected at \"%c\"\n", _curchar); |
1744 | continue; |
1745 | } |
1746 | skipws(); |
1747 | |
1748 | if (!strcmp(ident, "fixed_latency")) { |
1749 | skipws(); |
1750 | if (_curchar != '(') { |
1751 | parse_err(SYNERR1, "missing \"(\" in latency definition\n"); |
1752 | return; |
1753 | } |
1754 | next_char(); skipws(); |
1755 | if( !isdigit(_curchar) ) { |
1756 | parse_err(SYNERR1, "number expected for \"%c\" in latency definition\n", _curchar); |
1757 | return; |
1758 | } |
1759 | int fixed_latency = get_int(); |
1760 | skipws(); |
1761 | if (_curchar != ')') { |
1762 | parse_err(SYNERR1, "missing \")\" in latency definition\n"); |
1763 | return; |
1764 | } |
1765 | next_char(); skipws(); |
1766 | if (_curchar != ';') { |
1767 | parse_err(SYNERR1, "missing \";\" in latency definition\n"); |
1768 | return; |
1769 | } |
1770 | |
1771 | pipe_class->setFixedLatency(fixed_latency); |
1772 | next_char(); skipws(); |
1773 | continue; |
1774 | } |
1775 | |
1776 | if (!strcmp(ident, "zero_instructions") || |
1777 | !strcmp(ident, "no_instructions")) { |
1778 | skipws(); |
1779 | if (_curchar != ';') { |
1780 | parse_err(SYNERR1, "missing \";\" in latency definition\n"); |
1781 | return; |
1782 | } |
1783 | |
1784 | pipe_class->setInstructionCount(0); |
1785 | next_char(); skipws(); |
1786 | continue; |
1787 | } |
1788 | |
1789 | if (!strcmp(ident, "one_instruction_with_delay_slot") || |
1790 | !strcmp(ident, "single_instruction_with_delay_slot")) { |
1791 | skipws(); |
1792 | if (_curchar != ';') { |
1793 | parse_err(SYNERR1, "missing \";\" in latency definition\n"); |
1794 | return; |
1795 | } |
1796 | |
1797 | pipe_class->setInstructionCount(1); |
1798 | pipe_class->setBranchDelay(true); |
1799 | next_char(); skipws(); |
1800 | continue; |
1801 | } |
1802 | |
1803 | if (!strcmp(ident, "one_instruction") || |
1804 | !strcmp(ident, "single_instruction")) { |
1805 | skipws(); |
1806 | if (_curchar != ';') { |
1807 | parse_err(SYNERR1, "missing \";\" in latency definition\n"); |
1808 | return; |
1809 | } |
1810 | |
1811 | pipe_class->setInstructionCount(1); |
1812 | next_char(); skipws(); |
1813 | continue; |
1814 | } |
1815 | |
1816 | if (!strcmp(ident, "instructions_in_first_bundle") || |
1817 | !strcmp(ident, "instruction_count")) { |
1818 | skipws(); |
1819 | |
1820 | int number_of_instructions = 1; |
1821 | |
1822 | if (_curchar != '(') { |
1823 | parse_err(SYNERR1, "\"(\" expected at \"%c\"\n", _curchar); |
1824 | continue; |
1825 | } |
1826 | |
1827 | next_char(); skipws(); |
1828 | number_of_instructions = get_int(); |
1829 | |
1830 | skipws(); |
1831 | if (_curchar != ')') { |
1832 | parse_err(SYNERR1, "\")\" expected at \"%c\"\n", _curchar); |
1833 | continue; |
1834 | } |
1835 | |
1836 | next_char(); skipws(); |
1837 | if (_curchar != ';') { |
1838 | parse_err(SYNERR1, "missing \";\" in latency definition\n"); |
1839 | return; |
1840 | } |
1841 | |
1842 | pipe_class->setInstructionCount(number_of_instructions); |
1843 | next_char(); skipws(); |
1844 | continue; |
1845 | } |
1846 | |
1847 | if (!strcmp(ident, "multiple_bundles")) { |
1848 | skipws(); |
1849 | if (_curchar != ';') { |
1850 | parse_err(SYNERR1, "missing \";\" after multiple bundles\n"); |
1851 | return; |
1852 | } |
1853 | |
1854 | pipe_class->setMultipleBundles(true); |
1855 | next_char(); skipws(); |
1856 | continue; |
1857 | } |
1858 | |
1859 | if (!strcmp(ident, "has_delay_slot")) { |
1860 | skipws(); |
1861 | if (_curchar != ';') { |
1862 | parse_err(SYNERR1, "missing \";\" after \"has_delay_slot\"\n"); |
1863 | return; |
1864 | } |
1865 | |
1866 | pipe_class->setBranchDelay(true); |
1867 | next_char(); skipws(); |
1868 | continue; |
1869 | } |
1870 | |
1871 | if (!strcmp(ident, "force_serialization")) { |
1872 | skipws(); |
1873 | if (_curchar != ';') { |
1874 | parse_err(SYNERR1, "missing \";\" after \"force_serialization\"\n"); |
1875 | return; |
1876 | } |
1877 | |
1878 | pipe_class->setForceSerialization(true); |
1879 | next_char(); skipws(); |
1880 | continue; |
1881 | } |
1882 | |
1883 | if (!strcmp(ident, "may_have_no_code")) { |
1884 | skipws(); |
1885 | if (_curchar != ';') { |
1886 | parse_err(SYNERR1, "missing \";\" after \"may_have_no_code\"\n"); |
1887 | return; |
1888 | } |
1889 | |
1890 | pipe_class->setMayHaveNoCode(true); |
1891 | next_char(); skipws(); |
1892 | continue; |
1893 | } |
1894 | |
1895 | const Form *parm = pipe_class->_localNames[ident]; |
1896 | if (parm != NULL__null) { |
1897 | oper = parm->is_operand(); |
1898 | if (oper == NULL__null && !parm->is_opclass()) { |
1899 | parse_err(SYNERR1, "operand name expected at %s\n", ident); |
1900 | continue; |
1901 | } |
1902 | |
1903 | if (_curchar != ':') { |
1904 | parse_err(SYNERR1, "\":\" expected at \"%c\"\n", _curchar); |
1905 | continue; |
1906 | } |
1907 | next_char(); skipws(); |
1908 | stage = get_ident(); |
1909 | if (stage == NULL__null) { |
1910 | parse_err(SYNERR1, "pipeline stage identifier expected at \"%c\"\n", _curchar); |
1911 | continue; |
1912 | } |
1913 | |
1914 | skipws(); |
1915 | if (_curchar != '(') { |
1916 | parse_err(SYNERR1, "\"(\" expected at \"%c\"\n", _curchar); |
1917 | continue; |
1918 | } |
1919 | |
1920 | next_char(); |
1921 | read_or_write = get_ident(); |
1922 | if (read_or_write == NULL__null) { |
1923 | parse_err(SYNERR1, "\"read\" or \"write\" expected at \"%c\"\n", _curchar); |
1924 | continue; |
1925 | } |
1926 | |
1927 | is_read = strcmp(read_or_write, "read") == 0; |
1928 | is_write = strcmp(read_or_write, "write") == 0; |
1929 | if (!is_read && !is_write) { |
1930 | parse_err(SYNERR1, "\"read\" or \"write\" expected at \"%c\"\n", _curchar); |
1931 | continue; |
1932 | } |
1933 | |
1934 | skipws(); |
1935 | if (_curchar != ')') { |
1936 | parse_err(SYNERR1, "\")\" expected at \"%c\"\n", _curchar); |
1937 | continue; |
1938 | } |
1939 | |
1940 | next_char(); skipws(); |
1941 | int more_instrs = 0; |
1942 | if (_curchar == '+') { |
1943 | next_char(); skipws(); |
1944 | if (_curchar < '0' || _curchar > '9') { |
1945 | parse_err(SYNERR1, "<number> expected at \"%c\"\n", _curchar); |
1946 | continue; |
1947 | } |
1948 | while (_curchar >= '0' && _curchar <= '9') { |
1949 | more_instrs *= 10; |
1950 | more_instrs += _curchar - '0'; |
1951 | next_char(); |
1952 | } |
1953 | skipws(); |
1954 | } |
1955 | |
1956 | PipeClassOperandForm *pipe_operand = new PipeClassOperandForm(stage, is_write, more_instrs); |
1957 | pipe_class->_localUsage.Insert(ident, pipe_operand); |
1958 | |
1959 | if (_curchar == '%') |
1960 | continue; |
1961 | |
1962 | if (_curchar != ';') { |
1963 | parse_err(SYNERR1, "\";\" expected at \"%c\"\n", _curchar); |
1964 | continue; |
1965 | } |
1966 | next_char(); skipws(); |
1967 | continue; |
1968 | } |
1969 | |
1970 | // Scan for Resource Specifier |
1971 | const Form *res = pipeline._resdict[ident]; |
1972 | if (res != NULL__null) { |
1973 | int cyclecnt = 1; |
1974 | if (_curchar != ':') { |
1975 | parse_err(SYNERR1, "\":\" expected at \"%c\"\n", _curchar); |
1976 | continue; |
1977 | } |
1978 | next_char(); skipws(); |
1979 | stage = get_ident(); |
1980 | if (stage == NULL__null) { |
1981 | parse_err(SYNERR1, "pipeline stage identifier expected at \"%c\"\n", _curchar); |
1982 | continue; |
1983 | } |
1984 | |
1985 | skipws(); |
1986 | if (_curchar == '(') { |
1987 | next_char(); |
1988 | cyclecnt = get_int(); |
1989 | |
1990 | skipws(); |
1991 | if (_curchar != ')') { |
1992 | parse_err(SYNERR1, "\")\" expected at \"%c\"\n", _curchar); |
1993 | continue; |
1994 | } |
1995 | |
1996 | next_char(); skipws(); |
1997 | } |
1998 | |
1999 | PipeClassResourceForm *resource = new PipeClassResourceForm(ident, stage, cyclecnt); |
2000 | int stagenum = pipeline._stages.index(stage); |
2001 | if (pipeline._maxcycleused < (stagenum+cyclecnt)) |
2002 | pipeline._maxcycleused = (stagenum+cyclecnt); |
2003 | pipe_class->_resUsage.addForm(resource); |
2004 | |
2005 | if (_curchar == '%') |
2006 | continue; |
2007 | |
2008 | if (_curchar != ';') { |
2009 | parse_err(SYNERR1, "\";\" expected at \"%c\"\n", _curchar); |
2010 | continue; |
2011 | } |
2012 | next_char(); skipws(); |
2013 | continue; |
2014 | } |
2015 | |
2016 | parse_err(SYNERR1, "resource expected at \"%s\"\n", ident); |
2017 | return; |
2018 | } while(_curchar != '%'); |
2019 | |
2020 | next_char(); |
2021 | if (_curchar != '}') { |
2022 | parse_err(SYNERR1, "missing \"%%}\" in pipe_class definition\n"); |
2023 | return; |
2024 | } |
2025 | |
2026 | next_char(); |
2027 | } |
2028 | |
2029 | //------------------------------peep_parse------------------------------------- |
2030 | void ADLParser::peep_parse(void) { |
2031 | Peephole *peep; // Pointer to current peephole rule form |
2032 | char *desc = NULL__null; // String representation of rule |
2033 | |
2034 | skipws(); // Skip leading whitespace |
2035 | |
2036 | peep = new Peephole(); // Build new Peephole object |
2037 | // Check for open block sequence |
2038 | skipws(); // Skip leading whitespace |
2039 | if (_curchar == '%' && *(_ptr+1) == '{') { |
2040 | next_char(); next_char(); // Skip "%{" |
2041 | skipws(); |
2042 | while (_curchar != '%' && *(_ptr+1) != '}') { |
2043 | char *token = get_ident(); |
2044 | if (token == NULL__null) { |
2045 | parse_err(SYNERR1, "missing identifier inside peephole rule.\n"); |
2046 | return; |
2047 | } |
2048 | // check for legal subsections of peephole rule |
2049 | if (strcmp(token,"peepmatch")==0) { |
2050 | peep_match_parse(*peep); } |
2051 | else if (strcmp(token,"peepconstraint")==0) { |
2052 | peep_constraint_parse(*peep); } |
2053 | else if (strcmp(token,"peepreplace")==0) { |
2054 | peep_replace_parse(*peep); } |
2055 | else { |
2056 | parse_err(SYNERR1, "expected peepmatch, peepconstraint, or peepreplace for identifier %s.\n", token); |
2057 | } |
2058 | skipws(); |
2059 | } |
2060 | } |
2061 | else { |
2062 | parse_err(SYNERR1, "Missing %%{ ... %%} block after peephole keyword.\n"); |
2063 | return; |
2064 | } |
2065 | next_char(); // Skip past '%' |
2066 | next_char(); // Skip past '}' |
2067 | } |
2068 | |
2069 | // ******************** Private Level 2 Parse Functions ******************** |
2070 | //------------------------------constraint_parse------------------------------ |
2071 | Constraint *ADLParser::constraint_parse(void) { |
2072 | char *func; |
2073 | char *arg; |
2074 | |
2075 | // Check for constraint expression |
2076 | skipws(); |
2077 | if (_curchar != '(') { |
2078 | parse_err(SYNERR1, "missing constraint expression, (...)\n"); |
2079 | return NULL__null; |
2080 | } |
2081 | next_char(); // Skip past '(' |
2082 | |
2083 | // Get constraint function |
2084 | skipws(); |
2085 | func = get_ident(); |
2086 | if (func == NULL__null) { |
2087 | parse_err(SYNERR1, "missing function in constraint expression.\n"); |
2088 | return NULL__null; |
2089 | } |
2090 | if (strcmp(func,"ALLOC_IN_RC")==0 |
2091 | || strcmp(func,"IS_R_CLASS")==0) { |
2092 | // Check for '(' before argument |
2093 | skipws(); |
2094 | if (_curchar != '(') { |
2095 | parse_err(SYNERR1, "missing '(' for constraint function's argument.\n"); |
2096 | return NULL__null; |
2097 | } |
2098 | next_char(); |
2099 | |
2100 | // Get it's argument |
2101 | skipws(); |
2102 | arg = get_ident(); |
2103 | if (arg == NULL__null) { |
2104 | parse_err(SYNERR1, "missing argument for constraint function %s\n",func); |
2105 | return NULL__null; |
2106 | } |
2107 | // Check for ')' after argument |
2108 | skipws(); |
2109 | if (_curchar != ')') { |
2110 | parse_err(SYNERR1, "missing ')' after constraint function argument %s\n",arg); |
2111 | return NULL__null; |
2112 | } |
2113 | next_char(); |
2114 | } else { |
2115 | parse_err(SYNERR1, "Invalid constraint function %s\n",func); |
2116 | return NULL__null; |
2117 | } |
2118 | |
2119 | // Check for closing paren and ';' |
2120 | skipws(); |
2121 | if (_curchar != ')') { |
2122 | parse_err(SYNERR1, "Missing ')' for constraint function %s\n",func); |
2123 | return NULL__null; |
2124 | } |
2125 | next_char(); |
2126 | skipws(); |
2127 | if (_curchar != ';') { |
2128 | parse_err(SYNERR1, "Missing ';' after constraint.\n"); |
2129 | return NULL__null; |
2130 | } |
2131 | next_char(); |
2132 | |
2133 | // Create new "Constraint" |
2134 | Constraint *constraint = new Constraint(func,arg); |
2135 | return constraint; |
2136 | } |
2137 | |
2138 | //------------------------------constr_parse----------------------------------- |
2139 | ConstructRule *ADLParser::construct_parse(void) { |
2140 | return NULL__null; |
2141 | } |
2142 | |
2143 | |
2144 | //------------------------------reg_def_parse---------------------------------- |
2145 | void ADLParser::reg_def_parse(void) { |
2146 | char *rname; // Name of register being defined |
2147 | |
2148 | // Get register name |
2149 | skipws(); // Skip whitespace |
2150 | rname = get_ident(); |
2151 | if (rname == NULL__null) { |
2152 | parse_err(SYNERR1, "missing register name after reg_def\n"); |
2153 | return; |
2154 | } |
2155 | |
2156 | // Check for definition of register calling convention (save on call, ...), |
2157 | // register save type, and register encoding value. |
2158 | skipws(); |
2159 | char *callconv = NULL__null; |
2160 | char *c_conv = NULL__null; |
2161 | char *idealtype = NULL__null; |
2162 | char *encoding = NULL__null; |
2163 | char *concrete = NULL__null; |
2164 | if (_curchar == '(') { |
2165 | next_char(); |
2166 | callconv = get_ident(); |
2167 | // Parse the internal calling convention, must be NS, SOC, SOE, or AS. |
2168 | if (callconv == NULL__null) { |
2169 | parse_err(SYNERR1, "missing register calling convention value\n"); |
2170 | return; |
2171 | } |
2172 | if(strcmp(callconv, "SOC") && strcmp(callconv,"SOE") && |
2173 | strcmp(callconv, "NS") && strcmp(callconv, "AS")) { |
2174 | parse_err(SYNERR1, "invalid value for register calling convention\n"); |
2175 | } |
2176 | skipws(); |
2177 | if (_curchar != ',') { |
2178 | parse_err(SYNERR1, "missing comma in register definition statement\n"); |
2179 | return; |
2180 | } |
2181 | next_char(); |
2182 | |
2183 | // Parse the native calling convention, must be NS, SOC, SOE, AS |
2184 | c_conv = get_ident(); |
2185 | if (c_conv == NULL__null) { |
2186 | parse_err(SYNERR1, "missing register native calling convention value\n"); |
2187 | return; |
2188 | } |
2189 | if(strcmp(c_conv, "SOC") && strcmp(c_conv,"SOE") && |
2190 | strcmp(c_conv, "NS") && strcmp(c_conv, "AS")) { |
2191 | parse_err(SYNERR1, "invalid value for register calling convention\n"); |
2192 | } |
2193 | skipws(); |
2194 | if (_curchar != ',') { |
2195 | parse_err(SYNERR1, "missing comma in register definition statement\n"); |
2196 | return; |
2197 | } |
2198 | next_char(); |
2199 | skipws(); |
2200 | |
2201 | // Parse the ideal save type |
2202 | idealtype = get_ident(); |
2203 | if (idealtype == NULL__null) { |
2204 | parse_err(SYNERR1, "missing register save type value\n"); |
2205 | return; |
2206 | } |
2207 | skipws(); |
2208 | if (_curchar != ',') { |
2209 | parse_err(SYNERR1, "missing comma in register definition statement\n"); |
2210 | return; |
2211 | } |
2212 | next_char(); |
2213 | skipws(); |
2214 | |
2215 | // Parse the encoding value |
2216 | encoding = get_expr("encoding", ","); |
2217 | if (encoding == NULL__null) { |
2218 | parse_err(SYNERR1, "missing register encoding value\n"); |
2219 | return; |
2220 | } |
2221 | trim(encoding); |
2222 | if (_curchar != ',') { |
2223 | parse_err(SYNERR1, "missing comma in register definition statement\n"); |
2224 | return; |
2225 | } |
2226 | next_char(); |
2227 | skipws(); |
2228 | // Parse the concrete name type |
2229 | // concrete = get_ident(); |
2230 | concrete = get_expr("concrete", ")"); |
2231 | if (concrete == NULL__null) { |
2232 | parse_err(SYNERR1, "missing vm register name value\n"); |
2233 | return; |
2234 | } |
2235 | |
2236 | if (_curchar != ')') { |
2237 | parse_err(SYNERR1, "missing ')' in register definition statement\n"); |
2238 | return; |
2239 | } |
2240 | next_char(); |
2241 | } |
2242 | |
2243 | // Check for closing ';' |
2244 | skipws(); |
2245 | if (_curchar != ';') { |
2246 | parse_err(SYNERR1, "missing ';' after reg_def\n"); |
2247 | return; |
2248 | } |
2249 | next_char(); // move past ';' |
2250 | |
2251 | // Debug Stuff |
2252 | if (_AD._adl_debug > 1) { |
2253 | fprintf(stderrstderr,"Register Definition: %s ( %s, %s %s )\n", rname, |
2254 | (callconv ? callconv : ""), (c_conv ? c_conv : ""), concrete); |
2255 | } |
2256 | |
2257 | // Record new register definition. |
2258 | _AD._register->addRegDef(rname, callconv, c_conv, idealtype, encoding, concrete); |
2259 | return; |
2260 | } |
2261 | |
2262 | //------------------------------reg_class_parse-------------------------------- |
2263 | void ADLParser::reg_class_parse(void) { |
2264 | char *cname; // Name of register class being defined |
2265 | |
2266 | // Get register class name |
2267 | skipws(); // Skip leading whitespace |
2268 | cname = get_ident(); |
2269 | if (cname == NULL__null) { |
2270 | parse_err(SYNERR1, "missing register class name after 'reg_class'\n"); |
2271 | return; |
2272 | } |
2273 | // Debug Stuff |
2274 | if (_AD._adl_debug >1) fprintf(stderrstderr,"Register Class: %s\n", cname); |
2275 | |
2276 | skipws(); |
2277 | if (_curchar == '(') { |
2278 | // A register list is defined for the register class. |
2279 | // Collect registers into a generic RegClass register class. |
2280 | RegClass* reg_class = _AD._register->addRegClass<RegClass>(cname); |
2281 | |
2282 | next_char(); // Skip '(' |
2283 | skipws(); |
2284 | while (_curchar != ')') { |
2285 | char *rname = get_ident(); |
2286 | if (rname==NULL__null) { |
2287 | parse_err(SYNERR1, "missing identifier inside reg_class list.\n"); |
2288 | return; |
2289 | } |
2290 | RegDef *regDef = _AD._register->getRegDef(rname); |
2291 | if (!regDef) { |
2292 | parse_err(SEMERR2, "unknown identifier %s inside reg_class list.\n", rname); |
2293 | } else { |
2294 | reg_class->addReg(regDef); // add regDef to regClass |
2295 | } |
2296 | |
2297 | // Check for ',' and position to next token. |
2298 | skipws(); |
2299 | if (_curchar == ',') { |
2300 | next_char(); // Skip trailing ',' |
2301 | skipws(); |
2302 | } |
2303 | } |
2304 | next_char(); // Skip closing ')' |
2305 | } else if (_curchar == '%') { |
2306 | // A code snippet is defined for the register class. |
2307 | // Collect the code snippet into a CodeSnippetRegClass register class. |
2308 | CodeSnippetRegClass* reg_class = _AD._register->addRegClass<CodeSnippetRegClass>(cname); |
2309 | char *code = find_cpp_block("reg class"); |
2310 | if (code == NULL__null) { |
2311 | parse_err(SYNERR1, "missing code declaration for reg class.\n"); |
2312 | return; |
2313 | } |
2314 | reg_class->set_code_snippet(code); |
2315 | return; |
2316 | } |
2317 | |
2318 | // Check for terminating ';' |
2319 | skipws(); |
2320 | if (_curchar != ';') { |
2321 | parse_err(SYNERR1, "missing ';' at end of reg_class definition.\n"); |
2322 | return; |
2323 | } |
2324 | next_char(); // Skip trailing ';' |
2325 | |
2326 | // Check RegClass size, must be <= 32 registers in class. |
2327 | |
2328 | return; |
2329 | } |
2330 | |
2331 | //------------------------------reg_class_dynamic_parse------------------------ |
2332 | void ADLParser::reg_class_dynamic_parse(void) { |
2333 | char *cname; // Name of dynamic register class being defined |
2334 | |
2335 | // Get register class name |
2336 | skipws(); |
2337 | cname = get_ident(); |
2338 | if (cname == NULL__null) { |
2339 | parse_err(SYNERR1, "missing dynamic register class name after 'reg_class_dynamic'\n"); |
2340 | return; |
2341 | } |
2342 | |
2343 | if (_AD._adl_debug > 1) { |
2344 | fprintf(stdoutstdout, "Dynamic Register Class: %s\n", cname); |
2345 | } |
2346 | |
2347 | skipws(); |
2348 | if (_curchar != '(') { |
2349 | parse_err(SYNERR1, "missing '(' at the beginning of reg_class_dynamic definition\n"); |
2350 | return; |
2351 | } |
2352 | next_char(); |
2353 | skipws(); |
2354 | |
2355 | // Collect two register classes and the C++ code representing the condition code used to |
2356 | // select between the two classes into a ConditionalRegClass register class. |
2357 | ConditionalRegClass* reg_class = _AD._register->addRegClass<ConditionalRegClass>(cname); |
2358 | int i; |
2359 | for (i = 0; i < 2; i++) { |
2360 | char* name = get_ident(); |
2361 | if (name == NULL__null) { |
2362 | parse_err(SYNERR1, "missing class identifier inside reg_class_dynamic list.\n"); |
2363 | return; |
2364 | } |
2365 | RegClass* rc = _AD._register->getRegClass(name); |
2366 | if (rc == NULL__null) { |
2367 | parse_err(SEMERR2, "unknown identifier %s inside reg_class_dynamic list.\n", name); |
2368 | } else { |
2369 | reg_class->set_rclass_at_index(i, rc); |
2370 | } |
2371 | |
2372 | skipws(); |
2373 | if (_curchar == ',') { |
2374 | next_char(); |
2375 | skipws(); |
2376 | } else { |
2377 | parse_err(SYNERR1, "missing separator ',' inside reg_class_dynamic list.\n"); |
2378 | } |
2379 | } |
2380 | |
2381 | // Collect the condition code. |
2382 | skipws(); |
2383 | if (_curchar == '%') { |
2384 | char* code = find_cpp_block("reg class dynamic"); |
2385 | if (code == NULL__null) { |
2386 | parse_err(SYNERR1, "missing code declaration for reg_class_dynamic.\n"); |
2387 | return; |
2388 | } |
2389 | reg_class->set_condition_code(code); |
2390 | } else { |
2391 | parse_err(SYNERR1, "missing %% at the beginning of code block in reg_class_dynamic definition\n"); |
2392 | return; |
2393 | } |
2394 | |
2395 | skipws(); |
2396 | if (_curchar != ')') { |
2397 | parse_err(SYNERR1, "missing ')' at the end of reg_class_dynamic definition\n"); |
2398 | return; |
2399 | } |
2400 | next_char(); |
2401 | |
2402 | skipws(); |
2403 | if (_curchar != ';') { |
2404 | parse_err(SYNERR1, "missing ';' at the end of reg_class_dynamic definition.\n"); |
2405 | return; |
2406 | } |
2407 | next_char(); // Skip trailing ';' |
2408 | |
2409 | return; |
2410 | } |
2411 | |
2412 | //------------------------------alloc_class_parse------------------------------ |
2413 | void ADLParser::alloc_class_parse(void) { |
2414 | char *name; // Name of allocation class being defined |
2415 | |
2416 | // Get allocation class name |
2417 | skipws(); // Skip leading whitespace |
2418 | name = get_ident(); |
2419 | if (name == NULL__null) { |
2420 | parse_err(SYNERR1, "missing allocation class name after 'reg_class'\n"); |
2421 | return; |
2422 | } |
2423 | // Debug Stuff |
2424 | if (_AD._adl_debug >1) fprintf(stderrstderr,"Allocation Class: %s\n", name); |
2425 | |
2426 | AllocClass *alloc_class = _AD._register->addAllocClass(name); |
2427 | |
2428 | // Collect registers in class |
2429 | skipws(); |
2430 | if (_curchar == '(') { |
2431 | next_char(); // Skip '(' |
2432 | skipws(); |
2433 | while (_curchar != ')') { |
2434 | char *rname = get_ident(); |
2435 | if (rname==NULL__null) { |
2436 | parse_err(SYNERR1, "missing identifier inside reg_class list.\n"); |
2437 | return; |
2438 | } |
2439 | // Check if name is a RegDef |
2440 | RegDef *regDef = _AD._register->getRegDef(rname); |
2441 | if (regDef) { |
2442 | alloc_class->addReg(regDef); // add regDef to allocClass |
2443 | } else { |
2444 | |
2445 | // name must be a RegDef or a RegClass |
2446 | parse_err(SYNERR1, "name %s should be a previously defined reg_def.\n", rname); |
2447 | return; |
2448 | } |
2449 | |
2450 | // Check for ',' and position to next token. |
2451 | skipws(); |
2452 | if (_curchar == ',') { |
2453 | next_char(); // Skip trailing ',' |
2454 | skipws(); |
2455 | } |
2456 | } |
2457 | next_char(); // Skip closing ')' |
2458 | } |
2459 | |
2460 | // Check for terminating ';' |
2461 | skipws(); |
2462 | if (_curchar != ';') { |
2463 | parse_err(SYNERR1, "missing ';' at end of reg_class definition.\n"); |
2464 | return; |
2465 | } |
2466 | next_char(); // Skip trailing ';' |
2467 | |
2468 | return; |
2469 | } |
2470 | |
2471 | //------------------------------peep_match_child_parse------------------------- |
2472 | InstructForm *ADLParser::peep_match_child_parse(PeepMatch &match, int parent, int &position, int input){ |
2473 | char *token = NULL__null; |
2474 | int lparen = 0; // keep track of parenthesis nesting depth |
2475 | int rparen = 0; // position of instruction at this depth |
2476 | InstructForm *inst_seen = NULL__null; |
2477 | |
2478 | // Walk the match tree, |
2479 | // Record <parent, position, instruction name, input position> |
2480 | while ( lparen >= rparen ) { |
2481 | skipws(); |
2482 | // Left paren signals start of an input, collect with recursive call |
2483 | if (_curchar == '(') { |
2484 | ++lparen; |
2485 | next_char(); |
2486 | ( void ) peep_match_child_parse(match, parent, position, rparen); |
2487 | } |
2488 | // Right paren signals end of an input, may be more |
2489 | else if (_curchar == ')') { |
2490 | ++rparen; |
2491 | if( rparen == lparen ) { // IF rparen matches an lparen I've seen |
2492 | next_char(); // move past ')' |
2493 | } else { // ELSE leave ')' for parent |
2494 | assert( rparen == lparen + 1, "Should only see one extra ')'"){ if (!(rparen == lparen + 1)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 2494, "Should only see one extra ')'"); abort(); }}; |
2495 | // if an instruction was not specified for this paren-pair |
2496 | if( ! inst_seen ) { // record signal entry |
2497 | match.add_instruction( parent, position, NameList::_signal, input ); |
2498 | ++position; |
2499 | } |
2500 | // ++input; // TEMPORARY |
2501 | return inst_seen; |
2502 | } |
2503 | } |
2504 | // if no parens, then check for instruction name |
2505 | // This instruction is the parent of a sub-tree |
2506 | else if ((token = get_ident_dup()) != NULL__null) { |
2507 | const Form *form = _AD._globalNames[token]; |
2508 | if (form) { |
2509 | InstructForm *inst = form->is_instruction(); |
2510 | // Record the first instruction at this level |
2511 | if( inst_seen == NULL__null ) { |
2512 | inst_seen = inst; |
2513 | } |
2514 | if (inst) { |
2515 | match.add_instruction( parent, position, token, input ); |
2516 | parent = position; |
2517 | ++position; |
2518 | } else { |
2519 | parse_err(SYNERR1, "instruction name expected at identifier %s.\n", |
2520 | token); |
2521 | return inst_seen; |
2522 | } |
2523 | } |
2524 | else { |
2525 | parse_err(SYNERR1, "missing identifier in peepmatch rule.\n"); |
2526 | return NULL__null; |
2527 | } |
2528 | } |
2529 | else { |
2530 | parse_err(SYNERR1, "missing identifier in peepmatch rule.\n"); |
2531 | return NULL__null; |
2532 | } |
2533 | |
2534 | } // end while |
2535 | |
2536 | assert( false, "ShouldNotReachHere();"){ if (!(false)) { fprintf(stderr, "assert fails %s %d: %s\n", "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 2536, "ShouldNotReachHere();"); abort(); }}; |
2537 | return NULL__null; |
2538 | } |
2539 | |
2540 | //------------------------------peep_match_parse------------------------------- |
2541 | // Syntax for a peepmatch rule |
2542 | // |
2543 | // peepmatch ( root_instr_name [(instruction subtree)] [,(instruction subtree)]* ); |
2544 | // |
2545 | void ADLParser::peep_match_parse(Peephole &peep) { |
2546 | |
2547 | skipws(); |
2548 | // Check the structure of the rule |
2549 | // Check for open paren |
2550 | if (_curchar != '(') { |
2551 | parse_err(SYNERR1, "missing '(' at start of peepmatch rule.\n"); |
2552 | return; |
2553 | } |
2554 | next_char(); // skip '(' |
2555 | |
2556 | // Construct PeepMatch and parse the peepmatch rule. |
2557 | PeepMatch *match = new PeepMatch(_ptr); |
2558 | int parent = -1; // parent of root |
2559 | int position = 0; // zero-based positions |
2560 | int input = 0; // input position in parent's operands |
2561 | InstructForm *root= peep_match_child_parse( *match, parent, position, input); |
2562 | if( root == NULL__null ) { |
2563 | parse_err(SYNERR1, "missing instruction-name at start of peepmatch.\n"); |
2564 | return; |
2565 | } |
2566 | |
2567 | if( _curchar != ')' ) { |
2568 | parse_err(SYNERR1, "missing ')' at end of peepmatch.\n"); |
2569 | return; |
2570 | } |
2571 | next_char(); // skip ')' |
2572 | |
2573 | // Check for closing semicolon |
2574 | skipws(); |
2575 | if( _curchar != ';' ) { |
2576 | parse_err(SYNERR1, "missing ';' at end of peepmatch.\n"); |
2577 | return; |
2578 | } |
2579 | next_char(); // skip ';' |
2580 | |
2581 | // Store match into peep, and store peep into instruction |
2582 | peep.add_match(match); |
2583 | root->append_peephole(&peep); |
2584 | } |
2585 | |
2586 | //------------------------------peep_constraint_parse-------------------------- |
2587 | // Syntax for a peepconstraint rule |
2588 | // A parenthesized list of relations between operands in peepmatch subtree |
2589 | // |
2590 | // peepconstraint %{ |
2591 | // (instruction_number.operand_name |
2592 | // relational_op |
2593 | // instruction_number.operand_name OR register_name |
2594 | // [, ...] ); |
2595 | // |
2596 | // // instruction numbers are zero-based using topological order in peepmatch |
2597 | // |
2598 | void ADLParser::peep_constraint_parse(Peephole &peep) { |
2599 | |
2600 | skipws(); |
2601 | // Check the structure of the rule |
2602 | // Check for open paren |
2603 | if (_curchar != '(') { |
2604 | parse_err(SYNERR1, "missing '(' at start of peepconstraint rule.\n"); |
2605 | return; |
2606 | } |
2607 | else { |
2608 | next_char(); // Skip '(' |
2609 | } |
2610 | |
2611 | // Check for a constraint |
2612 | skipws(); |
2613 | while( _curchar != ')' ) { |
2614 | // Get information on the left instruction and its operand |
2615 | // left-instructions's number |
2616 | int left_inst = get_int(); |
2617 | // Left-instruction's operand |
2618 | skipws(); |
2619 | if( _curchar != '.' ) { |
2620 | parse_err(SYNERR1, "missing '.' in peepconstraint after instruction number.\n"); |
2621 | return; |
2622 | } |
2623 | next_char(); // Skip '.' |
2624 | char *left_op = get_ident_dup(); |
2625 | |
2626 | skipws(); |
2627 | // Collect relational operator |
2628 | char *relation = get_relation_dup(); |
2629 | |
2630 | skipws(); |
2631 | // Get information on the right instruction and its operand |
2632 | int right_inst; // Right-instructions's number |
2633 | if( isdigit(_curchar) ) { |
2634 | right_inst = get_int(); |
2635 | // Right-instruction's operand |
2636 | skipws(); |
2637 | if( _curchar != '.' ) { |
2638 | parse_err(SYNERR1, "missing '.' in peepconstraint after instruction number.\n"); |
2639 | return; |
2640 | } |
2641 | next_char(); // Skip '.' |
2642 | } else { |
2643 | right_inst = -1; // Flag as being a register constraint |
2644 | } |
2645 | |
2646 | char *right_op = get_ident_dup(); |
2647 | |
2648 | // Construct the next PeepConstraint |
2649 | PeepConstraint *constraint = new PeepConstraint( left_inst, left_op, |
2650 | relation, |
2651 | right_inst, right_op ); |
2652 | // And append it to the list for this peephole rule |
2653 | peep.append_constraint( constraint ); |
2654 | |
2655 | // Check for another constraint, or end of rule |
2656 | skipws(); |
2657 | if( _curchar == ',' ) { |
2658 | next_char(); // Skip ',' |
2659 | skipws(); |
2660 | } |
2661 | else if( _curchar != ')' ) { |
2662 | parse_err(SYNERR1, "expected ',' or ')' after peephole constraint.\n"); |
2663 | return; |
2664 | } |
2665 | } // end while( processing constraints ) |
2666 | next_char(); // Skip ')' |
2667 | |
2668 | // Check for terminating ';' |
2669 | skipws(); |
2670 | if (_curchar != ';') { |
2671 | parse_err(SYNERR1, "missing ';' at end of peepconstraint.\n"); |
2672 | return; |
2673 | } |
2674 | next_char(); // Skip trailing ';' |
2675 | } |
2676 | |
2677 | |
2678 | //------------------------------peep_replace_parse----------------------------- |
2679 | // Syntax for a peepreplace rule |
2680 | // root instruction name followed by a |
2681 | // parenthesized list of whitespace separated instruction.operand specifiers |
2682 | // |
2683 | // peepreplace ( instr_name ( [instruction_number.operand_name]* ) ); |
2684 | // |
2685 | // |
2686 | void ADLParser::peep_replace_parse(Peephole &peep) { |
2687 | int lparen = 0; // keep track of parenthesis nesting depth |
2688 | int rparen = 0; // keep track of parenthesis nesting depth |
2689 | int icount = 0; // count of instructions in rule for naming |
2690 | char *str = NULL__null; |
2691 | char *token = NULL__null; |
2692 | |
2693 | skipws(); |
2694 | // Check for open paren |
2695 | if (_curchar != '(') { |
2696 | parse_err(SYNERR1, "missing '(' at start of peepreplace rule.\n"); |
2697 | return; |
2698 | } |
2699 | else { |
2700 | lparen++; |
2701 | next_char(); |
2702 | } |
2703 | |
2704 | // Check for root instruction |
2705 | char *inst = get_ident_dup(); |
2706 | const Form *form = _AD._globalNames[inst]; |
2707 | if( form == NULL__null || form->is_instruction() == NULL__null ) { |
2708 | parse_err(SYNERR1, "Instruction name expected at start of peepreplace.\n"); |
2709 | return; |
2710 | } |
2711 | |
2712 | // Store string representation of rule into replace |
2713 | PeepReplace *replace = new PeepReplace(str); |
2714 | replace->add_instruction( inst ); |
2715 | |
2716 | skipws(); |
2717 | // Start of root's operand-list |
2718 | if (_curchar != '(') { |
2719 | parse_err(SYNERR1, "missing '(' at peepreplace root's operand-list.\n"); |
2720 | return; |
2721 | } |
2722 | else { |
2723 | lparen++; |
2724 | next_char(); |
2725 | } |
2726 | |
2727 | skipws(); |
2728 | // Get the list of operands |
2729 | while( _curchar != ')' ) { |
2730 | // Get information on an instruction and its operand |
2731 | // instructions's number |
2732 | int inst_num = get_int(); |
2733 | // Left-instruction's operand |
2734 | skipws(); |
2735 | if( _curchar != '.' ) { |
2736 | parse_err(SYNERR1, "missing '.' in peepreplace after instruction number.\n"); |
2737 | return; |
2738 | } |
2739 | next_char(); // Skip '.' |
2740 | char *inst_op = get_ident_dup(); |
2741 | if( inst_op == NULL__null ) { |
2742 | parse_err(SYNERR1, "missing operand identifier in peepreplace.\n"); |
2743 | return; |
2744 | } |
2745 | |
2746 | // Record this operand's position in peepmatch |
2747 | replace->add_operand( inst_num, inst_op ); |
2748 | skipws(); |
2749 | } |
2750 | |
2751 | // Check for the end of operands list |
2752 | skipws(); |
2753 | assert( _curchar == ')', "While loop should have advanced to ')'."){ if (!(_curchar == ')')) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 2753, "While loop should have advanced to ')'."); abort(); } }; |
2754 | next_char(); // Skip ')' |
2755 | |
2756 | skipws(); |
2757 | // Check for end of peepreplace |
2758 | if( _curchar != ')' ) { |
2759 | parse_err(SYNERR1, "missing ')' at end of peepmatch.\n"); |
2760 | parse_err(SYNERR1, "Support one replacement instruction.\n"); |
2761 | return; |
2762 | } |
2763 | next_char(); // Skip ')' |
2764 | |
2765 | // Check for closing semicolon |
2766 | skipws(); |
2767 | if( _curchar != ';' ) { |
2768 | parse_err(SYNERR1, "missing ';' at end of peepreplace.\n"); |
2769 | return; |
2770 | } |
2771 | next_char(); // skip ';' |
2772 | |
2773 | // Store replace into peep |
2774 | peep.add_replace( replace ); |
2775 | } |
2776 | |
2777 | //------------------------------pred_parse------------------------------------- |
2778 | Predicate *ADLParser::pred_parse(void) { |
2779 | Predicate *predicate; // Predicate class for operand |
2780 | char *rule = NULL__null; // String representation of predicate |
2781 | |
2782 | skipws(); // Skip leading whitespace |
2783 | int line = linenum(); |
2784 | if ( (rule = get_paren_expr("pred expression", true)) == NULL__null ) { |
2785 | parse_err(SYNERR1, "incorrect or missing expression for 'predicate'\n"); |
2786 | return NULL__null; |
2787 | } |
2788 | // Debug Stuff |
2789 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Predicate: %s\n", rule); |
2790 | if (_curchar != ';') { |
2791 | parse_err(SYNERR1, "missing ';' in predicate definition\n"); |
2792 | return NULL__null; |
2793 | } |
2794 | next_char(); // Point after the terminator |
2795 | |
2796 | predicate = new Predicate(rule); // Build new predicate object |
2797 | skipws(); |
2798 | return predicate; |
2799 | } |
2800 | |
2801 | |
2802 | //------------------------------ins_encode_parse_block------------------------- |
2803 | // Parse the block form of ins_encode. See ins_encode_parse for more details |
2804 | void ADLParser::ins_encode_parse_block(InstructForm& inst) { |
2805 | // Create a new encoding name based on the name of the instruction |
2806 | // definition, which should be unique. |
2807 | const char* prefix = "__ins_encode_"; |
2808 | char* ec_name = (char*) AllocateHeap(strlen(inst._ident) + strlen(prefix) + 1); |
2809 | sprintf(ec_name, "%s%s", prefix, inst._ident); |
2810 | |
2811 | assert(_AD._encode->encClass(ec_name) == NULL, "shouldn't already exist"){ if (!(_AD._encode->encClass(ec_name) == __null)) { fprintf (stderr, "assert fails %s %d: %s\n", "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 2811, "shouldn't already exist"); abort(); }}; |
2812 | EncClass* encoding = _AD._encode->add_EncClass(ec_name); |
2813 | encoding->_linenum = linenum(); |
2814 | |
2815 | // synthesize the arguments list for the enc_class from the |
2816 | // arguments to the instruct definition. |
2817 | const char* param = NULL__null; |
2818 | inst._parameters.reset(); |
2819 | while ((param = inst._parameters.iter()) != NULL__null) { |
2820 | OpClassForm* opForm = inst._localNames[param]->is_opclass(); |
2821 | assert(opForm != NULL, "sanity"){ if (!(opForm != __null)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 2821, "sanity"); abort(); }}; |
2822 | encoding->add_parameter(opForm->_ident, param); |
2823 | } |
2824 | |
2825 | if (!inst._is_postalloc_expand) { |
2826 | // Define a MacroAssembler instance for use by the encoding. The |
2827 | // name is chosen to match the __ idiom used for assembly in other |
2828 | // parts of hotspot and assumes the existence of the standard |
2829 | // #define __ _masm. |
2830 | encoding->add_code(" C2_MacroAssembler _masm(&cbuf);\n"); |
2831 | } |
2832 | |
2833 | // Parse the following %{ }% block |
2834 | ins_encode_parse_block_impl(inst, encoding, ec_name); |
2835 | |
2836 | // Build an encoding rule which invokes the encoding rule we just |
2837 | // created, passing all arguments that we received. |
2838 | InsEncode* encrule = new InsEncode(); // Encode class for instruction |
2839 | NameAndList* params = encrule->add_encode(ec_name); |
2840 | inst._parameters.reset(); |
2841 | while ((param = inst._parameters.iter()) != NULL__null) { |
2842 | params->add_entry(param); |
2843 | } |
2844 | |
2845 | // Check for duplicate ins_encode sections after parsing the block |
2846 | // so that parsing can continue and find any other errors. |
2847 | if (inst._insencode != NULL__null) { |
2848 | parse_err(SYNERR1, "Multiple ins_encode sections defined\n"); |
2849 | return; |
2850 | } |
2851 | |
2852 | // Set encode class of this instruction. |
2853 | inst._insencode = encrule; |
2854 | } |
2855 | |
2856 | |
2857 | void ADLParser::ins_encode_parse_block_impl(InstructForm& inst, EncClass* encoding, char* ec_name) { |
2858 | skipws_no_preproc(); // Skip leading whitespace |
2859 | // Prepend location descriptor, for debugging; cf. ADLParser::find_cpp_block |
2860 | if (_AD._adlocation_debug) { |
2861 | encoding->add_code(get_line_string()); |
2862 | } |
2863 | |
2864 | // Collect the parts of the encode description |
2865 | // (1) strings that are passed through to output |
2866 | // (2) replacement/substitution variable, preceeded by a '$' |
2867 | while ((_curchar != '%') && (*(_ptr+1) != '}')) { |
2868 | |
2869 | // (1) |
2870 | // Check if there is a string to pass through to output |
2871 | char *start = _ptr; // Record start of the next string |
2872 | while ((_curchar != '$') && ((_curchar != '%') || (*(_ptr+1) != '}')) ) { |
2873 | // If at the start of a comment, skip past it |
2874 | if( (_curchar == '/') && ((*(_ptr+1) == '/') || (*(_ptr+1) == '*')) ) { |
2875 | skipws_no_preproc(); |
2876 | } else { |
2877 | // ELSE advance to the next character, or start of the next line |
2878 | next_char_or_line(); |
2879 | } |
2880 | } |
2881 | // If a string was found, terminate it and record in EncClass |
2882 | if (start != _ptr) { |
2883 | *_ptr = '\0'; // Terminate the string |
2884 | encoding->add_code(start); |
2885 | } |
2886 | |
2887 | // (2) |
2888 | // If we are at a replacement variable, |
2889 | // copy it and record in EncClass |
2890 | if (_curchar == '$') { |
2891 | // Found replacement Variable |
2892 | char* rep_var = get_rep_var_ident_dup(); |
2893 | |
2894 | // Add flag to _strings list indicating we should check _rep_vars |
2895 | encoding->add_rep_var(rep_var); |
2896 | |
2897 | skipws(); |
2898 | |
2899 | // Check if this instruct is a MachConstantNode. |
2900 | if (strcmp(rep_var, "constanttablebase") == 0) { |
2901 | // This instruct is a MachConstantNode. |
2902 | inst.set_needs_constant_base(true); |
2903 | if (strncmp("MachCall", inst.mach_base_class(_globalNames), strlen("MachCall")) != 0 ) { |
2904 | inst.set_is_mach_constant(true); |
2905 | } |
2906 | |
2907 | if (_curchar == '(') { |
2908 | parse_err(SYNERR1, "constanttablebase in instruct %s cannot have an argument " |
2909 | "(only constantaddress and constantoffset)", ec_name); |
2910 | return; |
2911 | } |
2912 | } |
2913 | else if ((strcmp(rep_var, "constantaddress") == 0) || |
2914 | (strcmp(rep_var, "constantoffset") == 0)) { |
2915 | // This instruct is a MachConstantNode. |
2916 | inst.set_is_mach_constant(true); |
2917 | |
2918 | // If the constant keyword has an argument, parse it. |
2919 | if (_curchar == '(') constant_parse(inst); |
2920 | } |
2921 | } |
2922 | } // end while part of format description |
2923 | next_char(); // Skip '%' |
2924 | next_char(); // Skip '}' |
2925 | |
2926 | skipws(); |
2927 | |
2928 | if (_AD._adlocation_debug) { |
2929 | encoding->add_code(end_line_marker()); |
2930 | } |
2931 | |
2932 | // Debug Stuff |
2933 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "EncodingClass Form: %s\n", ec_name); |
2934 | } |
2935 | |
2936 | |
2937 | //------------------------------ins_encode_parse------------------------------- |
2938 | // Encode rules have the form |
2939 | // ins_encode( encode_class_name(parameter_list), ... ); |
2940 | // |
2941 | // The "encode_class_name" must be defined in the encode section |
2942 | // The parameter list contains $names that are locals. |
2943 | // |
2944 | // Alternatively it can be written like this: |
2945 | // |
2946 | // ins_encode %{ |
2947 | // ... // body |
2948 | // %} |
2949 | // |
2950 | // which synthesizes a new encoding class taking the same arguments as |
2951 | // the InstructForm, and automatically prefixes the definition with: |
2952 | // |
2953 | // C2_MacroAssembler masm(&cbuf);\n"); |
2954 | // |
2955 | // making it more compact to take advantage of the C2_MacroAssembler and |
2956 | // placing the assembly closer to it's use by instructions. |
2957 | void ADLParser::ins_encode_parse(InstructForm& inst) { |
2958 | |
2959 | // Parse encode class name |
2960 | skipws(); // Skip whitespace |
2961 | if (_curchar != '(') { |
2962 | // Check for ins_encode %{ form |
2963 | if ((_curchar == '%') && (*(_ptr+1) == '{')) { |
2964 | next_char(); // Skip '%' |
2965 | next_char(); // Skip '{' |
2966 | |
2967 | // Parse the block form of ins_encode |
2968 | ins_encode_parse_block(inst); |
2969 | return; |
2970 | } |
2971 | |
2972 | parse_err(SYNERR1, "missing '%%{' or '(' in ins_encode definition\n"); |
2973 | return; |
2974 | } |
2975 | next_char(); // move past '(' |
2976 | skipws(); |
2977 | |
2978 | InsEncode *encrule = new InsEncode(); // Encode class for instruction |
2979 | encrule->_linenum = linenum(); |
2980 | char *ec_name = NULL__null; // String representation of encode rule |
2981 | // identifier is optional. |
2982 | while (_curchar != ')') { |
2983 | ec_name = get_ident(); |
2984 | if (ec_name == NULL__null) { |
2985 | parse_err(SYNERR1, "Invalid encode class name after 'ins_encode('.\n"); |
2986 | return; |
2987 | } |
2988 | // Check that encoding is defined in the encode section |
2989 | EncClass *encode_class = _AD._encode->encClass(ec_name); |
2990 | if (encode_class == NULL__null) { |
2991 | // Like to defer checking these till later... |
2992 | // parse_err(WARN, "Using an undefined encode class '%s' in 'ins_encode'.\n", ec_name); |
2993 | } |
2994 | |
2995 | // Get list for encode method's parameters |
2996 | NameAndList *params = encrule->add_encode(ec_name); |
2997 | |
2998 | // Parse the parameters to this encode method. |
2999 | skipws(); |
3000 | if ( _curchar == '(' ) { |
3001 | next_char(); // move past '(' for parameters |
3002 | |
3003 | // Parse the encode method's parameters |
3004 | while (_curchar != ')') { |
3005 | char *param = get_ident_or_literal_constant("encoding operand"); |
3006 | if ( param != NULL__null ) { |
3007 | |
3008 | // Check if this instruct is a MachConstantNode. |
3009 | if (strcmp(param, "constanttablebase") == 0) { |
3010 | // This instruct is a MachConstantNode. |
3011 | inst.set_needs_constant_base(true); |
3012 | if (strncmp("MachCall", inst.mach_base_class(_globalNames), strlen("MachCall")) != 0 ) { |
3013 | inst.set_is_mach_constant(true); |
3014 | } |
3015 | |
3016 | if (_curchar == '(') { |
3017 | parse_err(SYNERR1, "constanttablebase in instruct %s cannot have an argument " |
3018 | "(only constantaddress and constantoffset)", ec_name); |
3019 | return; |
3020 | } |
3021 | } else { |
3022 | // Found a parameter: |
3023 | // Check it is a local name, add it to the list, then check for more |
3024 | // New: allow hex constants as parameters to an encode method. |
3025 | // New: allow parenthesized expressions as parameters. |
3026 | // New: allow "primary", "secondary", "tertiary" as parameters. |
3027 | // New: allow user-defined register name as parameter |
3028 | if ( (inst._localNames[param] == NULL__null) && |
3029 | !ADLParser::is_literal_constant(param) && |
3030 | (Opcode::as_opcode_type(param) == Opcode::NOT_AN_OPCODE) && |
3031 | ((_AD._register == NULL__null ) || (_AD._register->getRegDef(param) == NULL__null)) ) { |
3032 | parse_err(SYNERR1, "Using non-locally defined parameter %s for encoding %s.\n", param, ec_name); |
3033 | return; |
3034 | } |
3035 | } |
3036 | params->add_entry(param); |
3037 | |
3038 | skipws(); |
3039 | if (_curchar == ',' ) { |
3040 | // More parameters to come |
3041 | next_char(); // move past ',' between parameters |
3042 | skipws(); // Skip to next parameter |
3043 | } |
3044 | else if (_curchar == ')') { |
3045 | // Done with parameter list |
3046 | } |
3047 | else { |
3048 | // Only ',' or ')' are valid after a parameter name |
3049 | parse_err(SYNERR1, "expected ',' or ')' after parameter %s.\n", |
3050 | ec_name); |
3051 | return; |
3052 | } |
3053 | |
3054 | } else { |
3055 | skipws(); |
3056 | // Did not find a parameter |
3057 | if (_curchar == ',') { |
3058 | parse_err(SYNERR1, "Expected encode parameter before ',' in encoding %s.\n", ec_name); |
3059 | return; |
3060 | } |
3061 | if (_curchar != ')') { |
3062 | parse_err(SYNERR1, "Expected ')' after encode parameters.\n"); |
3063 | return; |
3064 | } |
3065 | } |
3066 | } // WHILE loop collecting parameters |
3067 | next_char(); // move past ')' at end of parameters |
3068 | } // done with parameter list for encoding |
3069 | |
3070 | // Check for ',' or ')' after encoding |
3071 | skipws(); // move to character after parameters |
3072 | if ( _curchar == ',' ) { |
3073 | // Found a ',' |
3074 | next_char(); // move past ',' between encode methods |
3075 | skipws(); |
3076 | } |
3077 | else if ( _curchar != ')' ) { |
3078 | // If not a ',' then only a ')' is allowed |
3079 | parse_err(SYNERR1, "Expected ')' after encoding %s.\n", ec_name); |
3080 | return; |
3081 | } |
3082 | |
3083 | // Check for ',' separating parameters |
3084 | // if ( _curchar != ',' && _curchar != ')' ) { |
3085 | // parse_err(SYNERR, "expected ',' or ')' after encode method inside ins_encode.\n"); |
3086 | // return NULL; |
3087 | // } |
3088 | |
3089 | } // done parsing ins_encode methods and their parameters |
3090 | if (_curchar != ')') { |
3091 | parse_err(SYNERR1, "Missing ')' at end of ins_encode description.\n"); |
3092 | return; |
3093 | } |
3094 | next_char(); // move past ')' |
3095 | skipws(); // Skip leading whitespace |
3096 | |
3097 | if ( _curchar != ';' ) { |
3098 | parse_err(SYNERR1, "Missing ';' at end of ins_encode.\n"); |
3099 | return; |
3100 | } |
3101 | next_char(); // move past ';' |
3102 | skipws(); // be friendly to oper_parse() |
3103 | |
3104 | // Check for duplicate ins_encode sections after parsing the block |
3105 | // so that parsing can continue and find any other errors. |
3106 | if (inst._insencode != NULL__null) { |
3107 | parse_err(SYNERR1, "Multiple ins_encode sections defined\n"); |
3108 | return; |
3109 | } |
3110 | |
3111 | // Debug Stuff |
3112 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Instruction Encode: %s\n", ec_name); |
3113 | |
3114 | // Set encode class of this instruction. |
3115 | inst._insencode = encrule; |
3116 | } |
3117 | |
3118 | //------------------------------postalloc_expand_parse--------------------------- |
3119 | // Encode rules have the form |
3120 | // postalloc_expand( encode_class_name(parameter_list) ); |
3121 | // |
3122 | // The "encode_class_name" must be defined in the encode section. |
3123 | // The parameter list contains $names that are locals. |
3124 | // |
3125 | // This is just a copy of ins_encode_parse without the loop. |
3126 | void ADLParser::postalloc_expand_parse(InstructForm& inst) { |
3127 | inst._is_postalloc_expand = true; |
3128 | |
3129 | // Parse encode class name. |
3130 | skipws(); // Skip whitespace. |
3131 | if (_curchar != '(') { |
3132 | // Check for postalloc_expand %{ form |
3133 | if ((_curchar == '%') && (*(_ptr+1) == '{')) { |
3134 | next_char(); // Skip '%' |
3135 | next_char(); // Skip '{' |
3136 | |
3137 | // Parse the block form of postalloc_expand |
3138 | ins_encode_parse_block(inst); |
3139 | return; |
3140 | } |
3141 | |
3142 | parse_err(SYNERR1, "missing '(' in postalloc_expand definition\n"); |
3143 | return; |
3144 | } |
3145 | next_char(); // Move past '('. |
3146 | skipws(); |
3147 | |
3148 | InsEncode *encrule = new InsEncode(); // Encode class for instruction. |
3149 | encrule->_linenum = linenum(); |
3150 | char *ec_name = NULL__null; // String representation of encode rule. |
3151 | // identifier is optional. |
3152 | if (_curchar != ')') { |
3153 | ec_name = get_ident(); |
3154 | if (ec_name == NULL__null) { |
3155 | parse_err(SYNERR1, "Invalid postalloc_expand class name after 'postalloc_expand('.\n"); |
3156 | return; |
3157 | } |
3158 | // Check that encoding is defined in the encode section. |
3159 | EncClass *encode_class = _AD._encode->encClass(ec_name); |
Value stored to 'encode_class' during its initialization is never read | |
3160 | |
3161 | // Get list for encode method's parameters |
3162 | NameAndList *params = encrule->add_encode(ec_name); |
3163 | |
3164 | // Parse the parameters to this encode method. |
3165 | skipws(); |
3166 | if (_curchar == '(') { |
3167 | next_char(); // Move past '(' for parameters. |
3168 | |
3169 | // Parse the encode method's parameters. |
3170 | while (_curchar != ')') { |
3171 | char *param = get_ident_or_literal_constant("encoding operand"); |
3172 | if (param != NULL__null) { |
3173 | // Found a parameter: |
3174 | |
3175 | // First check for constant table support. |
3176 | |
3177 | // Check if this instruct is a MachConstantNode. |
3178 | if (strcmp(param, "constanttablebase") == 0) { |
3179 | // This instruct is a MachConstantNode. |
3180 | inst.set_needs_constant_base(true); |
3181 | if (strncmp("MachCall", inst.mach_base_class(_globalNames), strlen("MachCall")) != 0 ) { |
3182 | inst.set_is_mach_constant(true); |
3183 | } |
3184 | |
3185 | if (_curchar == '(') { |
3186 | parse_err(SYNERR1, "constanttablebase in instruct %s cannot have an argument " |
3187 | "(only constantaddress and constantoffset)", ec_name); |
3188 | return; |
3189 | } |
3190 | } |
3191 | else if ((strcmp(param, "constantaddress") == 0) || |
3192 | (strcmp(param, "constantoffset") == 0)) { |
3193 | // This instruct is a MachConstantNode. |
3194 | inst.set_is_mach_constant(true); |
3195 | |
3196 | // If the constant keyword has an argument, parse it. |
3197 | if (_curchar == '(') constant_parse(inst); |
3198 | } |
3199 | |
3200 | // Else check it is a local name, add it to the list, then check for more. |
3201 | // New: allow hex constants as parameters to an encode method. |
3202 | // New: allow parenthesized expressions as parameters. |
3203 | // New: allow "primary", "secondary", "tertiary" as parameters. |
3204 | // New: allow user-defined register name as parameter. |
3205 | else if ((inst._localNames[param] == NULL__null) && |
3206 | !ADLParser::is_literal_constant(param) && |
3207 | (Opcode::as_opcode_type(param) == Opcode::NOT_AN_OPCODE) && |
3208 | ((_AD._register == NULL__null) || (_AD._register->getRegDef(param) == NULL__null))) { |
3209 | parse_err(SYNERR1, "Using non-locally defined parameter %s for encoding %s.\n", param, ec_name); |
3210 | return; |
3211 | } |
3212 | params->add_entry(param); |
3213 | |
3214 | skipws(); |
3215 | if (_curchar == ',') { |
3216 | // More parameters to come. |
3217 | next_char(); // Move past ',' between parameters. |
3218 | skipws(); // Skip to next parameter. |
3219 | } else if (_curchar == ')') { |
3220 | // Done with parameter list |
3221 | } else { |
3222 | // Only ',' or ')' are valid after a parameter name. |
3223 | parse_err(SYNERR1, "expected ',' or ')' after parameter %s.\n", ec_name); |
3224 | return; |
3225 | } |
3226 | |
3227 | } else { |
3228 | skipws(); |
3229 | // Did not find a parameter. |
3230 | if (_curchar == ',') { |
3231 | parse_err(SYNERR1, "Expected encode parameter before ',' in postalloc_expand %s.\n", ec_name); |
3232 | return; |
3233 | } |
3234 | if (_curchar != ')') { |
3235 | parse_err(SYNERR1, "Expected ')' after postalloc_expand parameters.\n"); |
3236 | return; |
3237 | } |
3238 | } |
3239 | } // WHILE loop collecting parameters. |
3240 | next_char(); // Move past ')' at end of parameters. |
3241 | } // Done with parameter list for encoding. |
3242 | |
3243 | // Check for ',' or ')' after encoding. |
3244 | skipws(); // Move to character after parameters. |
3245 | if (_curchar != ')') { |
3246 | // Only a ')' is allowed. |
3247 | parse_err(SYNERR1, "Expected ')' after postalloc_expand %s.\n", ec_name); |
3248 | return; |
3249 | } |
3250 | } // Done parsing postalloc_expand method and their parameters. |
3251 | if (_curchar != ')') { |
3252 | parse_err(SYNERR1, "Missing ')' at end of postalloc_expand description.\n"); |
3253 | return; |
3254 | } |
3255 | next_char(); // Move past ')'. |
3256 | skipws(); // Skip leading whitespace. |
3257 | |
3258 | if (_curchar != ';') { |
3259 | parse_err(SYNERR1, "Missing ';' at end of postalloc_expand.\n"); |
3260 | return; |
3261 | } |
3262 | next_char(); // Move past ';'. |
3263 | skipws(); // Be friendly to oper_parse(). |
3264 | |
3265 | // Debug Stuff. |
3266 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "Instruction postalloc_expand: %s\n", ec_name); |
3267 | |
3268 | // Set encode class of this instruction. |
3269 | inst._insencode = encrule; |
3270 | } |
3271 | |
3272 | |
3273 | //------------------------------constant_parse--------------------------------- |
3274 | // Parse a constant expression. |
3275 | void ADLParser::constant_parse(InstructForm& inst) { |
3276 | // Create a new encoding name based on the name of the instruction |
3277 | // definition, which should be unique. |
3278 | const char* prefix = "__constant_"; |
3279 | char* ec_name = (char*) AllocateHeap(strlen(inst._ident) + strlen(prefix) + 1); |
3280 | sprintf(ec_name, "%s%s", prefix, inst._ident); |
3281 | |
3282 | assert(_AD._encode->encClass(ec_name) == NULL, "shouldn't already exist"){ if (!(_AD._encode->encClass(ec_name) == __null)) { fprintf (stderr, "assert fails %s %d: %s\n", "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 3282, "shouldn't already exist"); abort(); }}; |
3283 | EncClass* encoding = _AD._encode->add_EncClass(ec_name); |
3284 | encoding->_linenum = linenum(); |
3285 | |
3286 | // synthesize the arguments list for the enc_class from the |
3287 | // arguments to the instruct definition. |
3288 | const char* param = NULL__null; |
3289 | inst._parameters.reset(); |
3290 | while ((param = inst._parameters.iter()) != NULL__null) { |
3291 | OpClassForm* opForm = inst._localNames[param]->is_opclass(); |
3292 | assert(opForm != NULL, "sanity"){ if (!(opForm != __null)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 3292, "sanity"); abort(); }}; |
3293 | encoding->add_parameter(opForm->_ident, param); |
3294 | } |
3295 | |
3296 | // Parse the following ( ) expression. |
3297 | constant_parse_expression(encoding, ec_name); |
3298 | |
3299 | // Build an encoding rule which invokes the encoding rule we just |
3300 | // created, passing all arguments that we received. |
3301 | InsEncode* encrule = new InsEncode(); // Encode class for instruction |
3302 | NameAndList* params = encrule->add_encode(ec_name); |
3303 | inst._parameters.reset(); |
3304 | while ((param = inst._parameters.iter()) != NULL__null) { |
3305 | params->add_entry(param); |
3306 | } |
3307 | |
3308 | // Set encode class of this instruction. |
3309 | inst._constant = encrule; |
3310 | } |
3311 | |
3312 | |
3313 | //------------------------------constant_parse_expression---------------------- |
3314 | void ADLParser::constant_parse_expression(EncClass* encoding, char* ec_name) { |
3315 | skipws(); |
3316 | |
3317 | // Prepend location descriptor, for debugging; cf. ADLParser::find_cpp_block |
3318 | if (_AD._adlocation_debug) { |
3319 | encoding->add_code(get_line_string()); |
3320 | } |
3321 | |
3322 | // Start code line. |
3323 | encoding->add_code(" _constant = C->output()->constant_table().add"); |
3324 | |
3325 | // Parse everything in ( ) expression. |
3326 | encoding->add_code("(this, "); |
3327 | next_char(); // Skip '(' |
3328 | int parens_depth = 1; |
3329 | |
3330 | // Collect the parts of the constant expression. |
3331 | // (1) strings that are passed through to output |
3332 | // (2) replacement/substitution variable, preceeded by a '$' |
3333 | while (parens_depth > 0) { |
3334 | if (_curchar == '(') { |
3335 | parens_depth++; |
3336 | encoding->add_code("("); |
3337 | next_char(); |
3338 | } |
3339 | else if (_curchar == ')') { |
3340 | parens_depth--; |
3341 | if (parens_depth > 0) |
3342 | encoding->add_code(")"); |
3343 | next_char(); |
3344 | } |
3345 | else { |
3346 | // (1) |
3347 | // Check if there is a string to pass through to output |
3348 | char *start = _ptr; // Record start of the next string |
3349 | while ((_curchar != '$') && (_curchar != '(') && (_curchar != ')')) { |
3350 | next_char(); |
3351 | } |
3352 | // If a string was found, terminate it and record in EncClass |
3353 | if (start != _ptr) { |
3354 | *_ptr = '\0'; // Terminate the string |
3355 | encoding->add_code(start); |
3356 | } |
3357 | |
3358 | // (2) |
3359 | // If we are at a replacement variable, copy it and record in EncClass. |
3360 | if (_curchar == '$') { |
3361 | // Found replacement Variable |
3362 | char* rep_var = get_rep_var_ident_dup(); |
3363 | encoding->add_rep_var(rep_var); |
3364 | } |
3365 | } |
3366 | } |
3367 | |
3368 | // Finish code line. |
3369 | encoding->add_code(");"); |
3370 | |
3371 | if (_AD._adlocation_debug) { |
3372 | encoding->add_code(end_line_marker()); |
3373 | } |
3374 | |
3375 | // Debug Stuff |
3376 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "EncodingClass Form: %s\n", ec_name); |
3377 | } |
3378 | |
3379 | |
3380 | //------------------------------size_parse----------------------------------- |
3381 | // Parse a 'size(<expr>)' attribute which specifies the size of the |
3382 | // emitted instructions in bytes. <expr> can be a C++ expression, |
3383 | // e.g. a constant. |
3384 | char* ADLParser::size_parse(InstructForm *instr) { |
3385 | char* sizeOfInstr = NULL__null; |
3386 | |
3387 | // Get value of the instruction's size |
3388 | skipws(); |
3389 | |
3390 | // Parse size |
3391 | sizeOfInstr = get_paren_expr("size expression"); |
3392 | if (sizeOfInstr == NULL__null) { |
3393 | parse_err(SYNERR1, "size of opcode expected at %c\n", _curchar); |
3394 | return NULL__null; |
3395 | } |
3396 | |
3397 | skipws(); |
3398 | |
3399 | // Check for terminator |
3400 | if (_curchar != ';') { |
3401 | parse_err(SYNERR1, "missing ';' in ins_attrib definition\n"); |
3402 | return NULL__null; |
3403 | } |
3404 | next_char(); // Advance past the ';' |
3405 | skipws(); // necessary for instr_parse() |
3406 | |
3407 | // Debug Stuff |
3408 | if (_AD._adl_debug > 1) { |
3409 | if (sizeOfInstr != NULL__null) { |
3410 | fprintf(stderrstderr,"size of opcode: %s\n", sizeOfInstr); |
3411 | } |
3412 | } |
3413 | |
3414 | return sizeOfInstr; |
3415 | } |
3416 | |
3417 | |
3418 | //------------------------------opcode_parse----------------------------------- |
3419 | Opcode * ADLParser::opcode_parse(InstructForm *instr) { |
3420 | char *primary = NULL__null; |
3421 | char *secondary = NULL__null; |
3422 | char *tertiary = NULL__null; |
3423 | |
3424 | char *val = NULL__null; |
3425 | Opcode *opcode = NULL__null; |
3426 | |
3427 | // Get value of the instruction's opcode |
3428 | skipws(); |
3429 | if (_curchar != '(') { // Check for parenthesized operand list |
3430 | parse_err(SYNERR1, "missing '(' in expand instruction declaration\n"); |
3431 | return NULL__null; |
3432 | } |
3433 | next_char(); // skip open paren |
3434 | skipws(); |
3435 | if (_curchar != ')') { |
3436 | // Parse primary, secondary, and tertiary opcodes, if provided. |
3437 | if ( (primary = get_ident_or_literal_constant("primary opcode")) == NULL__null ) { |
3438 | parse_err(SYNERR1, "primary hex opcode expected at %c\n", _curchar); |
3439 | return NULL__null; |
3440 | } |
3441 | skipws(); |
3442 | if (_curchar == ',') { |
3443 | next_char(); |
3444 | skipws(); |
3445 | // Parse secondary opcode |
3446 | if ( (secondary = get_ident_or_literal_constant("secondary opcode")) == NULL__null ) { |
3447 | parse_err(SYNERR1, "secondary hex opcode expected at %c\n", _curchar); |
3448 | return NULL__null; |
3449 | } |
3450 | skipws(); |
3451 | if (_curchar == ',') { |
3452 | next_char(); |
3453 | skipws(); |
3454 | // Parse tertiary opcode |
3455 | if ( (tertiary = get_ident_or_literal_constant("tertiary opcode")) == NULL__null ) { |
3456 | parse_err(SYNERR1,"tertiary hex opcode expected at %c\n", _curchar); |
3457 | return NULL__null; |
3458 | } |
3459 | skipws(); |
3460 | } |
3461 | } |
3462 | skipws(); |
3463 | if (_curchar != ')') { |
3464 | parse_err(SYNERR1, "Missing ')' in opcode description\n"); |
3465 | return NULL__null; |
3466 | } |
3467 | } |
3468 | next_char(); // Skip ')' |
3469 | skipws(); |
3470 | // Check for terminator |
3471 | if (_curchar != ';') { |
3472 | parse_err(SYNERR1, "missing ';' in ins_attrib definition\n"); |
3473 | return NULL__null; |
3474 | } |
3475 | next_char(); // Advance past the ';' |
3476 | skipws(); // necessary for instr_parse() |
3477 | |
3478 | // Debug Stuff |
3479 | if (_AD._adl_debug > 1) { |
3480 | if (primary != NULL__null) fprintf(stderrstderr,"primary opcode: %s\n", primary); |
3481 | if (secondary != NULL__null) fprintf(stderrstderr,"secondary opcode: %s\n", secondary); |
3482 | if (tertiary != NULL__null) fprintf(stderrstderr,"tertiary opcode: %s\n", tertiary); |
3483 | } |
3484 | |
3485 | // Generate new object and return |
3486 | opcode = new Opcode(primary, secondary, tertiary); |
3487 | return opcode; |
3488 | } |
3489 | |
3490 | |
3491 | //------------------------------interface_parse-------------------------------- |
3492 | Interface *ADLParser::interface_parse(void) { |
3493 | char *iface_name = NULL__null; // Name of interface class being used |
3494 | char *iface_code = NULL__null; // Describe components of this class |
3495 | |
3496 | // Get interface class name |
3497 | skipws(); // Skip whitespace |
3498 | if (_curchar != '(') { |
3499 | parse_err(SYNERR1, "Missing '(' at start of interface description.\n"); |
3500 | return NULL__null; |
3501 | } |
3502 | next_char(); // move past '(' |
3503 | skipws(); |
3504 | iface_name = get_ident(); |
3505 | if (iface_name == NULL__null) { |
3506 | parse_err(SYNERR1, "missing interface name after 'interface'.\n"); |
3507 | return NULL__null; |
3508 | } |
3509 | skipws(); |
3510 | if (_curchar != ')') { |
3511 | parse_err(SYNERR1, "Missing ')' after name of interface.\n"); |
3512 | return NULL__null; |
3513 | } |
3514 | next_char(); // move past ')' |
3515 | |
3516 | // Get details of the interface, |
3517 | // for the type of interface indicated by iface_name. |
3518 | Interface *inter = NULL__null; |
3519 | skipws(); |
3520 | if ( _curchar != ';' ) { |
3521 | if ( strcmp(iface_name,"MEMORY_INTER") == 0 ) { |
3522 | inter = mem_interface_parse(); |
3523 | } |
3524 | else if ( strcmp(iface_name,"COND_INTER") == 0 ) { |
3525 | inter = cond_interface_parse(); |
3526 | } |
3527 | // The parse routines consume the "%}" |
3528 | |
3529 | // Check for probable extra ';' after defining block. |
3530 | if ( _curchar == ';' ) { |
3531 | parse_err(SYNERR1, "Extra ';' after defining interface block.\n"); |
3532 | next_char(); // Skip ';' |
3533 | return NULL__null; |
3534 | } |
3535 | } else { |
3536 | next_char(); // move past ';' |
3537 | |
3538 | // Create appropriate interface object |
3539 | if ( strcmp(iface_name,"REG_INTER") == 0 ) { |
3540 | inter = new RegInterface(); |
3541 | } |
3542 | else if ( strcmp(iface_name,"CONST_INTER") == 0 ) { |
3543 | inter = new ConstInterface(); |
3544 | } |
3545 | } |
3546 | skipws(); // be friendly to oper_parse() |
3547 | // Debug Stuff |
3548 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Interface Form: %s\n", iface_name); |
3549 | |
3550 | // Create appropriate interface object and return. |
3551 | return inter; |
3552 | } |
3553 | |
3554 | |
3555 | //------------------------------mem_interface_parse---------------------------- |
3556 | Interface *ADLParser::mem_interface_parse(void) { |
3557 | // Fields for MemInterface |
3558 | char *base = NULL__null; |
3559 | char *index = NULL__null; |
3560 | char *scale = NULL__null; |
3561 | char *disp = NULL__null; |
3562 | |
3563 | if (_curchar != '%') { |
3564 | parse_err(SYNERR1, "Missing '%%{' for 'interface' block.\n"); |
3565 | return NULL__null; |
3566 | } |
3567 | next_char(); // Skip '%' |
3568 | if (_curchar != '{') { |
3569 | parse_err(SYNERR1, "Missing '%%{' for 'interface' block.\n"); |
3570 | return NULL__null; |
3571 | } |
3572 | next_char(); // Skip '{' |
3573 | skipws(); |
3574 | do { |
3575 | char *field = get_ident(); |
3576 | if (field == NULL__null) { |
3577 | parse_err(SYNERR1, "Expected keyword, base|index|scale|disp, or '%%}' ending interface.\n"); |
3578 | return NULL__null; |
3579 | } |
3580 | if ( strcmp(field,"base") == 0 ) { |
3581 | base = interface_field_parse(); |
3582 | } |
3583 | else if ( strcmp(field,"index") == 0 ) { |
3584 | index = interface_field_parse(); |
3585 | } |
3586 | else if ( strcmp(field,"scale") == 0 ) { |
3587 | scale = interface_field_parse(); |
3588 | } |
3589 | else if ( strcmp(field,"disp") == 0 ) { |
3590 | disp = interface_field_parse(); |
3591 | } |
3592 | else { |
3593 | parse_err(SYNERR1, "Expected keyword, base|index|scale|disp, or '%%}' ending interface.\n"); |
3594 | return NULL__null; |
3595 | } |
3596 | } while( _curchar != '%' ); |
3597 | next_char(); // Skip '%' |
3598 | if ( _curchar != '}' ) { |
3599 | parse_err(SYNERR1, "Missing '%%}' for 'interface' block.\n"); |
3600 | return NULL__null; |
3601 | } |
3602 | next_char(); // Skip '}' |
3603 | |
3604 | // Construct desired object and return |
3605 | Interface *inter = new MemInterface(base, index, scale, disp); |
3606 | return inter; |
3607 | } |
3608 | |
3609 | |
3610 | //------------------------------cond_interface_parse--------------------------- |
3611 | Interface *ADLParser::cond_interface_parse(void) { |
3612 | char *equal; |
3613 | char *not_equal; |
3614 | char *less; |
3615 | char *greater_equal; |
3616 | char *less_equal; |
3617 | char *greater; |
3618 | char *overflow; |
3619 | char *no_overflow; |
3620 | const char *equal_format = "eq"; |
3621 | const char *not_equal_format = "ne"; |
3622 | const char *less_format = "lt"; |
3623 | const char *greater_equal_format = "ge"; |
3624 | const char *less_equal_format = "le"; |
3625 | const char *greater_format = "gt"; |
3626 | const char *overflow_format = "o"; |
3627 | const char *no_overflow_format = "no"; |
3628 | |
3629 | if (_curchar != '%') { |
3630 | parse_err(SYNERR1, "Missing '%%{' for 'cond_interface' block.\n"); |
3631 | return NULL__null; |
3632 | } |
3633 | next_char(); // Skip '%' |
3634 | if (_curchar != '{') { |
3635 | parse_err(SYNERR1, "Missing '%%{' for 'cond_interface' block.\n"); |
3636 | return NULL__null; |
3637 | } |
3638 | next_char(); // Skip '{' |
3639 | skipws(); |
3640 | do { |
3641 | char *field = get_ident(); |
3642 | if (field == NULL__null) { |
3643 | parse_err(SYNERR1, "Expected keyword, base|index|scale|disp, or '%%}' ending interface.\n"); |
3644 | return NULL__null; |
3645 | } |
3646 | if ( strcmp(field,"equal") == 0 ) { |
3647 | equal = interface_field_parse(&equal_format); |
3648 | } |
3649 | else if ( strcmp(field,"not_equal") == 0 ) { |
3650 | not_equal = interface_field_parse(¬_equal_format); |
3651 | } |
3652 | else if ( strcmp(field,"less") == 0 ) { |
3653 | less = interface_field_parse(&less_format); |
3654 | } |
3655 | else if ( strcmp(field,"greater_equal") == 0 ) { |
3656 | greater_equal = interface_field_parse(&greater_equal_format); |
3657 | } |
3658 | else if ( strcmp(field,"less_equal") == 0 ) { |
3659 | less_equal = interface_field_parse(&less_equal_format); |
3660 | } |
3661 | else if ( strcmp(field,"greater") == 0 ) { |
3662 | greater = interface_field_parse(&greater_format); |
3663 | } |
3664 | else if ( strcmp(field,"overflow") == 0 ) { |
3665 | overflow = interface_field_parse(&overflow_format); |
3666 | } |
3667 | else if ( strcmp(field,"no_overflow") == 0 ) { |
3668 | no_overflow = interface_field_parse(&no_overflow_format); |
3669 | } |
3670 | else { |
3671 | parse_err(SYNERR1, "Expected keyword, base|index|scale|disp, or '%%}' ending interface.\n"); |
3672 | return NULL__null; |
3673 | } |
3674 | } while( _curchar != '%' ); |
3675 | next_char(); // Skip '%' |
3676 | if ( _curchar != '}' ) { |
3677 | parse_err(SYNERR1, "Missing '%%}' for 'interface' block.\n"); |
3678 | return NULL__null; |
3679 | } |
3680 | next_char(); // Skip '}' |
3681 | |
3682 | // Construct desired object and return |
3683 | Interface *inter = new CondInterface(equal, equal_format, |
3684 | not_equal, not_equal_format, |
3685 | less, less_format, |
3686 | greater_equal, greater_equal_format, |
3687 | less_equal, less_equal_format, |
3688 | greater, greater_format, |
3689 | overflow, overflow_format, |
3690 | no_overflow, no_overflow_format); |
3691 | return inter; |
3692 | } |
3693 | |
3694 | |
3695 | //------------------------------interface_field_parse-------------------------- |
3696 | char *ADLParser::interface_field_parse(const char ** format) { |
3697 | char *iface_field = NULL__null; |
3698 | |
3699 | // Get interface field |
3700 | skipws(); // Skip whitespace |
3701 | if (_curchar != '(') { |
3702 | parse_err(SYNERR1, "Missing '(' at start of interface field.\n"); |
3703 | return NULL__null; |
3704 | } |
3705 | next_char(); // move past '(' |
3706 | skipws(); |
3707 | if ( _curchar != '0' && _curchar != '$' ) { |
3708 | parse_err(SYNERR1, "missing or invalid interface field contents.\n"); |
3709 | return NULL__null; |
3710 | } |
3711 | iface_field = get_rep_var_ident(); |
3712 | if (iface_field == NULL__null) { |
3713 | parse_err(SYNERR1, "missing or invalid interface field contents.\n"); |
3714 | return NULL__null; |
3715 | } |
3716 | skipws(); |
3717 | if (format != NULL__null && _curchar == ',') { |
3718 | next_char(); |
3719 | skipws(); |
3720 | if (_curchar != '"') { |
3721 | parse_err(SYNERR1, "Missing '\"' in field format .\n"); |
3722 | return NULL__null; |
3723 | } |
3724 | next_char(); |
3725 | char *start = _ptr; // Record start of the next string |
3726 | while ((_curchar != '"') && (_curchar != '%') && (_curchar != '\n')) { |
3727 | if (_curchar == '\\') next_char(); // superquote |
3728 | if (_curchar == '\n') parse_err(SYNERR1, "newline in string"); // unimplemented! |
3729 | next_char(); |
3730 | } |
3731 | if (_curchar != '"') { |
3732 | parse_err(SYNERR1, "Missing '\"' at end of field format .\n"); |
3733 | return NULL__null; |
3734 | } |
3735 | // If a string was found, terminate it and record in FormatRule |
3736 | if ( start != _ptr ) { |
3737 | *_ptr = '\0'; // Terminate the string |
3738 | *format = start; |
3739 | } |
3740 | next_char(); |
3741 | skipws(); |
3742 | } |
3743 | if (_curchar != ')') { |
3744 | parse_err(SYNERR1, "Missing ')' after interface field.\n"); |
3745 | return NULL__null; |
3746 | } |
3747 | next_char(); // move past ')' |
3748 | skipws(); |
3749 | if ( _curchar != ';' ) { |
3750 | parse_err(SYNERR1, "Missing ';' at end of interface field.\n"); |
3751 | return NULL__null; |
3752 | } |
3753 | next_char(); // move past ';' |
3754 | skipws(); // be friendly to interface_parse() |
3755 | |
3756 | return iface_field; |
3757 | } |
3758 | |
3759 | |
3760 | //------------------------------match_parse------------------------------------ |
3761 | MatchRule *ADLParser::match_parse(FormDict &operands) { |
3762 | MatchRule *match; // Match Rule class for instruction/operand |
3763 | char *cnstr = NULL__null; // Code for constructor |
3764 | int depth = 0; // Counter for matching parentheses |
3765 | int numleaves = 0; // Counter for number of leaves in rule |
3766 | |
3767 | // Parse the match rule tree |
3768 | MatchNode *mnode = matchNode_parse(operands, depth, numleaves, true); |
3769 | |
3770 | // Either there is a block with a constructor, or a ';' here |
3771 | skipws(); // Skip whitespace |
3772 | if ( _curchar == ';' ) { // Semicolon is valid terminator |
3773 | cnstr = NULL__null; // no constructor for this form |
3774 | next_char(); // Move past the ';', replaced with '\0' |
3775 | } |
3776 | else if ((cnstr = find_cpp_block("match constructor")) == NULL__null ) { |
3777 | parse_err(SYNERR1, "invalid construction of match rule\n" |
3778 | "Missing ';' or invalid '%%{' and '%%}' constructor\n"); |
3779 | return NULL__null; // No MatchRule to return |
3780 | } |
3781 | if (_AD._adl_debug > 1) |
3782 | if (cnstr) fprintf(stderrstderr,"Match Constructor: %s\n", cnstr); |
3783 | // Build new MatchRule object |
3784 | match = new MatchRule(_AD, mnode, depth, cnstr, numleaves); |
3785 | skipws(); // Skip any trailing whitespace |
3786 | return match; // Return MatchRule object |
3787 | } |
3788 | |
3789 | //------------------------------format_parse----------------------------------- |
3790 | FormatRule* ADLParser::format_parse(void) { |
3791 | char *desc = NULL__null; |
3792 | FormatRule *format = (new FormatRule(desc)); |
3793 | |
3794 | // Without expression form, MUST have a code block; |
3795 | skipws(); // Skip whitespace |
3796 | if ( _curchar == ';' ) { // Semicolon is valid terminator |
3797 | desc = NULL__null; // no constructor for this form |
3798 | next_char(); // Move past the ';', replaced with '\0' |
3799 | } |
3800 | else if ( _curchar == '%' && *(_ptr+1) == '{') { |
3801 | next_char(); // Move past the '%' |
3802 | next_char(); // Move past the '{' |
3803 | |
3804 | skipws(); |
3805 | if (_curchar == '$') { |
3806 | char* ident = get_rep_var_ident(); |
3807 | if (strcmp(ident, "$$template") == 0) return template_parse(); |
3808 | parse_err(SYNERR1, "Unknown \"%s\" directive in format", ident); |
3809 | return NULL__null; |
3810 | } |
3811 | // Check for the opening '"' inside the format description |
3812 | if ( _curchar == '"' ) { |
3813 | next_char(); // Move past the initial '"' |
3814 | if( _curchar == '"' ) { // Handle empty format string case |
3815 | *_ptr = '\0'; // Terminate empty string |
3816 | format->_strings.addName(_ptr); |
3817 | } |
3818 | |
3819 | // Collect the parts of the format description |
3820 | // (1) strings that are passed through to tty->print |
3821 | // (2) replacement/substitution variable, preceeded by a '$' |
3822 | // (3) multi-token ANSIY C style strings |
3823 | while ( true ) { |
3824 | if ( _curchar == '%' || _curchar == '\n' ) { |
3825 | if ( _curchar != '"' ) { |
3826 | parse_err(SYNERR1, "missing '\"' at end of format block"); |
3827 | return NULL__null; |
3828 | } |
3829 | } |
3830 | |
3831 | // (1) |
3832 | // Check if there is a string to pass through to output |
3833 | char *start = _ptr; // Record start of the next string |
3834 | while ((_curchar != '$') && (_curchar != '"') && (_curchar != '%') && (_curchar != '\n')) { |
3835 | if (_curchar == '\\') { |
3836 | next_char(); // superquote |
3837 | if ((_curchar == '$') || (_curchar == '%')) |
3838 | // hack to avoid % escapes and warnings about undefined \ escapes |
3839 | *(_ptr-1) = _curchar; |
3840 | } |
3841 | if (_curchar == '\n') parse_err(SYNERR1, "newline in string"); // unimplemented! |
3842 | next_char(); |
3843 | } |
3844 | // If a string was found, terminate it and record in FormatRule |
3845 | if ( start != _ptr ) { |
3846 | *_ptr = '\0'; // Terminate the string |
3847 | format->_strings.addName(start); |
3848 | } |
3849 | |
3850 | // (2) |
3851 | // If we are at a replacement variable, |
3852 | // copy it and record in FormatRule |
3853 | if ( _curchar == '$' ) { |
3854 | next_char(); // Move past the '$' |
3855 | char* rep_var = get_ident(); // Nil terminate the variable name |
3856 | rep_var = strdup(rep_var);// Copy the string |
3857 | *_ptr = _curchar; // and replace Nil with original character |
3858 | format->_rep_vars.addName(rep_var); |
3859 | // Add flag to _strings list indicating we should check _rep_vars |
3860 | format->_strings.addName(NameList::_signal); |
3861 | } |
3862 | |
3863 | // (3) |
3864 | // Allow very long strings to be broken up, |
3865 | // using the ANSI C syntax "foo\n" <newline> "bar" |
3866 | if ( _curchar == '"') { |
3867 | next_char(); // Move past the '"' |
3868 | skipws(); // Skip white space before next string token |
3869 | if ( _curchar != '"') { |
3870 | break; |
3871 | } else { |
3872 | // Found one. Skip both " and the whitespace in between. |
3873 | next_char(); |
3874 | } |
3875 | } |
3876 | } // end while part of format description |
3877 | |
3878 | // Check for closing '"' and '%}' in format description |
3879 | skipws(); // Move to closing '%}' |
3880 | if ( _curchar != '%' ) { |
3881 | parse_err(SYNERR1, "non-blank characters between closing '\"' and '%%' in format"); |
3882 | return NULL__null; |
3883 | } |
3884 | } // Done with format description inside |
3885 | |
3886 | skipws(); |
3887 | // Past format description, at '%' |
3888 | if ( _curchar != '%' || *(_ptr+1) != '}' ) { |
3889 | parse_err(SYNERR1, "missing '%%}' at end of format block"); |
3890 | return NULL__null; |
3891 | } |
3892 | next_char(); // Move past the '%' |
3893 | next_char(); // Move past the '}' |
3894 | } |
3895 | else { // parameter list alone must terminate with a ';' |
3896 | parse_err(SYNERR1, "missing ';' after Format expression"); |
3897 | return NULL__null; |
3898 | } |
3899 | // Debug Stuff |
3900 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Format Rule: %s\n", desc); |
3901 | |
3902 | skipws(); |
3903 | return format; |
3904 | } |
3905 | |
3906 | |
3907 | //------------------------------template_parse----------------------------------- |
3908 | FormatRule* ADLParser::template_parse(void) { |
3909 | char *desc = NULL__null; |
3910 | FormatRule *format = (new FormatRule(desc)); |
3911 | |
3912 | skipws(); |
3913 | while ( (_curchar != '%') && (*(_ptr+1) != '}') ) { |
3914 | |
3915 | // (1) |
3916 | // Check if there is a string to pass through to output |
3917 | { |
3918 | char *start = _ptr; // Record start of the next string |
3919 | while ((_curchar != '$') && ((_curchar != '%') || (*(_ptr+1) != '}')) ) { |
3920 | // If at the start of a comment, skip past it |
3921 | if( (_curchar == '/') && ((*(_ptr+1) == '/') || (*(_ptr+1) == '*')) ) { |
3922 | skipws_no_preproc(); |
3923 | } else { |
3924 | // ELSE advance to the next character, or start of the next line |
3925 | next_char_or_line(); |
3926 | } |
3927 | } |
3928 | // If a string was found, terminate it and record in EncClass |
3929 | if ( start != _ptr ) { |
3930 | *_ptr = '\0'; // Terminate the string |
3931 | // Add flag to _strings list indicating we should check _rep_vars |
3932 | format->_strings.addName(NameList::_signal2); |
3933 | format->_strings.addName(start); |
3934 | } |
3935 | } |
3936 | |
3937 | // (2) |
3938 | // If we are at a replacement variable, |
3939 | // copy it and record in EncClass |
3940 | if ( _curchar == '$' ) { |
3941 | // Found replacement Variable |
3942 | char *rep_var = get_rep_var_ident_dup(); |
3943 | if (strcmp(rep_var, "$emit") == 0) { |
3944 | // switch to normal format parsing |
3945 | next_char(); |
3946 | next_char(); |
3947 | skipws(); |
3948 | // Check for the opening '"' inside the format description |
3949 | if ( _curchar == '"' ) { |
3950 | next_char(); // Move past the initial '"' |
3951 | if( _curchar == '"' ) { // Handle empty format string case |
3952 | *_ptr = '\0'; // Terminate empty string |
3953 | format->_strings.addName(_ptr); |
3954 | } |
3955 | |
3956 | // Collect the parts of the format description |
3957 | // (1) strings that are passed through to tty->print |
3958 | // (2) replacement/substitution variable, preceeded by a '$' |
3959 | // (3) multi-token ANSIY C style strings |
3960 | while ( true ) { |
3961 | if ( _curchar == '%' || _curchar == '\n' ) { |
3962 | parse_err(SYNERR1, "missing '\"' at end of format block"); |
3963 | return NULL__null; |
3964 | } |
3965 | |
3966 | // (1) |
3967 | // Check if there is a string to pass through to output |
3968 | char *start = _ptr; // Record start of the next string |
3969 | while ((_curchar != '$') && (_curchar != '"') && (_curchar != '%') && (_curchar != '\n')) { |
3970 | if (_curchar == '\\') next_char(); // superquote |
3971 | if (_curchar == '\n') parse_err(SYNERR1, "newline in string"); // unimplemented! |
3972 | next_char(); |
3973 | } |
3974 | // If a string was found, terminate it and record in FormatRule |
3975 | if ( start != _ptr ) { |
3976 | *_ptr = '\0'; // Terminate the string |
3977 | format->_strings.addName(start); |
3978 | } |
3979 | |
3980 | // (2) |
3981 | // If we are at a replacement variable, |
3982 | // copy it and record in FormatRule |
3983 | if ( _curchar == '$' ) { |
3984 | next_char(); // Move past the '$' |
3985 | char* next_rep_var = get_ident(); // Nil terminate the variable name |
3986 | next_rep_var = strdup(next_rep_var);// Copy the string |
3987 | *_ptr = _curchar; // and replace Nil with original character |
3988 | format->_rep_vars.addName(next_rep_var); |
3989 | // Add flag to _strings list indicating we should check _rep_vars |
3990 | format->_strings.addName(NameList::_signal); |
3991 | } |
3992 | |
3993 | // (3) |
3994 | // Allow very long strings to be broken up, |
3995 | // using the ANSI C syntax "foo\n" <newline> "bar" |
3996 | if ( _curchar == '"') { |
3997 | next_char(); // Move past the '"' |
3998 | skipws(); // Skip white space before next string token |
3999 | if ( _curchar != '"') { |
4000 | break; |
4001 | } else { |
4002 | // Found one. Skip both " and the whitespace in between. |
4003 | next_char(); |
4004 | } |
4005 | } |
4006 | } // end while part of format description |
4007 | } |
4008 | } else { |
4009 | // Add flag to _strings list indicating we should check _rep_vars |
4010 | format->_rep_vars.addName(rep_var); |
4011 | // Add flag to _strings list indicating we should check _rep_vars |
4012 | format->_strings.addName(NameList::_signal3); |
4013 | } |
4014 | } // end while part of format description |
4015 | } |
4016 | |
4017 | skipws(); |
4018 | // Past format description, at '%' |
4019 | if ( _curchar != '%' || *(_ptr+1) != '}' ) { |
4020 | parse_err(SYNERR1, "missing '%%}' at end of format block"); |
4021 | return NULL__null; |
4022 | } |
4023 | next_char(); // Move past the '%' |
4024 | next_char(); // Move past the '}' |
4025 | |
4026 | // Debug Stuff |
4027 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Format Rule: %s\n", desc); |
4028 | |
4029 | skipws(); |
4030 | return format; |
4031 | } |
4032 | |
4033 | |
4034 | //------------------------------effect_parse----------------------------------- |
4035 | void ADLParser::effect_parse(InstructForm *instr) { |
4036 | char* desc = NULL__null; |
4037 | |
4038 | skipws(); // Skip whitespace |
4039 | if (_curchar != '(') { |
4040 | parse_err(SYNERR1, "missing '(' in effect definition\n"); |
4041 | return; |
4042 | } |
4043 | // Get list of effect-operand pairs and insert into dictionary |
4044 | else get_effectlist(instr->_effects, instr->_localNames, instr->_has_call); |
4045 | |
4046 | // Debug Stuff |
4047 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Effect description: %s\n", desc); |
4048 | if (_curchar != ';') { |
4049 | parse_err(SYNERR1, "missing ';' in Effect definition\n"); |
4050 | } |
4051 | next_char(); // Skip ';' |
4052 | |
4053 | } |
4054 | |
4055 | //------------------------------expand_parse----------------------------------- |
4056 | ExpandRule* ADLParser::expand_parse(InstructForm *instr) { |
4057 | char *ident, *ident2; |
4058 | NameAndList *instr_and_operands = NULL__null; |
4059 | ExpandRule *exp = new ExpandRule(); |
4060 | |
4061 | // Expand is a block containing an ordered list of operands with initializers, |
4062 | // or instructions, each of which has an ordered list of operands. |
4063 | // Check for block delimiter |
4064 | skipws(); // Skip leading whitespace |
4065 | if ((_curchar != '%') |
4066 | || (next_char(), (_curchar != '{')) ) { // If not open block |
4067 | parse_err(SYNERR1, "missing '%%{' in expand definition\n"); |
4068 | return(NULL__null); |
4069 | } |
4070 | next_char(); // Maintain the invariant |
4071 | do { |
4072 | ident = get_ident(); // Grab next identifier |
4073 | if (ident == NULL__null) { |
4074 | parse_err(SYNERR1, "identifier expected at %c\n", _curchar); |
4075 | continue; |
4076 | } |
4077 | |
4078 | // Check whether we should parse an instruction or operand. |
4079 | const Form *form = _globalNames[ident]; |
4080 | bool parse_oper = false; |
4081 | bool parse_ins = false; |
4082 | if (form == NULL__null) { |
4083 | skipws(); |
4084 | // Check whether this looks like an instruction specification. If so, |
4085 | // just parse the instruction. The declaration of the instruction is |
4086 | // not needed here. |
4087 | if (_curchar == '(') parse_ins = true; |
4088 | } else if (form->is_instruction()) { |
4089 | parse_ins = true; |
4090 | } else if (form->is_operand()) { |
4091 | parse_oper = true; |
4092 | } else { |
4093 | parse_err(SYNERR1, "instruction/operand name expected at %s\n", ident); |
4094 | continue; |
4095 | } |
4096 | |
4097 | if (parse_oper) { |
4098 | // This is a new operand |
4099 | OperandForm *oper = form->is_operand(); |
4100 | if (oper == NULL__null) { |
4101 | parse_err(SYNERR1, "instruction/operand name expected at %s\n", ident); |
4102 | continue; |
4103 | } |
4104 | // Throw the operand on the _newopers list |
4105 | skipws(); |
4106 | ident = get_unique_ident(instr->_localNames,"Operand"); |
4107 | if (ident == NULL__null) { |
4108 | parse_err(SYNERR1, "identifier expected at %c\n", _curchar); |
4109 | continue; |
4110 | } |
4111 | exp->_newopers.addName(ident); |
4112 | // Add new operand to LocalNames |
4113 | instr->_localNames.Insert(ident, oper); |
4114 | // Grab any constructor code and save as a string |
4115 | char *c = NULL__null; |
4116 | skipws(); |
4117 | if (_curchar == '%') { // Need a constructor for the operand |
4118 | c = find_cpp_block("Operand Constructor"); |
4119 | if (c == NULL__null) { |
4120 | parse_err(SYNERR1, "Invalid code block for operand constructor\n", _curchar); |
4121 | continue; |
4122 | } |
4123 | // Add constructor to _newopconst Dict |
4124 | exp->_newopconst.Insert(ident, c); |
4125 | } |
4126 | else if (_curchar != ';') { // If no constructor, need a ; |
4127 | parse_err(SYNERR1, "Missing ; in expand rule operand declaration\n"); |
4128 | continue; |
4129 | } |
4130 | else next_char(); // Skip the ; |
4131 | skipws(); |
4132 | } |
4133 | else { |
4134 | assert(parse_ins, "sanity"){ if (!(parse_ins)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4134, "sanity"); abort(); }}; |
4135 | // Add instruction to list |
4136 | instr_and_operands = new NameAndList(ident); |
4137 | // Grab operands, build nameList of them, and then put into dictionary |
4138 | skipws(); |
4139 | if (_curchar != '(') { // Check for parenthesized operand list |
4140 | parse_err(SYNERR1, "missing '(' in expand instruction declaration\n"); |
4141 | continue; |
4142 | } |
4143 | do { |
4144 | next_char(); // skip open paren & comma characters |
4145 | skipws(); |
4146 | if (_curchar == ')') break; |
4147 | ident2 = get_ident(); |
4148 | skipws(); |
4149 | if (ident2 == NULL__null) { |
4150 | parse_err(SYNERR1, "identifier expected at %c\n", _curchar); |
4151 | continue; |
4152 | } // Check that you have a valid operand |
4153 | const Form *form2 = instr->_localNames[ident2]; |
4154 | if (!form2) { |
4155 | parse_err(SYNERR1, "operand name expected at %s\n", ident2); |
4156 | continue; |
4157 | } |
4158 | OperandForm *oper = form2->is_operand(); |
4159 | if (oper == NULL__null && !form2->is_opclass()) { |
4160 | parse_err(SYNERR1, "operand name expected at %s\n", ident2); |
4161 | continue; |
4162 | } // Add operand to list |
4163 | instr_and_operands->add_entry(ident2); |
4164 | } while(_curchar == ','); |
4165 | if (_curchar != ')') { |
4166 | parse_err(SYNERR1, "missing ')'in expand instruction declaration\n"); |
4167 | continue; |
4168 | } |
4169 | next_char(); |
4170 | if (_curchar != ';') { |
4171 | parse_err(SYNERR1, "missing ';'in expand instruction declaration\n"); |
4172 | continue; |
4173 | } |
4174 | next_char(); |
4175 | |
4176 | // Record both instruction name and its operand list |
4177 | exp->add_instruction(instr_and_operands); |
4178 | |
4179 | skipws(); |
4180 | } |
4181 | |
4182 | } while(_curchar != '%'); |
4183 | next_char(); |
4184 | if (_curchar != '}') { |
4185 | parse_err(SYNERR1, "missing '%%}' in expand rule definition\n"); |
4186 | return(NULL__null); |
4187 | } |
4188 | next_char(); |
4189 | |
4190 | // Debug Stuff |
4191 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Expand Rule:\n"); |
4192 | |
4193 | skipws(); |
4194 | return (exp); |
4195 | } |
4196 | |
4197 | //------------------------------rewrite_parse---------------------------------- |
4198 | RewriteRule* ADLParser::rewrite_parse(void) { |
4199 | char* params = NULL__null; |
4200 | char* desc = NULL__null; |
4201 | |
4202 | |
4203 | // This feature targeted for second generation description language. |
4204 | |
4205 | skipws(); // Skip whitespace |
4206 | // Get parameters for rewrite |
4207 | if ((params = get_paren_expr("rewrite parameters")) == NULL__null) { |
4208 | parse_err(SYNERR1, "missing '(' in rewrite rule\n"); |
4209 | return NULL__null; |
4210 | } |
4211 | // Debug Stuff |
4212 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Rewrite parameters: %s\n", params); |
4213 | |
4214 | // For now, grab entire block; |
4215 | skipws(); |
4216 | if ( (desc = find_cpp_block("rewrite block")) == NULL__null ) { |
4217 | parse_err(SYNERR1, "incorrect or missing block for 'rewrite'.\n"); |
4218 | return NULL__null; |
4219 | } |
4220 | // Debug Stuff |
4221 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Rewrite Rule: %s\n", desc); |
4222 | |
4223 | skipws(); |
4224 | return (new RewriteRule(params,desc)); |
4225 | } |
4226 | |
4227 | //------------------------------attr_parse------------------------------------- |
4228 | Attribute *ADLParser::attr_parse(char* ident) { |
4229 | Attribute *attrib; // Attribute class |
4230 | char *cost = NULL__null; // String representation of cost attribute |
4231 | |
4232 | skipws(); // Skip leading whitespace |
4233 | if ( (cost = get_paren_expr("attribute")) == NULL__null ) { |
4234 | parse_err(SYNERR1, "incorrect or missing expression for 'attribute'\n"); |
4235 | return NULL__null; |
4236 | } |
4237 | // Debug Stuff |
4238 | if (_AD._adl_debug > 1) fprintf(stderrstderr,"Attribute: %s\n", cost); |
4239 | if (_curchar != ';') { |
4240 | parse_err(SYNERR1, "missing ';' in attribute definition\n"); |
4241 | return NULL__null; |
4242 | } |
4243 | next_char(); // Point after the terminator |
4244 | |
4245 | skipws(); |
4246 | attrib = new Attribute(ident,cost,INS_ATTR0); // Build new predicate object |
4247 | return attrib; |
4248 | } |
4249 | |
4250 | |
4251 | //------------------------------matchNode_parse-------------------------------- |
4252 | MatchNode *ADLParser::matchNode_parse(FormDict &operands, int &depth, int &numleaves, bool atroot) { |
4253 | // Count depth of parenthesis nesting for both left and right children |
4254 | int lParens = depth; |
4255 | int rParens = depth; |
4256 | |
4257 | // MatchNode objects for left, right, and root of subtree. |
4258 | MatchNode *lChild = NULL__null; |
4259 | MatchNode *rChild = NULL__null; |
4260 | char *token; // Identifier which may be opcode or operand |
4261 | |
4262 | // Match expression starts with a '(' |
4263 | if (cur_char() != '(') |
4264 | return NULL__null; |
4265 | |
4266 | next_char(); // advance past '(' |
4267 | |
4268 | // Parse the opcode |
4269 | token = get_ident(); // Get identifier, opcode |
4270 | if (token == NULL__null) { |
4271 | parse_err(SYNERR1, "missing opcode in match expression\n"); |
4272 | return NULL__null; |
4273 | } |
4274 | |
4275 | // Take note if we see one of a few special operations - those that are |
4276 | // treated differently on different architectures in the sense that on |
4277 | // one architecture there is a match rule and on another there isn't (so |
4278 | // a call will eventually be generated). |
4279 | |
4280 | for (int i = _last_machine_leaf + 1; i < _last_opcode; i++) { |
4281 | if (strcmp(token, NodeClassNames[i]) == 0) { |
4282 | _AD.has_match_rule(i, true); |
4283 | } |
4284 | } |
4285 | |
4286 | // Lookup the root value in the operands dict to perform substitution |
4287 | const char *result = NULL__null; // Result type will be filled in later |
4288 | const char *name = token; // local name associated with this node |
4289 | const char *operation = token; // remember valid operation for later |
4290 | const Form *form = operands[token]; |
4291 | OpClassForm *opcForm = form ? form->is_opclass() : NULL__null; |
4292 | if (opcForm != NULL__null) { |
4293 | // If this token is an entry in the local names table, record its type |
4294 | if (!opcForm->ideal_only()) { |
4295 | operation = opcForm->_ident; |
4296 | result = operation; // Operands result in their own type |
4297 | } |
4298 | // Otherwise it is an ideal type, and so, has no local name |
4299 | else name = NULL__null; |
4300 | } |
4301 | |
4302 | // Parse the operands |
4303 | skipws(); |
4304 | if (cur_char() != ')') { |
4305 | |
4306 | // Parse the left child |
4307 | if (strcmp(operation,"Set")) |
4308 | lChild = matchChild_parse(operands, lParens, numleaves, false); |
4309 | else |
4310 | lChild = matchChild_parse(operands, lParens, numleaves, true); |
4311 | |
4312 | skipws(); |
4313 | if (cur_char() != ')' ) { |
4314 | if(strcmp(operation, "Set")) |
4315 | rChild = matchChild_parse(operands,rParens,numleaves,false); |
4316 | else |
4317 | rChild = matchChild_parse(operands,rParens,numleaves,true); |
4318 | } |
4319 | } |
4320 | |
4321 | // Check for required ')' |
4322 | skipws(); |
4323 | if (cur_char() != ')') { |
4324 | parse_err(SYNERR1, "missing ')' in match expression\n"); |
4325 | return NULL__null; |
4326 | } |
4327 | next_char(); // skip the ')' |
4328 | |
4329 | MatchNode* mroot = new MatchNode(_AD,result,name,operation,lChild,rChild); |
4330 | |
4331 | // If not the root, reduce this subtree to an internal operand |
4332 | if (!atroot) { |
4333 | mroot->build_internalop(); |
4334 | } |
4335 | // depth is greater of left and right paths. |
4336 | depth = (lParens > rParens) ? lParens : rParens; |
4337 | |
4338 | return mroot; |
4339 | } |
4340 | |
4341 | |
4342 | //------------------------------matchChild_parse------------------------------- |
4343 | MatchNode *ADLParser::matchChild_parse(FormDict &operands, int &parens, int &numleaves, bool atroot) { |
4344 | MatchNode *child = NULL__null; |
4345 | const char *result = NULL__null; |
4346 | const char *token = NULL__null; |
4347 | const char *opType = NULL__null; |
4348 | |
4349 | if (cur_char() == '(') { // child is an operation |
4350 | ++parens; |
4351 | child = matchNode_parse(operands, parens, numleaves, atroot); |
4352 | } |
4353 | else { // child is an operand |
4354 | token = get_ident(); |
4355 | const Form *form = operands[token]; |
4356 | OpClassForm *opcForm = form ? form->is_opclass() : NULL__null; |
4357 | if (opcForm != NULL__null) { |
4358 | opType = opcForm->_ident; |
4359 | result = opcForm->_ident; // an operand's result matches its type |
4360 | } else { |
4361 | parse_err(SYNERR1, "undefined operand %s in match rule\n", token); |
4362 | return NULL__null; |
4363 | } |
4364 | |
4365 | if (opType == NULL__null) { |
4366 | parse_err(SYNERR1, "missing type for argument '%s'\n", token); |
4367 | } |
4368 | |
4369 | child = new MatchNode(_AD, result, token, opType); |
4370 | ++numleaves; |
4371 | } |
4372 | |
4373 | return child; |
4374 | } |
4375 | |
4376 | |
4377 | |
4378 | // ******************** Private Utility Functions ************************* |
4379 | |
4380 | |
4381 | char* ADLParser::find_cpp_block(const char* description) { |
4382 | char *next; // Pointer for finding block delimiters |
4383 | char* cppBlock = NULL__null; // Beginning of C++ code block |
4384 | |
4385 | if (_curchar == '%') { // Encoding is a C++ expression |
4386 | next_char(); |
4387 | if (_curchar != '{') { |
4388 | parse_err(SYNERR1, "missing '{' in %s \n", description); |
4389 | return NULL__null; |
4390 | } |
4391 | next_char(); // Skip block delimiter |
4392 | skipws_no_preproc(); // Skip leading whitespace |
4393 | cppBlock = _ptr; // Point to start of expression |
4394 | int line = linenum(); |
4395 | next = _ptr + 1; |
4396 | while(((_curchar != '%') || (*next != '}')) && (_curchar != '\0')) { |
4397 | next_char_or_line(); |
4398 | next = _ptr+1; // Maintain the next pointer |
4399 | } // Grab string |
4400 | if (_curchar == '\0') { |
4401 | parse_err(SYNERR1, "invalid termination of %s \n", description); |
4402 | return NULL__null; |
4403 | } |
4404 | *_ptr = '\0'; // Terminate string |
4405 | _ptr += 2; // Skip block delimiter |
4406 | _curchar = *_ptr; // Maintain invariant |
4407 | |
4408 | // Prepend location descriptor, for debugging. |
4409 | if (_AD._adlocation_debug) { |
4410 | char* location = get_line_string(line); |
4411 | char* end_loc = end_line_marker(); |
4412 | char* result = (char *)AllocateHeap(strlen(location) + strlen(cppBlock) + strlen(end_loc) + 1); |
4413 | strcpy(result, location); |
4414 | strcat(result, cppBlock); |
4415 | strcat(result, end_loc); |
4416 | cppBlock = result; |
4417 | free(location); |
4418 | } |
4419 | } |
4420 | |
4421 | return cppBlock; |
4422 | } |
4423 | |
4424 | // Move to the closing token of the expression we are currently at, |
4425 | // as defined by stop_chars. Match parens and quotes. |
4426 | char* ADLParser::get_expr(const char *desc, const char *stop_chars) { |
4427 | char* expr = NULL__null; |
4428 | int paren = 0; |
4429 | |
4430 | expr = _ptr; |
4431 | while (paren > 0 || !strchr(stop_chars, _curchar)) { |
4432 | if (_curchar == '(') { // Down level of nesting |
4433 | paren++; // Bump the parenthesis counter |
4434 | next_char(); // maintain the invariant |
4435 | } |
4436 | else if (_curchar == ')') { // Up one level of nesting |
4437 | if (paren == 0) { |
4438 | // Paren underflow: We didn't encounter the required stop-char. |
4439 | parse_err(SYNERR1, "too many )'s, did not find %s after %s\n", |
4440 | stop_chars, desc); |
4441 | return NULL__null; |
4442 | } |
4443 | paren--; // Drop the parenthesis counter |
4444 | next_char(); // Maintain the invariant |
4445 | } |
4446 | else if (_curchar == '"' || _curchar == '\'') { |
4447 | int qchar = _curchar; |
4448 | while (true) { |
4449 | next_char(); |
4450 | if (_curchar == qchar) { next_char(); break; } |
4451 | if (_curchar == '\\') next_char(); // superquote |
4452 | if (_curchar == '\n' || _curchar == '\0') { |
4453 | parse_err(SYNERR1, "newline in string in %s\n", desc); |
4454 | return NULL__null; |
4455 | } |
4456 | } |
4457 | } |
4458 | else if (_curchar == '%' && (_ptr[1] == '{' || _ptr[1] == '}')) { |
4459 | // Make sure we do not stray into the next ADLC-level form. |
4460 | parse_err(SYNERR1, "unexpected %%%c in %s\n", _ptr[1], desc); |
4461 | return NULL__null; |
4462 | } |
4463 | else if (_curchar == '\0') { |
4464 | parse_err(SYNERR1, "unexpected EOF in %s\n", desc); |
4465 | return NULL__null; |
4466 | } |
4467 | else { |
4468 | // Always walk over whitespace, comments, preprocessor directives, etc. |
4469 | char* pre_skip_ptr = _ptr; |
4470 | skipws(); |
4471 | // If the parser declined to make progress on whitespace, |
4472 | // skip the next character, which is therefore NOT whitespace. |
4473 | if (pre_skip_ptr == _ptr) { |
4474 | next_char(); |
4475 | } else if (pre_skip_ptr+strlen(pre_skip_ptr) != _ptr+strlen(_ptr)) { |
4476 | parse_err(SYNERR1, "unimplemented: preprocessor must not elide subexpression in %s", desc); |
4477 | } |
4478 | } |
4479 | } |
4480 | |
4481 | assert(strchr(stop_chars, _curchar), "non-null return must be at stop-char"){ if (!(strchr(stop_chars, _curchar))) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4481, "non-null return must be at stop-char"); abort(); }}; |
4482 | *_ptr = '\0'; // Replace ')' or other stop-char with '\0' |
4483 | return expr; |
4484 | } |
4485 | |
4486 | // Helper function around get_expr |
4487 | // Sets _curchar to '(' so that get_paren_expr will search for a matching ')' |
4488 | char *ADLParser::get_paren_expr(const char *description, bool include_location) { |
4489 | int line = linenum(); |
4490 | if (_curchar != '(') // Escape if not valid starting position |
4491 | return NULL__null; |
4492 | next_char(); // Skip the required initial paren. |
4493 | char *token2 = get_expr(description, ")"); |
4494 | if (_curchar == ')') |
4495 | next_char(); // Skip required final paren. |
4496 | int junk = 0; |
4497 | if (include_location && _AD._adlocation_debug && !is_int_token(token2, junk)) { |
4498 | // Prepend location descriptor, for debugging. |
4499 | char* location = get_line_string(line); |
4500 | char* end_loc = end_line_marker(); |
4501 | char* result = (char *)AllocateHeap(strlen(location) + strlen(token2) + strlen(end_loc) + 1); |
4502 | strcpy(result, location); |
4503 | strcat(result, token2); |
4504 | strcat(result, end_loc); |
4505 | token2 = result; |
4506 | free(location); |
4507 | } |
4508 | return token2; |
4509 | } |
4510 | |
4511 | //------------------------------get_ident_common------------------------------- |
4512 | // Looks for an identifier in the buffer, and turns it into a null terminated |
4513 | // string(still inside the file buffer). Returns a pointer to the string or |
4514 | // NULL if some other token is found instead. |
4515 | char *ADLParser::get_ident_common(bool do_preproc) { |
4516 | char c; |
4517 | char *start; // Pointer to start of token |
4518 | char *end; // Pointer to end of token |
4519 | |
4520 | if( _curline == NULL__null ) // Return NULL at EOF. |
4521 | return NULL__null; |
4522 | |
4523 | skipws_common(do_preproc); // Skip whitespace before identifier |
4524 | start = end = _ptr; // Start points at first character |
4525 | end--; // unwind end by one to prepare for loop |
4526 | do { |
4527 | end++; // Increment end pointer |
4528 | c = *end; // Grab character to test |
4529 | } while ( ((c >= 'a') && (c <= 'z')) || ((c >= 'A') && (c <= 'Z')) |
4530 | || ((c >= '0') && (c <= '9')) |
4531 | || ((c == '_')) || ((c == ':')) || ((c == '#')) ); |
4532 | if (start == end) { // We popped out on the first try |
4533 | // It can occur that `start' contains the rest of the input file. |
4534 | // In this case the output should be truncated. |
4535 | if (strlen(start) > 24) { |
4536 | char buf[32]; |
4537 | strncpy(buf, start, 20); |
4538 | buf[20] = '\0'; |
4539 | strcat(buf, "[...]"); |
4540 | parse_err(SYNERR1, "Identifier expected, but found '%s'.", buf); |
4541 | } else { |
4542 | parse_err(SYNERR1, "Identifier expected, but found '%s'.", start); |
4543 | } |
4544 | start = NULL__null; |
4545 | } |
4546 | else { |
4547 | _curchar = c; // Save the first character of next token |
4548 | *end = '\0'; // NULL terminate the string in place |
4549 | } |
4550 | _ptr = end; // Reset _ptr to point to next char after token |
4551 | |
4552 | // Make sure we do not try to use #defined identifiers. If start is |
4553 | // NULL an error was already reported. |
4554 | if (do_preproc && start != NULL__null) { |
4555 | const char* def = _AD.get_preproc_def(start); |
4556 | if (def != NULL__null && strcmp(def, start)) { |
4557 | const char* def1 = def; |
4558 | const char* def2 = _AD.get_preproc_def(def1); |
4559 | // implement up to 2 levels of #define |
4560 | if (def2 != NULL__null && strcmp(def2, def1)) { |
4561 | def = def2; |
4562 | const char* def3 = _AD.get_preproc_def(def2); |
4563 | if (def3 != NULL__null && strcmp(def3, def2) && strcmp(def3, def1)) { |
4564 | parse_err(SYNERR1, "unimplemented: using %s defined as %s => %s => %s", |
4565 | start, def1, def2, def3); |
4566 | } |
4567 | } |
4568 | start = strdup(def); |
4569 | } |
4570 | } |
4571 | |
4572 | return start; // Pointer to token in filebuf |
4573 | } |
4574 | |
4575 | //------------------------------get_ident_dup---------------------------------- |
4576 | // Looks for an identifier in the buffer, and returns a duplicate |
4577 | // or NULL if some other token is found instead. |
4578 | char *ADLParser::get_ident_dup(void) { |
4579 | char *ident = get_ident(); |
4580 | |
4581 | // Duplicate an identifier before returning and restore string. |
4582 | if( ident != NULL__null ) { |
4583 | ident = strdup(ident); // Copy the string |
4584 | *_ptr = _curchar; // and replace Nil with original character |
4585 | } |
4586 | |
4587 | return ident; |
4588 | } |
4589 | |
4590 | //----------------------get_ident_or_literal_constant-------------------------- |
4591 | // Looks for an identifier in the buffer, or a parenthesized expression. |
4592 | char *ADLParser::get_ident_or_literal_constant(const char* description) { |
4593 | char* param = NULL__null; |
4594 | skipws(); |
4595 | if (_curchar == '(') { |
4596 | // Grab a constant expression. |
4597 | param = get_paren_expr(description); |
4598 | if (param[0] != '(') { |
4599 | char* buf = (char*) AllocateHeap(strlen(param) + 3); |
4600 | sprintf(buf, "(%s)", param); |
4601 | param = buf; |
4602 | } |
4603 | assert(is_literal_constant(param),{ if (!(is_literal_constant(param))) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4604, "expr must be recognizable as a constant"); abort(); } } |
4604 | "expr must be recognizable as a constant"){ if (!(is_literal_constant(param))) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4604, "expr must be recognizable as a constant"); abort(); } }; |
4605 | } else { |
4606 | param = get_ident(); |
4607 | } |
4608 | return param; |
4609 | } |
4610 | |
4611 | //------------------------------get_rep_var_ident----------------------------- |
4612 | // Do NOT duplicate, |
4613 | // Leave nil terminator in buffer |
4614 | // Preserve initial '$'(s) in string |
4615 | char *ADLParser::get_rep_var_ident(void) { |
4616 | // Remember starting point |
4617 | char *rep_var = _ptr; |
4618 | |
4619 | // Check for replacement variable indicator '$' and pass if present |
4620 | if ( _curchar == '$' ) { |
4621 | next_char(); |
4622 | } |
4623 | // Check for a subfield indicator, a second '$', and pass if present |
4624 | if ( _curchar == '$' ) { |
4625 | next_char(); |
4626 | } |
4627 | |
4628 | // Check for a control indicator, a third '$': |
4629 | if ( _curchar == '$' ) { |
4630 | next_char(); |
4631 | } |
4632 | |
4633 | // Check for more than three '$'s in sequence, SYNERR |
4634 | if( _curchar == '$' ) { |
4635 | parse_err(SYNERR1, "Replacement variables and field specifiers can not start with '$$$$'"); |
4636 | next_char(); |
4637 | return NULL__null; |
4638 | } |
4639 | |
4640 | // Nil terminate the variable name following the '$' |
4641 | char *rep_var_name = get_ident(); |
4642 | assert( rep_var_name != NULL,{ if (!(rep_var_name != __null)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4643, "Missing identifier after replacement variable indicator '$'" ); abort(); }} |
4643 | "Missing identifier after replacement variable indicator '$'"){ if (!(rep_var_name != __null)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4643, "Missing identifier after replacement variable indicator '$'" ); abort(); }}; |
4644 | |
4645 | return rep_var; |
4646 | } |
4647 | |
4648 | |
4649 | |
4650 | //------------------------------get_rep_var_ident_dup------------------------- |
4651 | // Return the next replacement variable identifier, skipping first '$' |
4652 | // given a pointer into a line of the buffer. |
4653 | // Null terminates string, still inside the file buffer, |
4654 | // Returns a pointer to a copy of the string, or NULL on failure |
4655 | char *ADLParser::get_rep_var_ident_dup(void) { |
4656 | if( _curchar != '$' ) return NULL__null; |
4657 | |
4658 | next_char(); // Move past the '$' |
4659 | char *rep_var = _ptr; // Remember starting point |
4660 | |
4661 | // Check for a subfield indicator, a second '$': |
4662 | if ( _curchar == '$' ) { |
4663 | next_char(); |
4664 | } |
4665 | |
4666 | // Check for a control indicator, a third '$': |
4667 | if ( _curchar == '$' ) { |
4668 | next_char(); |
4669 | } |
4670 | |
4671 | // Check for more than three '$'s in sequence, SYNERR |
4672 | if( _curchar == '$' ) { |
4673 | parse_err(SYNERR1, "Replacement variables and field specifiers can not start with '$$$$'"); |
4674 | next_char(); |
4675 | return NULL__null; |
4676 | } |
4677 | |
4678 | // Nil terminate the variable name following the '$' |
4679 | char *rep_var_name = get_ident(); |
4680 | assert( rep_var_name != NULL,{ if (!(rep_var_name != __null)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4681, "Missing identifier after replacement variable indicator '$'" ); abort(); }} |
4681 | "Missing identifier after replacement variable indicator '$'"){ if (!(rep_var_name != __null)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 4681, "Missing identifier after replacement variable indicator '$'" ); abort(); }}; |
4682 | rep_var = strdup(rep_var); // Copy the string |
4683 | *_ptr = _curchar; // and replace Nil with original character |
4684 | |
4685 | return rep_var; |
4686 | } |
4687 | |
4688 | |
4689 | //------------------------------get_unique_ident------------------------------ |
4690 | // Looks for an identifier in the buffer, terminates it with a NULL, |
4691 | // and checks that it is unique |
4692 | char *ADLParser::get_unique_ident(FormDict& dict, const char* nameDescription){ |
4693 | char* ident = get_ident(); |
4694 | |
4695 | if (ident == NULL__null) { |
4696 | parse_err(SYNERR1, "missing %s identifier at %c\n", nameDescription, _curchar); |
4697 | } |
4698 | else { |
4699 | if (dict[ident] != NULL__null) { |
4700 | parse_err(SYNERR1, "duplicate name %s for %s\n", ident, nameDescription); |
4701 | ident = NULL__null; |
4702 | } |
4703 | } |
4704 | |
4705 | return ident; |
4706 | } |
4707 | |
4708 | |
4709 | //------------------------------get_int---------------------------------------- |
4710 | // Looks for a character string integer in the buffer, and turns it into an int |
4711 | // invokes a parse_err if the next token is not an integer. |
4712 | // This routine does not leave the integer null-terminated. |
4713 | int ADLParser::get_int(void) { |
4714 | char c; |
4715 | char *start; // Pointer to start of token |
4716 | char *end; // Pointer to end of token |
4717 | int result; // Storage for integer result |
4718 | |
4719 | if( _curline == NULL__null ) // Return NULL at EOF. |
4720 | return 0; |
4721 | |
4722 | skipws(); // Skip whitespace before identifier |
4723 | start = end = _ptr; // Start points at first character |
4724 | c = *end; // Grab character to test |
4725 | while ((c >= '0' && c <= '9') || (c == '-' && end == start)) { |
4726 | end++; // Increment end pointer |
4727 | c = *end; // Grab character to test |
4728 | } |
4729 | if (start == end) { // We popped out on the first try |
4730 | parse_err(SYNERR1, "integer expected at %c\n", c); |
4731 | result = 0; |
4732 | } |
4733 | else { |
4734 | _curchar = c; // Save the first character of next token |
4735 | *end = '\0'; // NULL terminate the string in place |
4736 | result = atoi(start); // Convert the string to an integer |
4737 | *end = _curchar; // Restore buffer to original condition |
4738 | } |
4739 | |
4740 | // Reset _ptr to next char after token |
4741 | _ptr = end; |
4742 | |
4743 | return result; // integer |
4744 | } |
4745 | |
4746 | |
4747 | //------------------------------get_relation_dup------------------------------ |
4748 | // Looks for a relational operator in the buffer |
4749 | // invokes a parse_err if the next token is not a relation |
4750 | // This routine creates a duplicate of the string in the buffer. |
4751 | char *ADLParser::get_relation_dup(void) { |
4752 | char *result = NULL__null; // relational operator being returned |
4753 | |
4754 | if( _curline == NULL__null ) // Return NULL at EOF. |
4755 | return NULL__null; |
4756 | |
4757 | skipws(); // Skip whitespace before relation |
4758 | char *start = _ptr; // Store start of relational operator |
4759 | char first = *_ptr; // the first character |
4760 | if( (first == '=') || (first == '!') || (first == '<') || (first == '>') ) { |
4761 | next_char(); |
4762 | char second = *_ptr; // the second character |
4763 | if( second == '=' ) { |
4764 | next_char(); |
4765 | char tmp = *_ptr; |
4766 | *_ptr = '\0'; // NULL terminate |
4767 | result = strdup(start); // Duplicate the string |
4768 | *_ptr = tmp; // restore buffer |
4769 | } else { |
4770 | parse_err(SYNERR1, "relational operator expected at %s\n", _ptr); |
4771 | } |
4772 | } else { |
4773 | parse_err(SYNERR1, "relational operator expected at %s\n", _ptr); |
4774 | } |
4775 | |
4776 | return result; |
4777 | } |
4778 | |
4779 | |
4780 | |
4781 | //------------------------------get_oplist------------------------------------- |
4782 | // Looks for identifier pairs where first must be the name of an operand, and |
4783 | // second must be a name unique in the scope of this instruction. Stores the |
4784 | // names with a pointer to the OpClassForm of their type in a local name table. |
4785 | void ADLParser::get_oplist(NameList ¶meters, FormDict &operands) { |
4786 | OpClassForm *opclass = NULL__null; |
4787 | char *ident = NULL__null; |
4788 | |
4789 | do { |
4790 | next_char(); // skip open paren & comma characters |
4791 | skipws(); |
4792 | if (_curchar == ')') break; |
4793 | |
4794 | // Get operand type, and check it against global name table |
4795 | ident = get_ident(); |
4796 | if (ident == NULL__null) { |
4797 | parse_err(SYNERR1, "optype identifier expected at %c\n", _curchar); |
4798 | return; |
4799 | } |
4800 | else { |
4801 | const Form *form = _globalNames[ident]; |
4802 | if( form == NULL__null ) { |
4803 | parse_err(SYNERR1, "undefined operand type %s\n", ident); |
4804 | return; |
4805 | } |
4806 | |
4807 | // Check for valid operand type |
4808 | OpClassForm *opc = form->is_opclass(); |
4809 | OperandForm *oper = form->is_operand(); |
4810 | if((oper == NULL__null) && (opc == NULL__null)) { |
4811 | parse_err(SYNERR1, "identifier %s not operand type\n", ident); |
4812 | return; |
4813 | } |
4814 | opclass = opc; |
4815 | } |
4816 | // Debugging Stuff |
4817 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "\tOperand Type: %s\t", ident); |
4818 | |
4819 | // Get name of operand and add it to local name table |
4820 | if( (ident = get_unique_ident(operands, "operand")) == NULL__null) { |
4821 | return; |
4822 | } |
4823 | // Parameter names must not be global names. |
4824 | if( _globalNames[ident] != NULL__null ) { |
4825 | parse_err(SYNERR1, "Reuse of global name %s as operand.\n",ident); |
4826 | return; |
4827 | } |
4828 | operands.Insert(ident, opclass); |
4829 | parameters.addName(ident); |
4830 | |
4831 | // Debugging Stuff |
4832 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "\tOperand Name: %s\n", ident); |
4833 | skipws(); |
4834 | } while(_curchar == ','); |
4835 | |
4836 | if (_curchar != ')') parse_err(SYNERR1, "missing ')'\n"); |
4837 | else { |
4838 | next_char(); // set current character position past the close paren |
4839 | } |
4840 | } |
4841 | |
4842 | |
4843 | //------------------------------get_effectlist--------------------------------- |
4844 | // Looks for identifier pairs where first must be the name of a pre-defined, |
4845 | // effect, and the second must be the name of an operand defined in the |
4846 | // operand list of this instruction. Stores the names with a pointer to the |
4847 | // effect form in a local effects table. |
4848 | void ADLParser::get_effectlist(FormDict &effects, FormDict &operands, bool& has_call) { |
4849 | OperandForm *opForm; |
4850 | Effect *eForm; |
4851 | char *ident; |
4852 | |
4853 | do { |
4854 | next_char(); // skip open paren & comma characters |
4855 | skipws(); |
4856 | if (_curchar == ')') break; |
4857 | |
4858 | // Get effect type, and check it against global name table |
4859 | ident = get_ident(); |
4860 | if (ident == NULL__null) { |
4861 | parse_err(SYNERR1, "effect type identifier expected at %c\n", _curchar); |
4862 | return; |
4863 | } |
4864 | else { |
4865 | // Check for valid effect type |
4866 | const Form *form = _globalNames[ident]; |
4867 | if( form == NULL__null ) { |
4868 | parse_err(SYNERR1, "undefined effect type %s\n", ident); |
4869 | return; |
4870 | } |
4871 | else { |
4872 | if( (eForm = form->is_effect()) == NULL__null) { |
4873 | parse_err(SYNERR1, "identifier %s not effect type\n", ident); |
4874 | return; |
4875 | } |
4876 | } |
4877 | } |
4878 | // Debugging Stuff |
4879 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "\tEffect Type: %s\t", ident); |
4880 | skipws(); |
4881 | if (eForm->is(Component::CALL)) { |
4882 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "\n"); |
4883 | has_call = true; |
4884 | } else { |
4885 | // Get name of operand and check that it is in the local name table |
4886 | if( (ident = get_unique_ident(effects, "effect")) == NULL__null) { |
4887 | parse_err(SYNERR1, "missing operand identifier in effect list\n"); |
4888 | return; |
4889 | } |
4890 | const Form *form = operands[ident]; |
4891 | opForm = form ? form->is_operand() : NULL__null; |
4892 | if( opForm == NULL__null ) { |
4893 | if( form && form->is_opclass() ) { |
4894 | const char* cname = form->is_opclass()->_ident; |
4895 | parse_err(SYNERR1, "operand classes are illegal in effect lists (found %s %s)\n", cname, ident); |
4896 | } else { |
4897 | parse_err(SYNERR1, "undefined operand %s in effect list\n", ident); |
4898 | } |
4899 | return; |
4900 | } |
4901 | // Add the pair to the effects table |
4902 | effects.Insert(ident, eForm); |
4903 | // Debugging Stuff |
4904 | if (_AD._adl_debug > 1) fprintf(stderrstderr, "\tOperand Name: %s\n", ident); |
4905 | } |
4906 | skipws(); |
4907 | } while(_curchar == ','); |
4908 | |
4909 | if (_curchar != ')') parse_err(SYNERR1, "missing ')'\n"); |
4910 | else { |
4911 | next_char(); // set current character position past the close paren |
4912 | } |
4913 | } |
4914 | |
4915 | |
4916 | //-------------------------------preproc_line---------------------------------- |
4917 | // A "#line" keyword has been seen, so parse the rest of the line. |
4918 | void ADLParser::preproc_line(void) { |
4919 | int line = get_int(); |
4920 | skipws_no_preproc(); |
4921 | const char* file = NULL__null; |
4922 | if (_curchar == '"') { |
4923 | next_char(); // Move past the initial '"' |
4924 | file = _ptr; |
4925 | while (true) { |
4926 | if (_curchar == '\n') { |
4927 | parse_err(SYNERR1, "missing '\"' at end of #line directive"); |
4928 | return; |
4929 | } |
4930 | if (_curchar == '"') { |
4931 | *_ptr = '\0'; // Terminate the string |
4932 | next_char(); |
4933 | skipws_no_preproc(); |
4934 | break; |
4935 | } |
4936 | next_char(); |
4937 | } |
4938 | } |
4939 | ensure_end_of_line(); |
4940 | if (file != NULL__null) |
4941 | _AD._ADL_file._name = file; |
4942 | _buf.set_linenum(line); |
4943 | } |
4944 | |
4945 | //------------------------------preproc_define--------------------------------- |
4946 | // A "#define" keyword has been seen, so parse the rest of the line. |
4947 | void ADLParser::preproc_define(void) { |
4948 | char* flag = get_ident_no_preproc(); |
4949 | skipws_no_preproc(); |
4950 | // only #define x y is supported for now |
4951 | char* def = get_ident_no_preproc(); |
4952 | _AD.set_preproc_def(flag, def); |
4953 | skipws_no_preproc(); |
4954 | if (_curchar != '\n') { |
4955 | parse_err(SYNERR1, "non-identifier in preprocessor definition\n"); |
4956 | } |
4957 | } |
4958 | |
4959 | //------------------------------preproc_undef---------------------------------- |
4960 | // An "#undef" keyword has been seen, so parse the rest of the line. |
4961 | void ADLParser::preproc_undef(void) { |
4962 | char* flag = get_ident_no_preproc(); |
4963 | skipws_no_preproc(); |
4964 | ensure_end_of_line(); |
4965 | _AD.set_preproc_def(flag, NULL__null); |
4966 | } |
4967 | |
4968 | |
4969 | |
4970 | //------------------------------parse_err-------------------------------------- |
4971 | // Issue a parser error message, and skip to the end of the current line |
4972 | void ADLParser::parse_err(int flag, const char *fmt, ...) { |
4973 | va_list args; |
4974 | |
4975 | va_start(args, fmt)__builtin_va_start(args, fmt); |
4976 | if (flag == 1) |
4977 | _AD._syntax_errs += _AD.emit_msg(0, flag, linenum(), fmt, args); |
4978 | else if (flag == 2) |
4979 | _AD._semantic_errs += _AD.emit_msg(0, flag, linenum(), fmt, args); |
4980 | else |
4981 | _AD._warnings += _AD.emit_msg(0, flag, linenum(), fmt, args); |
4982 | |
4983 | int error_char = _curchar; |
4984 | char* error_ptr = _ptr+1; |
4985 | for(;*_ptr != '\n'; _ptr++) ; // Skip to the end of the current line |
4986 | _curchar = '\n'; |
4987 | va_end(args)__builtin_va_end(args); |
4988 | _AD._no_output = 1; |
4989 | |
4990 | if (flag == 1) { |
4991 | char* error_tail = strchr(error_ptr, '\n'); |
4992 | char tem = *error_ptr; |
4993 | error_ptr[-1] = '\0'; |
4994 | char* error_head = error_ptr-1; |
4995 | while (error_head > _curline && *error_head) --error_head; |
4996 | if (error_tail) *error_tail = '\0'; |
4997 | fprintf(stderrstderr, "Error Context: %s>>>%c<<<%s\n", |
4998 | error_head, error_char, error_ptr); |
4999 | if (error_tail) *error_tail = '\n'; |
5000 | error_ptr[-1] = tem; |
5001 | } |
5002 | } |
5003 | |
5004 | //---------------------------ensure_start_of_line------------------------------ |
5005 | // A preprocessor directive has been encountered. Be sure it has fallen at |
5006 | // the beginning of a line, or else report an error. |
5007 | void ADLParser::ensure_start_of_line(void) { |
5008 | if (_curchar == '\n') { next_line(); return; } |
5009 | assert( _ptr >= _curline && _ptr < _curline+strlen(_curline),{ if (!(_ptr >= _curline && _ptr < _curline+strlen (_curline))) { fprintf(stderr, "assert fails %s %d: %s\n", "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 5010, "Must be able to find which line we are in"); abort() ; }} |
5010 | "Must be able to find which line we are in" ){ if (!(_ptr >= _curline && _ptr < _curline+strlen (_curline))) { fprintf(stderr, "assert fails %s %d: %s\n", "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 5010, "Must be able to find which line we are in"); abort() ; }}; |
5011 | |
5012 | for (char *s = _curline; s < _ptr; s++) { |
5013 | if (*s > ' ') { |
5014 | parse_err(SYNERR1, "'%c' must be at beginning of line\n", _curchar); |
5015 | break; |
5016 | } |
5017 | } |
5018 | } |
5019 | |
5020 | //---------------------------ensure_end_of_line-------------------------------- |
5021 | // A preprocessor directive has been parsed. Be sure there is no trailing |
5022 | // garbage at the end of this line. Set the scan point to the beginning of |
5023 | // the next line. |
5024 | void ADLParser::ensure_end_of_line(void) { |
5025 | skipws_no_preproc(); |
5026 | if (_curchar != '\n' && _curchar != '\0') { |
5027 | parse_err(SYNERR1, "garbage char '%c' at end of line\n", _curchar); |
5028 | } else { |
5029 | next_char_or_line(); |
5030 | } |
5031 | } |
5032 | |
5033 | //---------------------------handle_preproc------------------------------------ |
5034 | // The '#' character introducing a preprocessor directive has been found. |
5035 | // Parse the whole directive name (e.g., #define, #endif) and take appropriate |
5036 | // action. If we are in an "untaken" span of text, simply keep track of |
5037 | // #ifdef nesting structure, so we can find out when to start taking text |
5038 | // again. (In this state, we "sort of support" C's #if directives, enough |
5039 | // to disregard their associated #else and #endif lines.) If we are in a |
5040 | // "taken" span of text, there are two cases: "#define" and "#undef" |
5041 | // directives are preserved and passed up to the caller, which eventually |
5042 | // passes control to the top-level parser loop, which handles #define and |
5043 | // #undef directly. (This prevents these directives from occurring in |
5044 | // arbitrary positions in the AD file--we require better structure than C.) |
5045 | // In the other case, and #ifdef, #ifndef, #else, or #endif is silently |
5046 | // processed as whitespace, with the "taken" state of the text correctly |
5047 | // updated. This routine returns "false" exactly in the case of a "taken" |
5048 | // #define or #undef, which tells the caller that a preprocessor token |
5049 | // has appeared which must be handled explicitly by the parse loop. |
5050 | bool ADLParser::handle_preproc_token() { |
5051 | assert(*_ptr == '#', "must be at start of preproc"){ if (!(*_ptr == '#')) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 5051, "must be at start of preproc"); abort(); }}; |
5052 | ensure_start_of_line(); |
5053 | next_char(); |
5054 | skipws_no_preproc(); |
5055 | char* start_ident = _ptr; |
5056 | char* ident = (_curchar == '\n') ? NULL__null : get_ident_no_preproc(); |
5057 | if (ident == NULL__null) { |
5058 | parse_err(SYNERR1, "expected preprocessor command, got end of line\n"); |
5059 | } else if (!strcmp(ident, "ifdef") || |
5060 | !strcmp(ident, "ifndef")) { |
5061 | char* flag = get_ident_no_preproc(); |
5062 | ensure_end_of_line(); |
5063 | // Test the identifier only if we are already in taken code: |
5064 | bool flag_def = preproc_taken() && (_AD.get_preproc_def(flag) != NULL__null); |
5065 | bool now_taken = !strcmp(ident, "ifdef") ? flag_def : !flag_def; |
5066 | begin_if_def(now_taken); |
5067 | } else if (!strcmp(ident, "if")) { |
5068 | if (preproc_taken()) |
5069 | parse_err(SYNERR1, "unimplemented: #%s %s", ident, _ptr+1); |
5070 | next_line(); |
5071 | // Intelligently skip this nested C preprocessor directive: |
5072 | begin_if_def(true); |
5073 | } else if (!strcmp(ident, "else")) { |
5074 | ensure_end_of_line(); |
5075 | invert_if_def(); |
5076 | } else if (!strcmp(ident, "endif")) { |
5077 | ensure_end_of_line(); |
5078 | end_if_def(); |
5079 | } else if (preproc_taken()) { |
5080 | // pass this token up to the main parser as "#define" or "#undef" |
5081 | _ptr = start_ident; |
5082 | _curchar = *--_ptr; |
5083 | if( _curchar != '#' ) { |
5084 | parse_err(SYNERR1, "no space allowed after # in #define or #undef"); |
5085 | assert(_curchar == '#', "no space allowed after # in #define or #undef"){ if (!(_curchar == '#')) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 5085, "no space allowed after # in #define or #undef"); abort (); }}; |
5086 | } |
5087 | return false; |
5088 | } |
5089 | return true; |
5090 | } |
5091 | |
5092 | //---------------------------skipws_common------------------------------------- |
5093 | // Skip whitespace, including comments and newlines, while keeping an accurate |
5094 | // line count. |
5095 | // Maybe handle certain preprocessor constructs: #ifdef, #ifndef, #else, #endif |
5096 | void ADLParser::skipws_common(bool do_preproc) { |
5097 | char *start = _ptr; |
5098 | char *next = _ptr + 1; |
5099 | |
5100 | if (*_ptr == '\0') { |
5101 | // Check for string terminator |
5102 | if (_curchar > ' ') return; |
5103 | if (_curchar == '\n') { |
5104 | if (!do_preproc) return; // let caller handle the newline |
5105 | next_line(); |
5106 | _ptr = _curline; next = _ptr + 1; |
5107 | } |
5108 | else if (_curchar == '#' || |
5109 | (_curchar == '/' && (*next == '/' || *next == '*'))) { |
5110 | parse_err(SYNERR1, "unimplemented: comment token in a funny place"); |
5111 | } |
5112 | } |
5113 | while(_curline != NULL__null) { // Check for end of file |
5114 | if (*_ptr == '\n') { // keep proper track of new lines |
5115 | if (!do_preproc) break; // let caller handle the newline |
5116 | next_line(); |
5117 | _ptr = _curline; next = _ptr + 1; |
5118 | } |
5119 | else if ((*_ptr == '/') && (*next == '/')) // C++ comment |
5120 | do { _ptr++; next++; } while(*_ptr != '\n'); // So go to end of line |
5121 | else if ((*_ptr == '/') && (*next == '*')) { // C comment |
5122 | _ptr++; next++; |
5123 | do { |
5124 | _ptr++; next++; |
5125 | if (*_ptr == '\n') { // keep proper track of new lines |
5126 | next_line(); // skip newlines within comments |
5127 | if (_curline == NULL__null) { // check for end of file |
5128 | parse_err(SYNERR1, "end-of-file detected inside comment\n"); |
5129 | break; |
5130 | } |
5131 | _ptr = _curline; next = _ptr + 1; |
5132 | } |
5133 | } while(!((*_ptr == '*') && (*next == '/'))); // Go to end of comment |
5134 | _ptr = ++next; next++; // increment _ptr past comment end |
5135 | } |
5136 | else if (do_preproc && *_ptr == '#') { |
5137 | // Note that this calls skipws_common(false) recursively! |
5138 | bool preproc_handled = handle_preproc_token(); |
5139 | if (!preproc_handled) { |
5140 | if (preproc_taken()) { |
5141 | return; // short circuit |
5142 | } |
5143 | ++_ptr; // skip the preprocessor character |
5144 | } |
5145 | next = _ptr+1; |
5146 | } else if(*_ptr > ' ' && !(do_preproc && !preproc_taken())) { |
5147 | break; |
5148 | } |
5149 | else if (*_ptr == '"' || *_ptr == '\'') { |
5150 | assert(do_preproc, "only skip strings if doing preproc"){ if (!(do_preproc)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 5150, "only skip strings if doing preproc"); abort(); }}; |
5151 | // skip untaken quoted string |
5152 | int qchar = *_ptr; |
5153 | while (true) { |
5154 | ++_ptr; |
5155 | if (*_ptr == qchar) { ++_ptr; break; } |
5156 | if (*_ptr == '\\') ++_ptr; |
5157 | if (*_ptr == '\n' || *_ptr == '\0') { |
5158 | parse_err(SYNERR1, "newline in string"); |
5159 | break; |
5160 | } |
5161 | } |
5162 | next = _ptr + 1; |
5163 | } |
5164 | else { ++_ptr; ++next; } |
5165 | } |
5166 | if( _curline != NULL__null ) // at end of file _curchar isn't valid |
5167 | _curchar = *_ptr; // reset _curchar to maintain invariant |
5168 | } |
5169 | |
5170 | //---------------------------cur_char----------------------------------------- |
5171 | char ADLParser::cur_char() { |
5172 | return (_curchar); |
5173 | } |
5174 | |
5175 | //---------------------------next_char----------------------------------------- |
5176 | void ADLParser::next_char() { |
5177 | if (_curchar == '\n') parse_err(WARN0, "must call next_line!"); |
5178 | _curchar = *++_ptr; |
5179 | // if ( _curchar == '\n' ) { |
5180 | // next_line(); |
5181 | // } |
5182 | } |
5183 | |
5184 | //---------------------------next_char_or_line--------------------------------- |
5185 | void ADLParser::next_char_or_line() { |
5186 | if ( _curchar != '\n' ) { |
5187 | _curchar = *++_ptr; |
5188 | } else { |
5189 | next_line(); |
5190 | _ptr = _curline; |
5191 | _curchar = *_ptr; // maintain invariant |
5192 | } |
5193 | } |
5194 | |
5195 | //---------------------------next_line----------------------------------------- |
5196 | void ADLParser::next_line() { |
5197 | _curline = _buf.get_line(); |
5198 | _curchar = ' '; |
5199 | } |
5200 | |
5201 | //------------------------get_line_string-------------------------------------- |
5202 | // Prepended location descriptor, for debugging. |
5203 | // Must return a malloced string (that can be freed if desired). |
5204 | char* ADLParser::get_line_string(int linenum) { |
5205 | const char* file = _AD._ADL_file._name; |
5206 | int line = linenum ? linenum : this->linenum(); |
5207 | char* location = (char *)AllocateHeap(strlen(file) + 100); |
5208 | sprintf(location, "\n#line %d \"%s\"\n", line, file); |
5209 | return location; |
5210 | } |
5211 | |
5212 | //-------------------------is_literal_constant--------------------------------- |
5213 | bool ADLParser::is_literal_constant(const char *param) { |
5214 | if (param[0] == 0) return false; // null string |
5215 | if (param[0] == '(') return true; // parenthesized expression |
5216 | if (param[0] == '0' && (param[1] == 'x' || param[1] == 'X')) { |
5217 | // Make sure it's a hex constant. |
5218 | int i = 2; |
5219 | do { |
5220 | if( !ADLParser::is_hex_digit(*(param+i)) ) return false; |
5221 | ++i; |
5222 | } while( *(param+i) != 0 ); |
5223 | return true; |
5224 | } |
5225 | return false; |
5226 | } |
5227 | |
5228 | //---------------------------is_hex_digit-------------------------------------- |
5229 | bool ADLParser::is_hex_digit(char digit) { |
5230 | return ((digit >= '0') && (digit <= '9')) |
5231 | ||((digit >= 'a') && (digit <= 'f')) |
5232 | ||((digit >= 'A') && (digit <= 'F')); |
5233 | } |
5234 | |
5235 | //---------------------------is_int_token-------------------------------------- |
5236 | bool ADLParser::is_int_token(const char* token, int& intval) { |
5237 | const char* cp = token; |
5238 | while (*cp != '\0' && *cp <= ' ') cp++; |
5239 | if (*cp == '-') cp++; |
5240 | int ndigit = 0; |
5241 | while (*cp >= '0' && *cp <= '9') { cp++; ndigit++; } |
5242 | while (*cp != '\0' && *cp <= ' ') cp++; |
5243 | if (ndigit == 0 || *cp != '\0') { |
5244 | return false; |
5245 | } |
5246 | intval = atoi(token); |
5247 | return true; |
5248 | } |
5249 | |
5250 | static const char* skip_expr_ws(const char* str) { |
5251 | const char * cp = str; |
5252 | while (cp[0]) { |
5253 | if (cp[0] <= ' ') { |
5254 | ++cp; |
5255 | } else if (cp[0] == '#') { |
5256 | ++cp; |
5257 | while (cp[0] == ' ') ++cp; |
5258 | assert(0 == strncmp(cp, "line", 4), "must be a #line directive"){ if (!(0 == strncmp(cp, "line", 4))) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 5258, "must be a #line directive"); abort(); }}; |
5259 | const char* eol = strchr(cp, '\n'); |
5260 | assert(eol != NULL, "must find end of line"){ if (!(eol != __null)) { fprintf(stderr, "assert fails %s %d: %s\n" , "/home/daniel/Projects/java/jdk/src/hotspot/share/adlc/adlparse.cpp" , 5260, "must find end of line"); abort(); }}; |
5261 | if (eol == NULL__null) eol = cp + strlen(cp); |
5262 | cp = eol; |
5263 | } else { |
5264 | break; |
5265 | } |
5266 | } |
5267 | return cp; |
5268 | } |
5269 | |
5270 | //-----------------------equivalent_expressions-------------------------------- |
5271 | bool ADLParser::equivalent_expressions(const char* str1, const char* str2) { |
5272 | if (str1 == str2) |
5273 | return true; |
5274 | else if (str1 == NULL__null || str2 == NULL__null) |
5275 | return false; |
5276 | const char* cp1 = str1; |
5277 | const char* cp2 = str2; |
5278 | char in_quote = '\0'; |
5279 | while (cp1[0] && cp2[0]) { |
5280 | if (!in_quote) { |
5281 | // skip spaces and/or cpp directives |
5282 | const char* cp1a = skip_expr_ws(cp1); |
5283 | const char* cp2a = skip_expr_ws(cp2); |
5284 | if (cp1a > cp1 && cp2a > cp2) { |
5285 | cp1 = cp1a; cp2 = cp2a; |
5286 | continue; |
5287 | } |
5288 | if (cp1a > cp1 || cp2a > cp2) break; // fail |
5289 | } |
5290 | // match one non-space char |
5291 | if (cp1[0] != cp2[0]) break; // fail |
5292 | char ch = cp1[0]; |
5293 | cp1++; cp2++; |
5294 | // watch for quotes |
5295 | if (in_quote && ch == '\\') { |
5296 | if (cp1[0] != cp2[0]) break; // fail |
5297 | if (!cp1[0]) break; |
5298 | cp1++; cp2++; |
5299 | } |
5300 | if (in_quote && ch == in_quote) { |
5301 | in_quote = '\0'; |
5302 | } else if (!in_quote && (ch == '"' || ch == '\'')) { |
5303 | in_quote = ch; |
5304 | } |
5305 | } |
5306 | return (!cp1[0] && !cp2[0]); |
5307 | } |
5308 | |
5309 | |
5310 | //-------------------------------trim------------------------------------------ |
5311 | void ADLParser::trim(char* &token) { |
5312 | while (*token <= ' ') token++; |
5313 | char* end = token + strlen(token); |
5314 | while (end > token && *(end-1) <= ' ') --end; |
5315 | *end = '\0'; |
5316 | } |