File grammar.y changed (mode: 100644) (index 254a1ca..93edad6) |
... |
... |
expr: intexpr { $$.ty = gInteger; } |
202 |
202 |
} |
} |
203 |
203 |
| rid { |
| rid { |
204 |
204 |
const struct typeandname *tan = getVariable($1.str); |
const struct typeandname *tan = getVariable($1.str); |
205 |
|
if (tan->lineno == lineno && tan->fn == fno) { |
|
206 |
|
char ebuf[1024]; |
|
207 |
|
snprintf(ebuf, 1024, "Use of variable %s before its declaration", $1.str); |
|
208 |
|
yyerrorex(semanticerror, ebuf); |
|
209 |
|
} else if (islinebreak && tan->lineno == lineno - 1 && tan->fn == fno) { |
|
210 |
|
char ebuf[1024]; |
|
211 |
|
snprintf(ebuf, 1024, "Use of variable %s before its declaration", $1.str); |
|
212 |
|
yyerrorline(semanticerror, lineno - 1, ebuf); |
|
213 |
|
} else if (tan->isarray) { |
|
|
205 |
|
if (tan->isarray) { |
214 |
206 |
char ebuf[1024]; |
char ebuf[1024]; |
215 |
207 |
snprintf(ebuf, 1024, "Index missing for array variable %s", $1.str); |
snprintf(ebuf, 1024, "Index missing for array variable %s", $1.str); |
216 |
208 |
yyerrorex(semanticerror, ebuf); |
yyerrorex(semanticerror, ebuf); |
217 |
209 |
} |
} |
|
210 |
|
checkwrongshadowing(tan, 0); |
218 |
211 |
if(infunction && ht_lookup(curtab, $1.str) && !ht_lookup(&initialized, $1.str) ){ |
if(infunction && ht_lookup(curtab, $1.str) && !ht_lookup(&initialized, $1.str) ){ |
219 |
212 |
char ebuf[1024]; |
char ebuf[1024]; |
220 |
213 |
snprintf(ebuf, 1024, "Variable %s is uninitialized", $1.str); |
snprintf(ebuf, 1024, "Variable %s is uninitialized", $1.str); |
221 |
|
//yyerrorex(3, ebuf); |
|
222 |
214 |
yyerrorline(semanticerror, lineno - 1, ebuf); |
yyerrorline(semanticerror, lineno - 1, ebuf); |
223 |
215 |
} |
} |
224 |
216 |
$$.ty = tan->ty; |
$$.ty = tan->ty; |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
376 |
368 |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
377 |
369 |
} |
} |
378 |
370 |
| SET rid EQUALS expr newline { |
| SET rid EQUALS expr newline { |
379 |
|
if (getVariable($2.str)->isarray) { |
|
|
371 |
|
const struct typeandname *tan = getVariable($2.str); |
|
372 |
|
checkwrongshadowing(tan, -1); |
|
373 |
|
if (tan->isarray) { |
380 |
374 |
char ebuf[1024]; |
char ebuf[1024]; |
381 |
375 |
snprintf(ebuf, 1024, "Index missing for array variable %s", $2.str); |
snprintf(ebuf, 1024, "Index missing for array variable %s", $2.str); |
382 |
376 |
yyerrorline(semanticerror, lineno - 1, ebuf); |
yyerrorline(semanticerror, lineno - 1, ebuf); |
383 |
377 |
} |
} |
384 |
|
canconvert($4.ty, getVariable($2.str)->ty, -1); |
|
|
378 |
|
canconvert($4.ty, tan->ty, -1); |
385 |
379 |
$$.ty = gAny; |
$$.ty = gAny; |
386 |
|
if (getVariable($2.str)->isconst) { |
|
|
380 |
|
if (tan->isconst) { |
387 |
381 |
char ebuf[1024]; |
char ebuf[1024]; |
388 |
382 |
snprintf(ebuf, 1024, "Cannot assign to constant %s", $2.str); |
snprintf(ebuf, 1024, "Cannot assign to constant %s", $2.str); |
389 |
383 |
yyerrorline(semanticerror, lineno - 1, ebuf); |
yyerrorline(semanticerror, lineno - 1, ebuf); |
|
... |
... |
rid: ID |
502 |
496 |
|
|
503 |
497 |
vartypedecl: type rid { |
vartypedecl: type rid { |
504 |
498 |
struct typeandname *tan = newtypeandname($1.ty, $2.str); |
struct typeandname *tan = newtypeandname($1.ty, $2.str); |
505 |
|
$$ = checkvardecl(tan); |
|
|
499 |
|
tan->lineno = lineno; |
|
500 |
|
$$ = checkvartypedecl(tan); |
506 |
501 |
} |
} |
507 |
502 |
| CONSTANT type rid { |
| CONSTANT type rid { |
508 |
503 |
if (infunction) { |
if (infunction) { |
|
... |
... |
vartypedecl: type rid { |
510 |
505 |
} |
} |
511 |
506 |
struct typeandname *tan = newtypeandname($2.ty, $3.str); |
struct typeandname *tan = newtypeandname($2.ty, $3.str); |
512 |
507 |
tan->isconst = 1; |
tan->isconst = 1; |
513 |
|
$$ = checkvardecl(tan); |
|
|
508 |
|
$$ = checkvartypedecl(tan); |
514 |
509 |
} |
} |
515 |
510 |
| type ARRAY rid { |
| type ARRAY rid { |
516 |
511 |
struct typeandname *tan = newtypeandname($1.ty, $3.str); |
struct typeandname *tan = newtypeandname($1.ty, $3.str); |
|
... |
... |
vartypedecl: type rid { |
523 |
518 |
| type TYPE { |
| type TYPE { |
524 |
519 |
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
525 |
520 |
struct typeandname *tan = newtypeandname($1.ty, "type"); |
struct typeandname *tan = newtypeandname($1.ty, "type"); |
526 |
|
$$ = checkvardecl(tan); |
|
|
521 |
|
$$ = checkvartypedecl(tan); |
527 |
522 |
} |
} |
528 |
523 |
|
|
529 |
524 |
| CONSTANT type TYPE { |
| CONSTANT type TYPE { |
|
... |
... |
vartypedecl: type rid { |
533 |
528 |
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
534 |
529 |
struct typeandname *tan = newtypeandname($2.ty, "type"); |
struct typeandname *tan = newtypeandname($2.ty, "type"); |
535 |
530 |
tan->isconst = 1; |
tan->isconst = 1; |
536 |
|
$$ = checkvardecl(tan); |
|
|
531 |
|
$$ = checkvartypedecl(tan); |
537 |
532 |
} |
} |
538 |
533 |
| type ARRAY TYPE { |
| type ARRAY TYPE { |
539 |
534 |
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
yyerrorex(syntaxerror, "Invalid variable name \"type\""); |
|
... |
... |
lvardecl: LOCAL vardecl { } |
560 |
555 |
vardecl: vartypedecl newline { |
vardecl: vartypedecl newline { |
561 |
556 |
const struct typeandname *tan = getVariable($1.str); |
const struct typeandname *tan = getVariable($1.str); |
562 |
557 |
if (tan->isconst) { |
if (tan->isconst) { |
563 |
|
yyerrorline(syntaxerror, lineno - 1, "Constants must be initialized"); |
|
564 |
|
} |
|
565 |
|
if(infunction && flagenabled(flag_shadowing) ){ |
|
566 |
|
if( ht_lookup(&globals, tan->name)){ |
|
567 |
|
char buf[1024]; |
|
568 |
|
snprintf(buf, 1024, "Local variable %s shadows global variable", tan->name); |
|
569 |
|
yyerrorline(semanticerror, lineno -1, buf); |
|
570 |
|
} else if( ht_lookup(¶ms, tan->name)){ |
|
571 |
|
char buf[1024]; |
|
572 |
|
snprintf(buf, 1024, "Local variable %s shadows parameter", tan->name); |
|
573 |
|
yyerrorline(semanticerror, lineno -1, buf); |
|
574 |
|
} |
|
|
558 |
|
yyerrorline(syntaxerror, lineno - 1, "Constants must be initialized"); |
575 |
559 |
} |
} |
576 |
560 |
$$.ty = gNothing; |
$$.ty = gNothing; |
577 |
561 |
} |
} |
|
... |
... |
vardecl: vartypedecl newline { |
583 |
567 |
if(infunction && !ht_lookup(&initialized, tan->name)){ |
if(infunction && !ht_lookup(&initialized, tan->name)){ |
584 |
568 |
put(&initialized, tan->name, (void*)1); |
put(&initialized, tan->name, (void*)1); |
585 |
569 |
} |
} |
586 |
|
if(infunction && flagenabled(flag_shadowing) ){ |
|
587 |
|
if( ht_lookup(&globals, tan->name)){ |
|
588 |
|
char buf[1024]; |
|
589 |
|
snprintf(buf, 1024, "Local variable %s shadows global variable", tan->name); |
|
590 |
|
yyerrorex(semanticerror, buf); |
|
591 |
|
} else if( ht_lookup(¶ms, tan->name)){ |
|
592 |
|
char buf[1024]; |
|
593 |
|
snprintf(buf, 1024, "Local variable %s shadows parameter", tan->name); |
|
594 |
|
yyerrorex(semanticerror, buf); |
|
595 |
|
} |
|
596 |
|
} |
|
597 |
570 |
canconvert($3.ty, tan->ty, -1); |
canconvert($3.ty, tan->ty, -1); |
598 |
571 |
$$.ty = gNothing; |
$$.ty = gNothing; |
599 |
572 |
} |
} |
File misc.c changed (mode: 100644) (index c6d42f3..64411e0) |
... |
... |
struct hashtable initialized; |
41 |
41 |
|
|
42 |
42 |
struct hashtable bad_natives_in_globals; |
struct hashtable bad_natives_in_globals; |
43 |
43 |
|
|
|
44 |
|
struct hashtable shadowed_variables; |
|
45 |
|
|
44 |
46 |
|
|
45 |
47 |
struct hashtable *curtab; |
struct hashtable *curtab; |
46 |
48 |
|
|
|
... |
... |
static void checkvarname(struct typeandname *tan, bool isarray) |
658 |
660 |
} |
} |
659 |
661 |
} |
} |
660 |
662 |
|
|
661 |
|
union node checkvardecl(struct typeandname *tan) |
|
|
663 |
|
void checkallshadowing(struct typeandname *tan){ |
|
664 |
|
struct typeandname *global = ht_lookup(&globals, tan->name); |
|
665 |
|
char buf[1024]; |
|
666 |
|
|
|
667 |
|
if( global ){ |
|
668 |
|
|
|
669 |
|
// once a variable is shadowed with an incompatible type every usage |
|
670 |
|
// of the shadowed variable in the script file (sic) cannot be used |
|
671 |
|
// safely anymore. Usages of the shadowed variable in the script |
|
672 |
|
// above the shadowing still work fine. |
|
673 |
|
tan->lineno = lineno; |
|
674 |
|
ht_put(&shadowed_variables, tan->name, tan); |
|
675 |
|
|
|
676 |
|
if(flagenabled(flag_shadowing)){ |
|
677 |
|
snprintf(buf, 1024, "Local variable %s shadows global variable", tan->name); |
|
678 |
|
yyerrorline(semanticerror, lineno, buf); |
|
679 |
|
} |
|
680 |
|
} else if( flagenabled(flag_shadowing) && ht_lookup(¶ms, tan->name)){ |
|
681 |
|
snprintf(buf, 1024, "Local variable %s shadows parameter", tan->name); |
|
682 |
|
yyerrorline(semanticerror, lineno, buf); |
|
683 |
|
} |
|
684 |
|
} |
|
685 |
|
|
|
686 |
|
union node checkvartypedecl(struct typeandname *tan) |
662 |
687 |
{ |
{ |
663 |
688 |
const char *name = tan->name; |
const char *name = tan->name; |
664 |
689 |
union node ret; |
union node ret; |
|
... |
... |
union node checkvardecl(struct typeandname *tan) |
667 |
692 |
ret.str = name; |
ret.str = name; |
668 |
693 |
put(curtab, name, tan); |
put(curtab, name, tan); |
669 |
694 |
|
|
|
695 |
|
|
|
696 |
|
if(infunction ){ |
|
697 |
|
// always an error |
|
698 |
|
checkwrongshadowing(tan, 0); |
|
699 |
|
|
|
700 |
|
// flag driven |
|
701 |
|
checkallshadowing(tan); |
|
702 |
|
} |
670 |
703 |
return ret; |
return ret; |
671 |
704 |
} |
} |
672 |
705 |
|
|
|
... |
... |
union node checkarraydecl(struct typeandname *tan) |
685 |
718 |
|
|
686 |
719 |
return ret; |
return ret; |
687 |
720 |
} |
} |
|
721 |
|
|
|
722 |
|
void checkwrongshadowing(const struct typeandname *tan, int linemod){ |
|
723 |
|
struct typeandname *global; |
|
724 |
|
char buf[1024]; |
|
725 |
|
if( (global = ht_lookup(&shadowed_variables, tan->name)) ){ |
|
726 |
|
if(! typeeq(getPrimitiveAncestor(global->ty), getPrimitiveAncestor(tan->ty))){ |
|
727 |
|
snprintf(buf, 1024, "Global variable %s is used after it was shadowed with an incompatible type in line %d", tan->name, global->lineno); |
|
728 |
|
yyerrorline(semanticerror, lineno - linemod, buf); |
|
729 |
|
} |
|
730 |
|
} |
|
731 |
|
} |
|
732 |
|
|