File grammar.y changed (mode: 100644) (index 7d1cfd1..bfa2fb6) |
... |
... |
funcdefns: /* empty */ |
107 |
107 |
|
|
108 |
108 |
globdefs: /* empty */ |
globdefs: /* empty */ |
109 |
109 |
| GLOBALS newline vardecls ENDGLOBALS endglobalsmarker |
| GLOBALS newline vardecls ENDGLOBALS endglobalsmarker |
110 |
|
| GLOBALS vardecls ENDGLOBALS endglobalsmarker {yyerrorline(0, lineno - 1, "Missing linebreak before global declaration");} |
|
|
110 |
|
| GLOBALS vardecls ENDGLOBALS endglobalsmarker {yyerrorline(syntaxerror, lineno - 1, "Missing linebreak before global declaration");} |
111 |
111 |
; |
; |
112 |
112 |
|
|
113 |
113 |
endglobalsmarker: /* empty */ {afterendglobals = 1;} |
endglobalsmarker: /* empty */ {afterendglobals = 1;} |
|
... |
... |
expr: intexpr { $$.ty = gInteger; } |
147 |
147 |
char ebuf[1024]; |
char ebuf[1024]; |
148 |
148 |
snprintf(ebuf, 1024, "Undefined function %s", $2.str); |
snprintf(ebuf, 1024, "Undefined function %s", $2.str); |
149 |
149 |
getsuggestions($2.str, ebuf, 1024, 1, &functions); |
getsuggestions($2.str, ebuf, 1024, 1, &functions); |
150 |
|
yyerrorex(3, ebuf); |
|
|
150 |
|
yyerrorex(semanticerror, ebuf); |
151 |
151 |
$$.ty = gCode; |
$$.ty = gCode; |
152 |
152 |
} else { |
} else { |
153 |
153 |
if (fd->p->head != NULL) { |
if (fd->p->head != NULL) { |
154 |
154 |
char ebuf[1024]; |
char ebuf[1024]; |
155 |
155 |
snprintf(ebuf, 1024, "Function %s must not take any arguments when used as code", $2.str); |
snprintf(ebuf, 1024, "Function %s must not take any arguments when used as code", $2.str); |
156 |
|
yyerrorex(3, ebuf); |
|
|
156 |
|
yyerrorex(semanticerror, ebuf); |
157 |
157 |
} |
} |
158 |
158 |
if( fd->ret == gBoolean) { |
if( fd->ret == gBoolean) { |
159 |
159 |
$$.ty = gCodeReturnsBoolean; |
$$.ty = gCodeReturnsBoolean; |
|
... |
... |
expr: intexpr { $$.ty = gInteger; } |
189 |
189 |
if (!tan->isarray) { |
if (!tan->isarray) { |
190 |
190 |
char ebuf[1024]; |
char ebuf[1024]; |
191 |
191 |
snprintf(ebuf, 1024, "%s not an array", $1.str); |
snprintf(ebuf, 1024, "%s not an array", $1.str); |
192 |
|
yyerrorex(3, ebuf); |
|
|
192 |
|
yyerrorex(semanticerror, ebuf); |
193 |
193 |
} |
} |
194 |
194 |
else { |
else { |
195 |
195 |
canconvert($3.ty, gInteger, 0); |
canconvert($3.ty, gInteger, 0); |
|
... |
... |
expr: intexpr { $$.ty = gInteger; } |
202 |
202 |
if (tan->lineno == lineno && tan->fn == fno) { |
if (tan->lineno == lineno && tan->fn == fno) { |
203 |
203 |
char ebuf[1024]; |
char ebuf[1024]; |
204 |
204 |
snprintf(ebuf, 1024, "Use of variable %s before its declaration", $1.str); |
snprintf(ebuf, 1024, "Use of variable %s before its declaration", $1.str); |
205 |
|
yyerrorex(3, ebuf); |
|
|
205 |
|
yyerrorex(semanticerror, ebuf); |
206 |
206 |
} else if (islinebreak && tan->lineno == lineno - 1 && tan->fn == fno) { |
} else if (islinebreak && tan->lineno == lineno - 1 && tan->fn == fno) { |
207 |
207 |
char ebuf[1024]; |
char ebuf[1024]; |
208 |
208 |
snprintf(ebuf, 1024, "Use of variable %s before its declaration", $1.str); |
snprintf(ebuf, 1024, "Use of variable %s before its declaration", $1.str); |
209 |
|
yyerrorline(3, lineno - 1, ebuf); |
|
|
209 |
|
yyerrorline(semanticerror, lineno - 1, ebuf); |
210 |
210 |
} else if (tan->isarray) { |
} else if (tan->isarray) { |
211 |
211 |
char ebuf[1024]; |
char ebuf[1024]; |
212 |
212 |
snprintf(ebuf, 1024, "Index missing for array variable %s", $1.str); |
snprintf(ebuf, 1024, "Index missing for array variable %s", $1.str); |
213 |
|
yyerrorex(3, ebuf); |
|
|
213 |
|
yyerrorex(semanticerror, ebuf); |
214 |
214 |
} |
} |
215 |
215 |
if(infunction && ht_lookup(curtab, $1.str) && !ht_lookup(&initialized, $1.str) ){ |
if(infunction && ht_lookup(curtab, $1.str) && !ht_lookup(&initialized, $1.str) ){ |
216 |
216 |
char ebuf[1024]; |
char ebuf[1024]; |
217 |
217 |
snprintf(ebuf, 1024, "Variable %s is uninitialized", $1.str); |
snprintf(ebuf, 1024, "Variable %s is uninitialized", $1.str); |
218 |
218 |
//yyerrorex(3, ebuf); |
//yyerrorex(3, ebuf); |
219 |
|
yyerrorline(3, lineno - 1, ebuf); |
|
|
219 |
|
yyerrorline(semanticerror, lineno - 1, ebuf); |
220 |
220 |
} |
} |
221 |
221 |
$$.ty = tan->ty; |
$$.ty = tan->ty; |
222 |
222 |
} |
} |
223 |
|
| expr EQUALS expr {yyerrorex(0, "Single = in expression, should probably be =="); checkeqtest($1.ty, $3.ty); $$.ty = gBoolean;} |
|
224 |
|
| LPAREN expr {yyerrorex(0, "Mssing ')'"); $$.ty = $2.ty;} |
|
|
223 |
|
| expr EQUALS expr {yyerrorex(syntaxerror, "Single = in expression, should probably be =="); checkeqtest($1.ty, $3.ty); $$.ty = gBoolean;} |
|
224 |
|
| LPAREN expr {yyerrorex(syntaxerror, "Mssing ')'"); $$.ty = $2.ty;} |
225 |
225 |
|
|
226 |
226 |
// incomplete expressions |
// incomplete expressions |
227 |
|
| expr LEQ { checkcomparisonsimple($1.ty); yyerrorex(3, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
228 |
|
| expr GEQ { checkcomparisonsimple($1.ty); yyerrorex(3, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
229 |
|
| expr LESS { checkcomparisonsimple($1.ty); yyerrorex(3, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
230 |
|
| expr GREATER { checkcomparisonsimple($1.ty); yyerrorex(3, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
231 |
|
| expr EQCOMP { yyerrorex(3, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
232 |
|
| expr NEQ { yyerrorex(3, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
233 |
|
| expr AND { canconvert($1.ty, gBoolean, 0); yyerrorex(3, "Missing expression for logical and"); $$.ty = gBoolean; } |
|
234 |
|
| expr OR { canconvert($1.ty, gBoolean, 0); yyerrorex(3, "Missing expression for logical or"); $$.ty = gBoolean; } |
|
235 |
|
| NOT { yyerrorex(3, "Missing expression for logical negation"); $$.ty = gBoolean; } |
|
|
227 |
|
| expr LEQ { checkcomparisonsimple($1.ty); yyerrorex(syntaxerror, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
228 |
|
| expr GEQ { checkcomparisonsimple($1.ty); yyerrorex(syntaxerror, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
229 |
|
| expr LESS { checkcomparisonsimple($1.ty); yyerrorex(syntaxerror, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
230 |
|
| expr GREATER { checkcomparisonsimple($1.ty); yyerrorex(syntaxerror, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
231 |
|
| expr EQCOMP { yyerrorex(syntaxerror, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
232 |
|
| expr NEQ { yyerrorex(syntaxerror, "Missing expression for comparison"); $$.ty = gBoolean; } |
|
233 |
|
| expr AND { canconvert($1.ty, gBoolean, 0); yyerrorex(syntaxerror, "Missing expression for logical and"); $$.ty = gBoolean; } |
|
234 |
|
| expr OR { canconvert($1.ty, gBoolean, 0); yyerrorex(syntaxerror, "Missing expression for logical or"); $$.ty = gBoolean; } |
|
235 |
|
| NOT { yyerrorex(syntaxerror, "Missing expression for logical negation"); $$.ty = gBoolean; } |
236 |
236 |
; |
; |
237 |
237 |
|
|
238 |
238 |
funccall: rid LPAREN exprlistcompl RPAREN { |
funccall: rid LPAREN exprlistcompl RPAREN { |
239 |
239 |
$$ = checkfunccall($1.str, $3.pl); |
$$ = checkfunccall($1.str, $3.pl); |
240 |
240 |
} |
} |
241 |
241 |
| rid LPAREN exprlistcompl newline { |
| rid LPAREN exprlistcompl newline { |
242 |
|
yyerrorex(0, "Missing ')'"); |
|
|
242 |
|
yyerrorex(syntaxerror, "Missing ')'"); |
243 |
243 |
$$ = checkfunccall($1.str, $3.pl); |
$$ = checkfunccall($1.str, $3.pl); |
244 |
244 |
} |
} |
245 |
245 |
; |
; |
|
... |
... |
nativefuncdecl: NATIVE rid TAKES optparam_list RETURNS opttype |
281 |
281 |
if (ht_lookup(&locals, $2.str) || ht_lookup(¶ms, $2.str) || ht_lookup(&globals, $2.str)) { |
if (ht_lookup(&locals, $2.str) || ht_lookup(¶ms, $2.str) || ht_lookup(&globals, $2.str)) { |
282 |
282 |
char buf[1024]; |
char buf[1024]; |
283 |
283 |
snprintf(buf, 1024, "%s already defined as variable", $2.str); |
snprintf(buf, 1024, "%s already defined as variable", $2.str); |
284 |
|
yyerrorex(3, buf); |
|
|
284 |
|
yyerrorex(semanticerror, buf); |
285 |
285 |
} else if (ht_lookup(&types, $2.str)) { |
} else if (ht_lookup(&types, $2.str)) { |
286 |
286 |
char buf[1024]; |
char buf[1024]; |
287 |
287 |
snprintf(buf, 1024, "%s already defined as type", $2.str); |
snprintf(buf, 1024, "%s already defined as type", $2.str); |
288 |
|
yyerrorex(3, buf); |
|
|
288 |
|
yyerrorex(semanticerror, buf); |
289 |
289 |
} |
} |
290 |
290 |
$$.fd = newfuncdecl(); |
$$.fd = newfuncdecl(); |
291 |
291 |
$$.fd->name = strdup($2.str); |
$$.fd->name = strdup($2.str); |
|
... |
... |
nativefuncdecl: NATIVE rid TAKES optparam_list RETURNS opttype |
306 |
306 |
|
|
307 |
307 |
funcdefn: newline |
funcdefn: newline |
308 |
308 |
| funcdefncore |
| funcdefncore |
309 |
|
| statement { yyerrorex(0, "Statement outside of function"); } |
|
|
309 |
|
| statement { yyerrorex(syntaxerror, "Statement outside of function"); } |
310 |
310 |
; |
; |
311 |
311 |
|
|
312 |
312 |
funcdefncore: funcbegin localblock codeblock funcend { |
funcdefncore: funcbegin localblock codeblock funcend { |
313 |
313 |
if(retval != gNothing) { |
if(retval != gNothing) { |
314 |
314 |
if(!getTypeTag($3.ty)) |
if(!getTypeTag($3.ty)) |
315 |
|
yyerrorline(1, lineno - 1, "Missing return"); |
|
|
315 |
|
yyerrorline(semanticerror, lineno - 1, "Missing return"); |
316 |
316 |
else if ( flagenabled(flag_rb) ) |
else if ( flagenabled(flag_rb) ) |
317 |
317 |
canconvertreturn($3.ty, retval, -1); |
canconvertreturn($3.ty, retval, -1); |
318 |
318 |
} |
} |
319 |
319 |
fnannotations = 0; |
fnannotations = 0; |
320 |
320 |
} |
} |
321 |
|
| funcbegin localblock codeblock {yyerrorex(0, "Missing endfunction"); ht_clear(¶ms); ht_clear(&locals); ht_clear(&initialized); curtab = &globals; fnannotations = 0;} |
|
|
321 |
|
| funcbegin localblock codeblock {yyerrorex(syntaxerror, "Missing endfunction"); ht_clear(¶ms); ht_clear(&locals); ht_clear(&initialized); curtab = &globals; fnannotations = 0;} |
322 |
322 |
; |
; |
323 |
323 |
|
|
324 |
324 |
funcend: ENDFUNCTION { ht_clear(¶ms); ht_clear(&locals); ht_clear(&initialized); curtab = &globals; inblock = 0; inconstant = 0; infunction = 0; } |
funcend: ENDFUNCTION { ht_clear(¶ms); ht_clear(&locals); ht_clear(&initialized); curtab = &globals; inblock = 0; inconstant = 0; infunction = 0; } |
325 |
325 |
; |
; |
326 |
326 |
|
|
327 |
327 |
returnorreturns: RETURNS |
returnorreturns: RETURNS |
328 |
|
| RETURN {yyerrorex(3,"Expected \"returns\" instead of \"return\"");} |
|
|
328 |
|
| RETURN {yyerrorex(syntaxerror,"Expected \"returns\" instead of \"return\"");} |
329 |
329 |
; |
; |
330 |
330 |
|
|
331 |
331 |
funcbegin: FUNCTION rid TAKES optparam_list returnorreturns opttype { |
funcbegin: FUNCTION rid TAKES optparam_list returnorreturns opttype { |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
361 |
361 |
| SET rid EQUALS expr newline { if (getVariable($2.str)->isarray) { |
| SET rid EQUALS expr newline { if (getVariable($2.str)->isarray) { |
362 |
362 |
char ebuf[1024]; |
char ebuf[1024]; |
363 |
363 |
snprintf(ebuf, 1024, "Index missing for array variable %s", $2.str); |
snprintf(ebuf, 1024, "Index missing for array variable %s", $2.str); |
364 |
|
yyerrorline(3, lineno - 1, ebuf); |
|
|
364 |
|
yyerrorline(semanticerror, lineno - 1, ebuf); |
365 |
365 |
} |
} |
366 |
366 |
canconvert($4.ty, getVariable($2.str)->ty, -1); |
canconvert($4.ty, getVariable($2.str)->ty, -1); |
367 |
367 |
$$.ty = gAny; |
$$.ty = gAny; |
368 |
368 |
if (getVariable($2.str)->isconst) { |
if (getVariable($2.str)->isconst) { |
369 |
369 |
char ebuf[1024]; |
char ebuf[1024]; |
370 |
370 |
snprintf(ebuf, 1024, "Cannot assign to constant %s", $2.str); |
snprintf(ebuf, 1024, "Cannot assign to constant %s", $2.str); |
371 |
|
yyerrorline(3, lineno - 1, ebuf); |
|
|
371 |
|
yyerrorline(semanticerror, lineno - 1, ebuf); |
372 |
372 |
} |
} |
373 |
373 |
if (inconstant) |
if (inconstant) |
374 |
374 |
validateGlobalAssignment($2.str); |
validateGlobalAssignment($2.str); |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
384 |
384 |
if (!tan->isarray) { |
if (!tan->isarray) { |
385 |
385 |
char ebuf[1024]; |
char ebuf[1024]; |
386 |
386 |
snprintf(ebuf, 1024, "%s is not an array", $2.str); |
snprintf(ebuf, 1024, "%s is not an array", $2.str); |
387 |
|
yyerrorline(3, lineno - 1, ebuf); |
|
|
387 |
|
yyerrorline(semanticerror, lineno - 1, ebuf); |
388 |
388 |
} |
} |
389 |
389 |
canconvert($7.ty, tan->ty, -1); |
canconvert($7.ty, tan->ty, -1); |
390 |
390 |
if (inconstant) |
if (inconstant) |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
392 |
392 |
} |
} |
393 |
393 |
} |
} |
394 |
394 |
| loopstart newline codeblock loopend newline {$$.ty = $3.ty;} |
| loopstart newline codeblock loopend newline {$$.ty = $3.ty;} |
395 |
|
| loopstart newline codeblock {$$.ty = $3.ty; yyerrorex(0, "Missing endloop");} |
|
396 |
|
| EXITWHEN expr newline { canconvert($2.ty, gBoolean, -1); if (!inloop) yyerrorline(0, lineno - 1, "Exitwhen outside of loop"); $$.ty = gAny;} |
|
|
395 |
|
| loopstart newline codeblock {$$.ty = $3.ty; yyerrorex(syntaxerror, "Missing endloop");} |
|
396 |
|
| EXITWHEN expr newline { canconvert($2.ty, gBoolean, -1); if (!inloop) yyerrorline(syntaxerror, lineno - 1, "Exitwhen outside of loop"); $$.ty = gAny;} |
397 |
397 |
| RETURN expr newline { |
| RETURN expr newline { |
398 |
398 |
$$.ty = mkretty($2.ty, 1); |
$$.ty = mkretty($2.ty, 1); |
399 |
399 |
if(retval == gNothing) |
if(retval == gNothing) |
400 |
|
yyerrorline(1, lineno - 1, "Cannot return value from function that returns nothing"); |
|
|
400 |
|
yyerrorline(semanticerror, lineno - 1, "Cannot return value from function that returns nothing"); |
401 |
401 |
else if (! flagenabled(flag_rb) ) |
else if (! flagenabled(flag_rb) ) |
402 |
402 |
canconvertreturn($2.ty, retval, 0); |
canconvertreturn($2.ty, retval, 0); |
403 |
403 |
} |
} |
404 |
404 |
| RETURN newline { |
| RETURN newline { |
405 |
405 |
if (retval != gNothing) |
if (retval != gNothing) |
406 |
|
yyerrorline(1, lineno - 1, "Return nothing in function that should return value"); |
|
|
406 |
|
yyerrorline(semanticerror, lineno - 1, "Return nothing in function that should return value"); |
407 |
407 |
$$.ty = mkretty(gAny, 1); |
$$.ty = mkretty(gAny, 1); |
408 |
408 |
} |
} |
409 |
409 |
| DEBUG statement {$$.ty = gAny;} |
| DEBUG statement {$$.ty = gAny;} |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
411 |
411 |
| IF expr THEN newline codeblock elsifseq elseseq { |
| IF expr THEN newline codeblock elsifseq elseseq { |
412 |
412 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
413 |
413 |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
414 |
|
yyerrorex(0, "Missing endif"); |
|
|
414 |
|
yyerrorex(syntaxerror, "Missing endif"); |
415 |
415 |
} |
} |
416 |
416 |
| IF expr newline { |
| IF expr newline { |
417 |
417 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
418 |
418 |
$$.ty = gAny; |
$$.ty = gAny; |
419 |
|
yyerrorex(0, "Missing then or non valid expression"); |
|
|
419 |
|
yyerrorex(syntaxerror, "Missing then or non valid expression"); |
420 |
420 |
} |
} |
421 |
|
| SET funccall newline{ $$.ty = gAny; yyerrorline(0, lineno - 1, "Call expected instead of set");} |
|
|
421 |
|
| SET funccall newline{ $$.ty = gAny; yyerrorline(semanticerror, lineno - 1, "Call expected instead of set");} |
422 |
422 |
| lvardecl { |
| lvardecl { |
423 |
423 |
$$.ty = gAny; |
$$.ty = gAny; |
424 |
|
yyerrorex(0, "Local declaration after first statement"); |
|
|
424 |
|
yyerrorex(semanticerror, "Local declaration after first statement"); |
425 |
425 |
} |
} |
426 |
426 |
| error {$$.ty = gAny; } |
| error {$$.ty = gAny; } |
427 |
427 |
; |
; |
|
... |
... |
vartypedecl: type rid { |
477 |
477 |
} |
} |
478 |
478 |
| CONSTANT type rid { |
| CONSTANT type rid { |
479 |
479 |
if (infunction) { |
if (infunction) { |
480 |
|
yyerrorex(3, "Local constants are not allowed"); |
|
|
480 |
|
yyerrorex(semanticerror, "Local constants are not allowed"); |
481 |
481 |
} |
} |
482 |
482 |
struct typeandname *tan = newtypeandname($2.ty, $3.str); |
struct typeandname *tan = newtypeandname($2.ty, $3.str); |
483 |
483 |
tan->isconst = 1; |
tan->isconst = 1; |
|
... |
... |
vartypedecl: type rid { |
492 |
492 |
|
|
493 |
493 |
// using "type" as variable name |
// using "type" as variable name |
494 |
494 |
| type TYPE { |
| type TYPE { |
495 |
|
yyerrorex(3, "Invalid variable name \"type\""); |
|
|
495 |
|
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
496 |
496 |
struct typeandname *tan = newtypeandname($1.ty, "type"); |
struct typeandname *tan = newtypeandname($1.ty, "type"); |
497 |
497 |
$$ = checkvardecl(tan); |
$$ = checkvardecl(tan); |
498 |
498 |
} |
} |
499 |
499 |
|
|
500 |
500 |
| CONSTANT type TYPE { |
| CONSTANT type TYPE { |
501 |
501 |
if (infunction) { |
if (infunction) { |
502 |
|
yyerrorex(3, "Local constants are not allowed"); |
|
|
502 |
|
yyerrorex(semanticerror, "Local constants are not allowed"); |
503 |
503 |
} |
} |
504 |
|
yyerrorex(3, "Invalid variable name \"type\""); |
|
|
504 |
|
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
505 |
505 |
struct typeandname *tan = newtypeandname($2.ty, "type"); |
struct typeandname *tan = newtypeandname($2.ty, "type"); |
506 |
506 |
tan->isconst = 1; |
tan->isconst = 1; |
507 |
507 |
$$ = checkvardecl(tan); |
$$ = checkvardecl(tan); |
508 |
508 |
} |
} |
509 |
509 |
| type ARRAY TYPE { |
| type ARRAY TYPE { |
510 |
|
yyerrorex(3, "Invalid variable name \"type\""); |
|
|
510 |
|
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
511 |
511 |
struct typeandname *tan = newtypeandname($1.ty, "type"); |
struct typeandname *tan = newtypeandname($1.ty, "type"); |
512 |
512 |
tan->isarray = 1; |
tan->isarray = 1; |
513 |
513 |
$$ = checkarraydecl(tan); |
$$ = checkarraydecl(tan); |
|
... |
... |
endlocalsmarker: /* empty */ { fCurrent = 0; } |
524 |
524 |
; |
; |
525 |
525 |
|
|
526 |
526 |
lvardecl: LOCAL vardecl { } |
lvardecl: LOCAL vardecl { } |
527 |
|
| CONSTANT LOCAL vardecl { yyerrorex(3, "Local variables can not be declared constant"); } |
|
528 |
|
| typedef { yyerrorex(3,"Types can not be extended inside functions"); } |
|
|
527 |
|
| CONSTANT LOCAL vardecl { yyerrorex(syntaxerror, "Local variables can not be declared constant"); } |
|
528 |
|
| typedef { yyerrorex(syntaxerror,"Types can not be extended inside functions"); } |
529 |
529 |
; |
; |
530 |
530 |
|
|
531 |
531 |
vardecl: vartypedecl newline { |
vardecl: vartypedecl newline { |
532 |
532 |
const struct typeandname *tan = getVariable($1.str); |
const struct typeandname *tan = getVariable($1.str); |
533 |
533 |
if (tan->isconst) { |
if (tan->isconst) { |
534 |
|
yyerrorline(3, lineno - 1, "Constants must be initialized"); |
|
|
534 |
|
yyerrorline(syntaxerror, lineno - 1, "Constants must be initialized"); |
535 |
535 |
} |
} |
536 |
536 |
if(infunction && flagenabled(flag_shadowing) ){ |
if(infunction && flagenabled(flag_shadowing) ){ |
537 |
537 |
if( ht_lookup(&globals, tan->name)){ |
if( ht_lookup(&globals, tan->name)){ |
538 |
538 |
char buf[1024]; |
char buf[1024]; |
539 |
539 |
snprintf(buf, 1024, "Local variable %s shadows global variable", tan->name); |
snprintf(buf, 1024, "Local variable %s shadows global variable", tan->name); |
540 |
|
yyerrorline(3, lineno -1, buf); |
|
|
540 |
|
yyerrorline(semanticerror, lineno -1, buf); |
541 |
541 |
} else if( ht_lookup(¶ms, tan->name)){ |
} else if( ht_lookup(¶ms, tan->name)){ |
542 |
542 |
char buf[1024]; |
char buf[1024]; |
543 |
543 |
snprintf(buf, 1024, "Local variable %s shadows parameter", tan->name); |
snprintf(buf, 1024, "Local variable %s shadows parameter", tan->name); |
544 |
|
yyerrorline(3, lineno -1, buf); |
|
|
544 |
|
yyerrorline(semanticerror, lineno -1, buf); |
545 |
545 |
} |
} |
546 |
546 |
} |
} |
547 |
547 |
$$.ty = gNothing; |
$$.ty = gNothing; |
|
... |
... |
vardecl: vartypedecl newline { |
549 |
549 |
| vartypedecl EQUALS expr newline { |
| vartypedecl EQUALS expr newline { |
550 |
550 |
const struct typeandname *tan = getVariable($1.str); |
const struct typeandname *tan = getVariable($1.str); |
551 |
551 |
if (tan->isarray) { |
if (tan->isarray) { |
552 |
|
yyerrorex(3, "Arrays cannot be directly initialized"); |
|
|
552 |
|
yyerrorex(syntaxerror, "Arrays cannot be directly initialized"); |
553 |
553 |
} |
} |
554 |
554 |
if(infunction && !ht_lookup(&initialized, tan->name)){ |
if(infunction && !ht_lookup(&initialized, tan->name)){ |
555 |
555 |
put(&initialized, tan->name, (void*)1); |
put(&initialized, tan->name, (void*)1); |
|
... |
... |
vardecl: vartypedecl newline { |
558 |
558 |
if( ht_lookup(&globals, tan->name)){ |
if( ht_lookup(&globals, tan->name)){ |
559 |
559 |
char buf[1024]; |
char buf[1024]; |
560 |
560 |
snprintf(buf, 1024, "Local variable %s shadows global variable", tan->name); |
snprintf(buf, 1024, "Local variable %s shadows global variable", tan->name); |
561 |
|
yyerror(buf); |
|
|
561 |
|
yyerrorex(semanticerror, buf); |
562 |
562 |
} else if( ht_lookup(¶ms, tan->name)){ |
} else if( ht_lookup(¶ms, tan->name)){ |
563 |
563 |
char buf[1024]; |
char buf[1024]; |
564 |
564 |
snprintf(buf, 1024, "Local variable %s shadows parameter", tan->name); |
snprintf(buf, 1024, "Local variable %s shadows parameter", tan->name); |
565 |
|
yyerror(buf); |
|
|
565 |
|
yyerrorex(semanticerror, buf); |
566 |
566 |
} |
} |
567 |
567 |
} |
} |
568 |
568 |
canconvert($3.ty, tan->ty, -1); |
canconvert($3.ty, tan->ty, -1); |
|
... |
... |
typedef: TYPE rid EXTENDS type { |
575 |
575 |
if (ht_lookup(&types, $2.str)) { |
if (ht_lookup(&types, $2.str)) { |
576 |
576 |
char buf[1024]; |
char buf[1024]; |
577 |
577 |
snprintf(buf, 1024, "Multiply defined type %s", $2.str); |
snprintf(buf, 1024, "Multiply defined type %s", $2.str); |
578 |
|
yyerrorex(3, buf); |
|
|
578 |
|
yyerrorex(semanticerror, buf); |
579 |
579 |
} else if (ht_lookup(&functions, $2.str)) { |
} else if (ht_lookup(&functions, $2.str)) { |
580 |
580 |
char buf[1024]; |
char buf[1024]; |
581 |
581 |
snprintf(buf, 1024, "%s already defined as function", $2.str); |
snprintf(buf, 1024, "%s already defined as function", $2.str); |
582 |
|
yyerrorex(3, buf); |
|
|
582 |
|
yyerrorex(semanticerror, buf); |
583 |
583 |
} |
} |
584 |
584 |
else |
else |
585 |
585 |
put(&types, $2.str, newtypenode($2.str, $4.ty)); |
put(&types, $2.str, newtypenode($2.str, $4.ty)); |
|
... |
... |
type: primtype { $$.ty = $1.ty; } |
595 |
595 |
char buf[1024]; |
char buf[1024]; |
596 |
596 |
snprintf(buf, 1024, "Undefined type %s", $1.str); |
snprintf(buf, 1024, "Undefined type %s", $1.str); |
597 |
597 |
getsuggestions($1.str, buf, 1024, 1, &types); |
getsuggestions($1.str, buf, 1024, 1, &types); |
598 |
|
yyerrorex(3, buf); |
|
|
598 |
|
yyerrorex(semanticerror, buf); |
599 |
599 |
$$.ty = gAny; |
$$.ty = gAny; |
600 |
600 |
} |
} |
601 |
601 |
else |
else |
File misc.c changed (mode: 100644) (index 0b7c060..dd7921c) |
... |
... |
struct funcdecl *fFilter, *fCondition; |
53 |
53 |
|
|
54 |
54 |
struct hashtable available_flags; |
struct hashtable available_flags; |
55 |
55 |
|
|
56 |
|
void yyerrorline (int errorlevel, int line, const char *s) |
|
|
56 |
|
void yyerrorline (enum errortype type, int line, const char *s) |
57 |
57 |
{ |
{ |
58 |
|
//if (showerrorlevel[errorlevel]) { |
|
|
58 |
|
if(flagenabled(flag_syntaxerror) && type == syntaxerror){ |
|
59 |
|
ignorederrors++; |
|
60 |
|
return; |
|
61 |
|
} |
|
62 |
|
|
|
63 |
|
if(flagenabled(flag_semanticerror) && type == semanticerror){ |
|
64 |
|
ignorederrors++; |
|
65 |
|
return; |
|
66 |
|
} |
|
67 |
|
|
59 |
68 |
haderrors++; |
haderrors++; |
60 |
69 |
printf ("%s:%d: %s\n", curfile, line, s); |
printf ("%s:%d: %s\n", curfile, line, s); |
61 |
|
//} |
|
62 |
|
//else |
|
63 |
|
// ignorederrors++; |
|
64 |
70 |
} |
} |
65 |
71 |
|
|
66 |
|
void yyerrorex (int errorlevel, const char *s) |
|
|
72 |
|
void yyerrorex (enum errortype type, const char *s) |
67 |
73 |
{ |
{ |
68 |
|
//if (showerrorlevel[errorlevel]) { |
|
69 |
|
haderrors++; |
|
70 |
|
printf ("%s:%d: %s\n", curfile, lineno, s); |
|
71 |
|
//} |
|
72 |
|
//else |
|
73 |
|
// ignorederrors++; |
|
|
74 |
|
yyerrorline(type, lineno, s); |
74 |
75 |
} |
} |
75 |
76 |
|
|
76 |
77 |
void yyerror (const char *s) /* Called by yyparse on error */ |
void yyerror (const char *s) /* Called by yyparse on error */ |
77 |
78 |
{ |
{ |
78 |
|
yyerrorex(0, s); |
|
|
79 |
|
yyerrorex(syntaxerror, s); |
79 |
80 |
} |
} |
80 |
81 |
|
|
81 |
82 |
void put(struct hashtable *h, const char *name, void *val){ |
void put(struct hashtable *h, const char *name, void *val){ |
82 |
83 |
if( !ht_put(h, name, val) ){ |
if( !ht_put(h, name, val) ){ |
83 |
84 |
char ebuf[1024]; |
char ebuf[1024]; |
84 |
85 |
snprintf(ebuf, 1024, "Symbol %s multiply defined", name); |
snprintf(ebuf, 1024, "Symbol %s multiply defined", name); |
85 |
|
yyerrorline(3, islinebreak ? lineno - 1 : lineno, ebuf); |
|
|
86 |
|
yyerrorline(semanticerror, islinebreak ? lineno - 1 : lineno, ebuf); |
86 |
87 |
} |
} |
87 |
88 |
} |
} |
88 |
89 |
|
|
|
... |
... |
const struct typeandname *getVariable(const char *varname) |
239 |
240 |
|
|
240 |
241 |
snprintf(ebuf, 1024, "Undeclared variable %s", varname); |
snprintf(ebuf, 1024, "Undeclared variable %s", varname); |
241 |
242 |
getsuggestions(varname, ebuf, 1024, 3, &locals, ¶ms, &globals); |
getsuggestions(varname, ebuf, 1024, 3, &locals, ¶ms, &globals); |
242 |
|
yyerrorline(2, islinebreak ? lineno - 1 : lineno, ebuf); |
|
|
243 |
|
yyerrorline(semanticerror, islinebreak ? lineno - 1 : lineno, ebuf); |
243 |
244 |
|
|
244 |
245 |
// Store it as unidentified variable |
// Store it as unidentified variable |
245 |
246 |
const struct typeandname *newtan = newtypeandname(gAny, varname); |
const struct typeandname *newtan = newtypeandname(gAny, varname); |
|
... |
... |
void validateGlobalAssignment(const char *varname) |
257 |
258 |
result = ht_lookup(&globals, varname); |
result = ht_lookup(&globals, varname); |
258 |
259 |
if (result) { |
if (result) { |
259 |
260 |
snprintf(ebuf, 1024, "Assignment to global variable %s in constant function", varname); |
snprintf(ebuf, 1024, "Assignment to global variable %s in constant function", varname); |
260 |
|
yyerrorline(2, lineno - 1, ebuf); |
|
|
261 |
|
yyerrorline(semanticerror, lineno - 1, ebuf); |
261 |
262 |
} |
} |
262 |
263 |
} |
} |
263 |
264 |
|
|
|
... |
... |
void checkParameters(const struct paramlist *func, const struct paramlist *inp, |
270 |
271 |
if (fi == NULL && pi == NULL) |
if (fi == NULL && pi == NULL) |
271 |
272 |
return; |
return; |
272 |
273 |
if (fi == NULL && pi != NULL) { |
if (fi == NULL && pi != NULL) { |
273 |
|
yyerrorex(3, "Too many arguments passed to function"); |
|
|
274 |
|
yyerrorex(semanticerror, "Too many arguments passed to function"); |
274 |
275 |
return; |
return; |
275 |
276 |
} |
} |
276 |
277 |
if (fi != NULL && pi == NULL) { |
if (fi != NULL && pi == NULL) { |
277 |
|
yyerrorex(3, "Not enough arguments passed to function"); |
|
|
278 |
|
yyerrorex(semanticerror, "Not enough arguments passed to function"); |
278 |
279 |
return; |
return; |
279 |
280 |
} |
} |
280 |
281 |
canconvert(pi->ty, fi->ty, 0); |
canconvert(pi->ty, fi->ty, 0); |
|
... |
... |
const struct typenode *binop(const struct typenode *a, const struct typenode *b) |
297 |
298 |
if (typeeq(b, gAny)) |
if (typeeq(b, gAny)) |
298 |
299 |
return a; |
return a; |
299 |
300 |
if ((!typeeq(a, gInteger) && !typeeq(a, gReal)) || (!typeeq(b, gInteger) && !typeeq(b, gReal))) { |
if ((!typeeq(a, gInteger) && !typeeq(a, gReal)) || (!typeeq(b, gInteger) && !typeeq(b, gReal))) { |
300 |
|
yyerrorline(3, islinebreak ? lineno - 1 : lineno, "Bad types for binary operator"); |
|
|
301 |
|
yyerrorline(semanticerror, islinebreak ? lineno - 1 : lineno, "Bad types for binary operator"); |
301 |
302 |
} |
} |
302 |
303 |
return gReal; |
return gReal; |
303 |
304 |
} |
} |
|
... |
... |
void canconvert(const struct typenode *ufrom, const struct typenode *uto, const |
355 |
356 |
return; |
return; |
356 |
357 |
|
|
357 |
358 |
snprintf(ebuf, 1024, "Cannot convert %s to %s", ufrom->typename, uto->typename); |
snprintf(ebuf, 1024, "Cannot convert %s to %s", ufrom->typename, uto->typename); |
358 |
|
yyerrorline(3, lineno + linemod, ebuf); |
|
|
359 |
|
yyerrorline(semanticerror, lineno + linemod, ebuf); |
359 |
360 |
return; |
return; |
360 |
361 |
} |
} |
361 |
362 |
|
|
|
... |
... |
void canconvertreturn(const struct typenode *ufrom, const struct typenode *uto, |
385 |
386 |
if ((typeeq(to, gReal)) && (typeeq(from, gInteger))) { |
if ((typeeq(to, gReal)) && (typeeq(from, gInteger))) { |
386 |
387 |
// can't return integer when it expects a real (added 9.5.2005) |
// can't return integer when it expects a real (added 9.5.2005) |
387 |
388 |
snprintf(ebuf, 1024, "Cannot convert returned value from %s to %s", getTypePtr(from)->typename, getTypePtr(to)->typename); |
snprintf(ebuf, 1024, "Cannot convert returned value from %s to %s", getTypePtr(from)->typename, getTypePtr(to)->typename); |
388 |
|
yyerrorline(1, lineno + linemod, ebuf); |
|
|
389 |
|
yyerrorline(semanticerror, lineno + linemod, ebuf); |
389 |
390 |
return; |
return; |
390 |
391 |
} |
} |
391 |
392 |
|
|
|
... |
... |
void canconvertreturn(const struct typenode *ufrom, const struct typenode *uto, |
397 |
398 |
} |
} |
398 |
399 |
|
|
399 |
400 |
snprintf(ebuf, 1024, "Cannot convert returned value from %s to %s", getTypePtr(ufrom)->typename, getTypePtr(uto)->typename); |
snprintf(ebuf, 1024, "Cannot convert returned value from %s to %s", getTypePtr(ufrom)->typename, getTypePtr(uto)->typename); |
400 |
|
yyerrorline(1, lineno + linemod, ebuf); |
|
|
401 |
|
yyerrorline(semanticerror, lineno + linemod, ebuf); |
401 |
402 |
return; |
return; |
402 |
403 |
} |
} |
403 |
404 |
|
|
|
... |
... |
void isnumeric(const struct typenode *ty) |
405 |
406 |
{ |
{ |
406 |
407 |
ty = getPrimitiveAncestor(ty); |
ty = getPrimitiveAncestor(ty); |
407 |
408 |
if (!(ty == gInteger || ty == gReal || ty == gAny)) |
if (!(ty == gInteger || ty == gReal || ty == gAny)) |
408 |
|
yyerrorline(3, islinebreak ? lineno - 1 : lineno, "Cannot be converted to numeric type"); |
|
|
409 |
|
yyerrorline(semanticerror, islinebreak ? lineno - 1 : lineno, "Cannot be converted to numeric type"); |
409 |
410 |
} |
} |
410 |
411 |
|
|
411 |
412 |
void checkcomparisonsimple(const struct typenode *a) |
void checkcomparisonsimple(const struct typenode *a) |
|
... |
... |
void checkcomparisonsimple(const struct typenode *a) |
413 |
414 |
const struct typenode *pa; |
const struct typenode *pa; |
414 |
415 |
pa = getPrimitiveAncestor(a); |
pa = getPrimitiveAncestor(a); |
415 |
416 |
if (typeeq(pa, gString) || typeeq(pa, gHandle) || typeeq(pa, gCode) || typeeq(pa, gBoolean)) { |
if (typeeq(pa, gString) || typeeq(pa, gHandle) || typeeq(pa, gCode) || typeeq(pa, gBoolean)) { |
416 |
|
yyerrorex(3, "Comparing the order/size of 2 variables only works on reals and integers"); |
|
|
417 |
|
yyerrorex(semanticerror, "Comparing the order/size of 2 variables only works on reals and integers"); |
417 |
418 |
return; |
return; |
418 |
419 |
} |
} |
419 |
420 |
if (typeeq(pa, gNull)) |
if (typeeq(pa, gNull)) |
420 |
|
yyerrorex(3, "Comparing null is not allowed"); |
|
|
421 |
|
yyerrorex(semanticerror, "Comparing null is not allowed"); |
421 |
422 |
} |
} |
422 |
423 |
|
|
423 |
424 |
void checkcomparison(const struct typenode *a, const struct typenode *b) |
void checkcomparison(const struct typenode *a, const struct typenode *b) |
|
... |
... |
void checkcomparison(const struct typenode *a, const struct typenode *b) |
426 |
427 |
pa = getPrimitiveAncestor(a); |
pa = getPrimitiveAncestor(a); |
427 |
428 |
pb = getPrimitiveAncestor(b); |
pb = getPrimitiveAncestor(b); |
428 |
429 |
if (typeeq(pa, gString) || typeeq(pa, gHandle) || typeeq(pa, gCode) || typeeq(pa, gBoolean) || typeeq(pb, gString) || typeeq(pb, gCode) || typeeq(pb, gHandle) || typeeq(pb, gBoolean)) { |
if (typeeq(pa, gString) || typeeq(pa, gHandle) || typeeq(pa, gCode) || typeeq(pa, gBoolean) || typeeq(pb, gString) || typeeq(pb, gCode) || typeeq(pb, gHandle) || typeeq(pb, gBoolean)) { |
429 |
|
yyerrorex(3, "Comparing the order/size of 2 variables only works on reals and integers"); |
|
|
430 |
|
yyerrorex(semanticerror, "Comparing the order/size of 2 variables only works on reals and integers"); |
430 |
431 |
return; |
return; |
431 |
432 |
} |
} |
432 |
433 |
if (typeeq(pa, gNull) && typeeq(pb, gNull)) |
if (typeeq(pa, gNull) && typeeq(pb, gNull)) |
433 |
|
yyerrorex(3, "Comparing null is not allowed"); |
|
|
434 |
|
yyerrorex(semanticerror, "Comparing null is not allowed"); |
434 |
435 |
} |
} |
435 |
436 |
|
|
436 |
437 |
void checkeqtest(const struct typenode *a, const struct typenode *b) |
void checkeqtest(const struct typenode *a, const struct typenode *b) |
|
... |
... |
void checkeqtest(const struct typenode *a, const struct typenode *b) |
443 |
444 |
if (typeeq(pa, gNull) || typeeq(pb, gNull)) |
if (typeeq(pa, gNull) || typeeq(pb, gNull)) |
444 |
445 |
return; |
return; |
445 |
446 |
if (!typeeq(pa, pb)) { |
if (!typeeq(pa, pb)) { |
446 |
|
yyerrorex(3, "Comparing two variables of different primitive types (except real and integer) is not allowed"); |
|
|
447 |
|
yyerrorex(semanticerror, "Comparing two variables of different primitive types (except real and integer) is not allowed"); |
447 |
448 |
return; |
return; |
448 |
449 |
} |
} |
449 |
450 |
} |
} |
|
... |
... |
union node checkfunctionheader(const char *fnname, struct paramlist *pl, const s |
489 |
490 |
if (ht_lookup(&locals, fnname) || ht_lookup(¶ms, fnname) || ht_lookup(&globals, fnname)) { |
if (ht_lookup(&locals, fnname) || ht_lookup(¶ms, fnname) || ht_lookup(&globals, fnname)) { |
490 |
491 |
char buf[1024]; |
char buf[1024]; |
491 |
492 |
snprintf(buf, 1024, "%s already defined as variable", fnname); |
snprintf(buf, 1024, "%s already defined as variable", fnname); |
492 |
|
yyerrorex(3, buf); |
|
|
493 |
|
yyerrorex(semanticerror, buf); |
493 |
494 |
} else if (ht_lookup(&types, fnname)) { |
} else if (ht_lookup(&types, fnname)) { |
494 |
495 |
char buf[1024]; |
char buf[1024]; |
495 |
496 |
snprintf(buf, 1024, "%s already defined as type", fnname); |
snprintf(buf, 1024, "%s already defined as type", fnname); |
496 |
|
yyerrorex(3, buf); |
|
|
497 |
|
yyerrorex(semanticerror, buf); |
497 |
498 |
} |
} |
498 |
499 |
|
|
499 |
500 |
curtab = &locals; |
curtab = &locals; |
|
... |
... |
union node checkfunctionheader(const char *fnname, struct paramlist *pl, const s |
515 |
516 |
if (ht_lookup(&functions, tan->name)) { |
if (ht_lookup(&functions, tan->name)) { |
516 |
517 |
char buf[1024]; |
char buf[1024]; |
517 |
518 |
snprintf(buf, 1024, "%s already defined as function", tan->name); |
snprintf(buf, 1024, "%s already defined as function", tan->name); |
518 |
|
yyerrorex(3, buf); |
|
|
519 |
|
yyerrorex(semanticerror, buf); |
519 |
520 |
} else if (ht_lookup(&types, tan->name)) { |
} else if (ht_lookup(&types, tan->name)) { |
520 |
521 |
char buf[1024]; |
char buf[1024]; |
521 |
522 |
snprintf(buf, 1024, "%s already defined as type", tan->name); |
snprintf(buf, 1024, "%s already defined as type", tan->name); |
522 |
|
yyerrorex(3, buf); |
|
|
523 |
|
yyerrorex(semanticerror, buf); |
523 |
524 |
} |
} |
524 |
525 |
|
|
525 |
526 |
if( flagenabled(flag_shadowing) ){ |
if( flagenabled(flag_shadowing) ){ |
526 |
527 |
if( ht_lookup(&globals, tan->name) ){ |
if( ht_lookup(&globals, tan->name) ){ |
527 |
528 |
char buf[1024]; |
char buf[1024]; |
528 |
529 |
snprintf(buf, 1024, "Parmeter %s shadows global variable", tan->name); |
snprintf(buf, 1024, "Parmeter %s shadows global variable", tan->name); |
529 |
|
yyerrorex(3, buf); |
|
|
530 |
|
yyerrorex(semanticerror, buf); |
530 |
531 |
} |
} |
531 |
532 |
} |
} |
532 |
533 |
|
|
|
... |
... |
union node checkfunccall(const char *fnname, struct paramlist *pl) |
546 |
547 |
char ebuf[1024]; |
char ebuf[1024]; |
547 |
548 |
snprintf(ebuf, 1024, "Undeclared function %s", fnname); |
snprintf(ebuf, 1024, "Undeclared function %s", fnname); |
548 |
549 |
getsuggestions(fnname, ebuf, 1024, 1, &functions); |
getsuggestions(fnname, ebuf, 1024, 1, &functions); |
549 |
|
yyerrorex(3, ebuf); |
|
|
550 |
|
yyerrorex(semanticerror, ebuf); |
550 |
551 |
ret.ty = gAny; |
ret.ty = gAny; |
551 |
552 |
} else { |
} else { |
552 |
553 |
if (inconstant && !(fd->isconst)) { |
if (inconstant && !(fd->isconst)) { |
553 |
554 |
char ebuf[1024]; |
char ebuf[1024]; |
554 |
555 |
snprintf(ebuf, 1024, "Call to non-constant function %s in constant function", fnname); |
snprintf(ebuf, 1024, "Call to non-constant function %s in constant function", fnname); |
555 |
|
yyerrorex(3, ebuf); |
|
|
556 |
|
yyerrorex(semanticerror, ebuf); |
556 |
557 |
} |
} |
557 |
558 |
if (fd == fCurrent && fCurrent) |
if (fd == fCurrent && fCurrent) |
558 |
|
yyerrorex(3, "Recursive function calls are not permitted in local declarations"); |
|
|
559 |
|
yyerrorex(semanticerror, "Recursive function calls are not permitted in local declarations"); |
559 |
560 |
checkParameters(fd->p, pl, fd == fFilter || fd == fCondition); |
checkParameters(fd->p, pl, fd == fFilter || fd == fCondition); |
560 |
561 |
ret.ty = fd->ret; |
ret.ty = fd->ret; |
561 |
562 |
} |
} |
|
... |
... |
static void checkvarname(struct typeandname *tan, bool isarray) |
568 |
569 |
if (ht_lookup(&functions, name)) { |
if (ht_lookup(&functions, name)) { |
569 |
570 |
char buf[1024]; |
char buf[1024]; |
570 |
571 |
snprintf(buf, 1024, "Symbol %s already defined as function", name); |
snprintf(buf, 1024, "Symbol %s already defined as function", name); |
571 |
|
yyerrorex(3, buf); |
|
|
572 |
|
yyerrorex(semanticerror, buf); |
572 |
573 |
} else if (ht_lookup(&types, name)) { |
} else if (ht_lookup(&types, name)) { |
573 |
574 |
char buf[1024]; |
char buf[1024]; |
574 |
575 |
snprintf(buf, 1024, "Symbol %s already defined as type", name); |
snprintf(buf, 1024, "Symbol %s already defined as type", name); |
575 |
|
yyerrorex(3, buf); |
|
|
576 |
|
yyerrorex(semanticerror, buf); |
576 |
577 |
} |
} |
577 |
578 |
|
|
578 |
579 |
struct typeandname *existing = ht_lookup(&locals, name); |
struct typeandname *existing = ht_lookup(&locals, name); |
|
... |
... |
static void checkvarname(struct typeandname *tan, bool isarray) |
582 |
583 |
existing = ht_lookup(¶ms, name); |
existing = ht_lookup(¶ms, name); |
583 |
584 |
if ( isarray && infunction && existing) { |
if ( isarray && infunction && existing) { |
584 |
585 |
snprintf(buf, 1024, "Symbol %s already defined as function parameter", name); |
snprintf(buf, 1024, "Symbol %s already defined as function parameter", name); |
585 |
|
yyerrorex(3, buf); |
|
|
586 |
|
yyerrorex(semanticerror, buf); |
586 |
587 |
} |
} |
587 |
588 |
if (!existing) { |
if (!existing) { |
588 |
589 |
existing = ht_lookup(&globals, name); |
existing = ht_lookup(&globals, name); |
589 |
590 |
if ( isarray && infunction && existing) { |
if ( isarray && infunction && existing) { |
590 |
591 |
snprintf(buf, 1024, "Symbol %s already defined as global variable", name); |
snprintf(buf, 1024, "Symbol %s already defined as global variable", name); |
591 |
|
yyerrorex(3, buf); |
|
|
592 |
|
yyerrorex(semanticerror, buf); |
592 |
593 |
} |
} |
593 |
594 |
} |
} |
594 |
595 |
} |
} |
|
... |
... |
union node checkarraydecl(struct typeandname *tan) |
619 |
620 |
union node ret; |
union node ret; |
620 |
621 |
|
|
621 |
622 |
if (getPrimitiveAncestor(tan->ty) == gCode) |
if (getPrimitiveAncestor(tan->ty) == gCode) |
622 |
|
yyerrorex(3, "Code arrays are not allowed"); |
|
|
623 |
|
yyerrorex(semanticerror, "Code arrays are not allowed"); |
623 |
624 |
|
|
624 |
625 |
checkvarname(tan, true); |
checkvarname(tan, true); |
625 |
626 |
|
|
File token.l changed (mode: 100644) (index dd84558..d06b8d1) |
... |
... |
TABS [\x01-\x09\x0B\x0C\x0E-\x1F] |
55 |
55 |
|
|
56 |
56 |
[']\\[btrnf\\]['] { return UNITTYPEINT; } |
[']\\[btrnf\\]['] { return UNITTYPEINT; } |
57 |
57 |
[']\\(.|[\r\n])['] { |
[']\\(.|[\r\n])['] { |
58 |
|
yyerrorline(3, lineno, "Invalid escape character sequence"); |
|
|
58 |
|
yyerrorline(syntaxerror, lineno, "Invalid escape character sequence"); |
59 |
59 |
return UNITTYPEINT; |
return UNITTYPEINT; |
60 |
60 |
} |
} |
61 |
61 |
|
|
|
... |
... |
TABS [\x01-\x09\x0B\x0C\x0E-\x1F] |
66 |
66 |
} |
} |
67 |
67 |
<RAWCODE>['] { |
<RAWCODE>['] { |
68 |
68 |
if(rawcodelen != 4 && rawcodelen != 1){ |
if(rawcodelen != 4 && rawcodelen != 1){ |
69 |
|
yyerrorline(3, rawcodestartline, "Rawcodes must consist of 1 or 4 characters"); |
|
|
69 |
|
yyerrorline(syntaxerror, rawcodestartline, "Rawcodes must consist of 1 or 4 characters"); |
70 |
70 |
} |
} |
71 |
71 |
BEGIN(INITIAL); |
BEGIN(INITIAL); |
72 |
72 |
return UNITTYPEINT; |
return UNITTYPEINT; |
|
... |
... |
TABS [\x01-\x09\x0B\x0C\x0E-\x1F] |
74 |
74 |
|
|
75 |
75 |
<RAWCODE>\\[btrnf"\\] { |
<RAWCODE>\\[btrnf"\\] { |
76 |
76 |
rawcodelen++; |
rawcodelen++; |
77 |
|
yyerrorline(3, rawcodestartline, "Escaped chars are only allowed if they are the only char in the rawcode."); |
|
|
77 |
|
yyerrorline(syntaxerror, rawcodestartline, "Escaped chars are only allowed if they are the only char in the rawcode."); |
78 |
78 |
} |
} |
79 |
79 |
<RAWCODE>\\(.|[\r\n]) { |
<RAWCODE>\\(.|[\r\n]) { |
80 |
|
yyerrorline(3, lineno, "Invalid escape character sequence"); |
|
|
80 |
|
yyerrorline(syntaxerror, lineno, "Invalid escape character sequence"); |
81 |
81 |
rawcodelen++; |
rawcodelen++; |
82 |
82 |
} |
} |
83 |
83 |
|
|
|
... |
... |
TABS [\x01-\x09\x0B\x0C\x0E-\x1F] |
88 |
88 |
["] { BEGIN(INSTRING); } |
["] { BEGIN(INSTRING); } |
89 |
89 |
<INSTRING>["] { BEGIN(INITIAL); return STRINGLIT; } |
<INSTRING>["] { BEGIN(INITIAL); return STRINGLIT; } |
90 |
90 |
<INSTRING>\\[btrnf"\\] { } |
<INSTRING>\\[btrnf"\\] { } |
91 |
|
<INSTRING>\\(.|[\r\n]) { yyerrorline(3, lineno, "Invalid escape character sequence"); } |
|
|
91 |
|
<INSTRING>\\(.|[\r\n]) { yyerrorline(syntaxerror, lineno, "Invalid escape character sequence"); } |
92 |
92 |
|
|
93 |
93 |
<INSTRING>\r?\n { lineno++; isconstant = 0; } |
<INSTRING>\r?\n { lineno++; isconstant = 0; } |
94 |
94 |
<INSTRING>\r { lineno++; isconstant = 0; } |
<INSTRING>\r { lineno++; isconstant = 0; } |
|
... |
... |
TABS [\x01-\x09\x0B\x0C\x0E-\x1F] |
98 |
98 |
{NEWLINE} lineno++; islinebreak=1; isconstant=0; return NEWLINE; |
{NEWLINE} lineno++; islinebreak=1; isconstant=0; return NEWLINE; |
99 |
99 |
|
|
100 |
100 |
{DIGIT}*"."{DIGIT}* return REALLIT; |
{DIGIT}*"."{DIGIT}* return REALLIT; |
101 |
|
"0"({OCTDIGIT}*("8"|"9"){OCTDIGIT}*)+ yyerrorline(3, lineno, "Invalid digit in octal integer notation"); return INTLIT; |
|
|
101 |
|
"0"({OCTDIGIT}*("8"|"9"){OCTDIGIT}*)+ yyerrorline(syntaxerror, lineno, "Invalid digit in octal integer notation"); return INTLIT; |
102 |
102 |
({DIGIT}+)|(("0x"|"$"){HEXDIGIT}+) return INTLIT; |
({DIGIT}+)|(("0x"|"$"){HEXDIGIT}+) return INTLIT; |
103 |
103 |
|
|
104 |
|
"if" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before if"); islinebreak=0; return IF; |
|
|
104 |
|
"if" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before if"); islinebreak=0; return IF; |
105 |
105 |
"not" return NOT; |
"not" return NOT; |
106 |
106 |
"then" return THEN; |
"then" return THEN; |
107 |
|
"type" if (!islinebreak) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before type declaration"); yyerrorline(3, lineno, ebuf); } islinebreak=0; return TYPE; |
|
|
107 |
|
"type" if (!islinebreak) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before type declaration"); yyerrorline(syntaxerror, lineno, ebuf); } islinebreak=0; return TYPE; |
108 |
108 |
"extends" return EXTENDS; |
"extends" return EXTENDS; |
109 |
109 |
"handle" islinebreak=0; return HANDLE; |
"handle" islinebreak=0; return HANDLE; |
110 |
|
"globals" if (!islinebreak) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before globals block"); yyerrorline(3, lineno, ebuf); } islinebreak=0; inblock=1; return GLOBALS; |
|
|
110 |
|
"globals" if (!islinebreak) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before globals block"); yyerrorline(syntaxerror, lineno, ebuf); } islinebreak=0; inblock=1; return GLOBALS; |
111 |
111 |
"endglobals" islinebreak=0; inblock=0; return ENDGLOBALS; |
"endglobals" islinebreak=0; inblock=0; return ENDGLOBALS; |
112 |
112 |
"constant" isconstant = islinebreak; islinebreak=0; return CONSTANT; |
"constant" isconstant = islinebreak; islinebreak=0; return CONSTANT; |
113 |
|
"native" if (!islinebreak && !isconstant) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before native declaration"); yyerrorline(3, lineno, ebuf); } islinebreak=0; return NATIVE; |
|
|
113 |
|
"native" if (!islinebreak && !isconstant) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before native declaration"); yyerrorline(syntaxerror, lineno, ebuf); } islinebreak=0; return NATIVE; |
114 |
114 |
"takes" return TAKES; |
"takes" return TAKES; |
115 |
115 |
"returns" return RETURNS; |
"returns" return RETURNS; |
116 |
|
"function" if (!islinebreak && !isconstant && !inblock) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before function declaration"); yyerrorline(3, lineno, ebuf); } islinebreak=0; return FUNCTION; |
|
|
116 |
|
"function" if (!islinebreak && !isconstant && !inblock) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before function declaration"); yyerrorline(syntaxerror, lineno, ebuf); } islinebreak=0; return FUNCTION; |
117 |
117 |
"endfunction" islinebreak=0; return ENDFUNCTION; |
"endfunction" islinebreak=0; return ENDFUNCTION; |
118 |
|
"local" if (!islinebreak) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before local declaration"); yyerrorline(3, lineno, ebuf); } islinebreak=0; return LOCAL; |
|
|
118 |
|
"local" if (!islinebreak) { char ebuf[1024]; sprintf(ebuf, "Missing linebreak before local declaration"); yyerrorline(syntaxerror, lineno, ebuf); } islinebreak=0; return LOCAL; |
119 |
119 |
"array" return ARRAY; |
"array" return ARRAY; |
120 |
|
"set" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before assignment"); islinebreak=0; return SET; |
|
121 |
|
"call" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before function call"); islinebreak=0; return CALL; |
|
122 |
|
"else" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before else"); islinebreak=0; return ELSE; |
|
123 |
|
"elseif" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before elseif"); islinebreak=0; return ELSEIF; |
|
124 |
|
"endif" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before endif"); islinebreak=0; return ENDIF; |
|
125 |
|
"loop" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before loop"); islinebreak=0; return LOOP; |
|
126 |
|
"exitwhen" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before exitwhen"); islinebreak=0; return EXITWHEN; |
|
127 |
|
"return" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before return"); islinebreak=0; return RETURN; |
|
|
120 |
|
"set" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before assignment"); islinebreak=0; return SET; |
|
121 |
|
"call" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before function call"); islinebreak=0; return CALL; |
|
122 |
|
"else" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before else"); islinebreak=0; return ELSE; |
|
123 |
|
"elseif" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before elseif"); islinebreak=0; return ELSEIF; |
|
124 |
|
"endif" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before endif"); islinebreak=0; return ENDIF; |
|
125 |
|
"loop" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before loop"); islinebreak=0; return LOOP; |
|
126 |
|
"exitwhen" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before exitwhen"); islinebreak=0; return EXITWHEN; |
|
127 |
|
"return" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before return"); islinebreak=0; return RETURN; |
128 |
128 |
"debug" return DEBUG; |
"debug" return DEBUG; |
129 |
|
"endloop" if (!islinebreak) yyerrorline(3, lineno, "Missing linebreak before endloop"); islinebreak=0; return ENDLOOP; |
|
|
129 |
|
"endloop" if (!islinebreak) yyerrorline(syntaxerror, lineno, "Missing linebreak before endloop"); islinebreak=0; return ENDLOOP; |
130 |
130 |
"null" return TNULL; |
"null" return TNULL; |
131 |
131 |
"true" return TTRUE; |
"true" return TTRUE; |
132 |
132 |
"false" return TFALSE; |
"false" return TFALSE; |
|
... |
... |
TABS [\x01-\x09\x0B\x0C\x0E-\x1F] |
164 |
164 |
\xEF\xBB\xBF /* utf8 bom */ |
\xEF\xBB\xBF /* utf8 bom */ |
165 |
165 |
[\x01-\x09\x0B\x0C\x0E-\x1F]+ /* eat up tabs */ |
[\x01-\x09\x0B\x0C\x0E-\x1F]+ /* eat up tabs */ |
166 |
166 |
|
|
167 |
|
. { char ebuf[1024]; sprintf(ebuf, "Unrecognized character %s (ASCII %u)", yytext, (unsigned char)yytext[0] ); yyerrorline(3, lineno, ebuf); } |
|
|
167 |
|
. { char ebuf[1024]; sprintf(ebuf, "Unrecognized character %s (ASCII %u)", yytext, (unsigned char)yytext[0] ); yyerrorline(syntaxerror, lineno, ebuf); } |
168 |
168 |
|
|
169 |
169 |
%% |
%% |
170 |
170 |
|
|