File grammar.y changed (mode: 100644) (index a3277eb..b8c3220) |
... |
... |
int main(int argc, char **argv) |
130 |
130 |
%token INTLIT |
%token INTLIT |
131 |
131 |
%token REALLIT |
%token REALLIT |
132 |
132 |
%token UNITTYPEINT |
%token UNITTYPEINT |
|
133 |
|
%token DERECTIVE |
133 |
134 |
|
|
134 |
135 |
%right EQUALS |
%right EQUALS |
135 |
136 |
%left AND |
%left AND |
|
... |
... |
int main(int argc, char **argv) |
141 |
142 |
|
|
142 |
143 |
%% |
%% |
143 |
144 |
|
|
|
145 |
|
|
|
146 |
|
|
144 |
147 |
program: topscopes globdefs topscopes funcdefns |
program: topscopes globdefs topscopes funcdefns |
145 |
148 |
; |
; |
146 |
149 |
|
|
|
... |
... |
funcdefns: /* empty */ |
157 |
160 |
; |
; |
158 |
161 |
|
|
159 |
162 |
globdefs: /* empty */ |
globdefs: /* empty */ |
160 |
|
| GLOBALS NEWLINE vardecls ENDGLOBALS endglobalsmarker |
|
|
163 |
|
| GLOBALS newline vardecls ENDGLOBALS endglobalsmarker |
161 |
164 |
| GLOBALS vardecls ENDGLOBALS endglobalsmarker {yyerrorline(0, lineno - 1, "Missing linebreak before global declaration");} |
| GLOBALS vardecls ENDGLOBALS endglobalsmarker {yyerrorline(0, lineno - 1, "Missing linebreak before global declaration");} |
162 |
165 |
; |
; |
163 |
166 |
|
|
|
... |
... |
vardecls: /* empty */ |
168 |
171 |
| vd vardecls |
| vd vardecls |
169 |
172 |
; |
; |
170 |
173 |
|
|
171 |
|
vd: NEWLINE |
|
|
174 |
|
vd: newline |
172 |
175 |
| vardecl |
| vardecl |
173 |
176 |
; |
; |
174 |
177 |
|
|
|
... |
... |
funcdecls: /* empty */ |
176 |
179 |
| fd funcdecls |
| fd funcdecls |
177 |
180 |
; |
; |
178 |
181 |
|
|
179 |
|
fd: NEWLINE |
|
|
182 |
|
fd: newline |
180 |
183 |
| funcdecl |
| funcdecl |
181 |
184 |
; |
; |
182 |
185 |
|
|
|
... |
... |
typedefs: /* empty */ |
184 |
187 |
| td typedefs |
| td typedefs |
185 |
188 |
; |
; |
186 |
189 |
|
|
187 |
|
td: NEWLINE |
|
|
190 |
|
td: newline |
188 |
191 |
| typedef |
| typedef |
189 |
192 |
; |
; |
190 |
193 |
|
|
|
... |
... |
funccall: rid LPAREN exprlistcompl RPAREN { |
302 |
305 |
$$.ty = fd->ret; |
$$.ty = fd->ret; |
303 |
306 |
} |
} |
304 |
307 |
} |
} |
305 |
|
| rid LPAREN exprlistcompl NEWLINE { |
|
|
308 |
|
| rid LPAREN exprlistcompl newline { |
306 |
309 |
yyerrorex(0, "Missing ')'"); |
yyerrorex(0, "Missing ')'"); |
307 |
310 |
struct funcdecl *fd = lookup(&functions, $1.str); |
struct funcdecl *fd = lookup(&functions, $1.str); |
308 |
311 |
if (fd == NULL) { |
if (fd == NULL) { |
|
... |
... |
nativefuncdecl: NATIVE rid TAKES optparam_list RETURNS opttype |
379 |
382 |
} |
} |
380 |
383 |
; |
; |
381 |
384 |
|
|
382 |
|
funcdefn: NEWLINE |
|
|
385 |
|
funcdefn: newline |
383 |
386 |
| funcdefncore |
| funcdefncore |
384 |
387 |
| statement { yyerrorex(0, "Statement outside of function"); } |
| statement { yyerrorex(0, "Statement outside of function"); } |
385 |
388 |
; |
; |
386 |
389 |
|
|
387 |
|
funcdefncore: funcbegin localblock codeblock funcend { |
|
|
390 |
|
derectives: /* empty */ |
|
391 |
|
| DERECTIVE derectives |
|
392 |
|
; |
|
393 |
|
|
|
394 |
|
funcdefncore: derectives funcbegin localblock codeblock funcend { |
388 |
395 |
if(retval != gNothing) { |
if(retval != gNothing) { |
389 |
|
//if ($3.ty == gAny || $3.ty == gNone) |
|
390 |
|
if(!getTypeTag($3.ty)) |
|
|
396 |
|
if(!getTypeTag($4.ty)) |
391 |
397 |
yyerrorline(1, lineno - 1, "Missing return"); |
yyerrorline(1, lineno - 1, "Missing return"); |
392 |
398 |
else if (returnbug) |
else if (returnbug) |
393 |
|
canconvertreturn($3.ty, retval, -1); |
|
|
399 |
|
canconvertreturn($4.ty, retval, -1); |
394 |
400 |
} |
} |
395 |
401 |
} |
} |
396 |
|
| funcbegin localblock codeblock {yyerrorex(0, "Missing endfunction"); clear(¶ms); clear(&locals); clear(&initialized); curtab = &globals;} |
|
|
402 |
|
| derectives funcbegin localblock codeblock {yyerrorex(0, "Missing endfunction"); clear(¶ms); clear(&locals); clear(&initialized); curtab = &globals;} |
397 |
403 |
; |
; |
398 |
404 |
|
|
399 |
405 |
funcend: ENDFUNCTION { clear(¶ms); clear(&locals); clear(&initialized); curtab = &globals; inblock = 0; inconstant = 0; infunction = 0; } |
funcend: ENDFUNCTION { clear(¶ms); clear(&locals); clear(&initialized); curtab = &globals; inblock = 0; inconstant = 0; infunction = 0; } |
|
... |
... |
codeblock: /* empty */ { $$.ty = gEmpty; } |
492 |
498 |
} |
} |
493 |
499 |
; |
; |
494 |
500 |
|
|
495 |
|
statement: NEWLINE { $$.ty = gEmpty; } |
|
496 |
|
| CALL funccall NEWLINE{ $$.ty = gAny;} |
|
|
501 |
|
statement: newline { $$.ty = gEmpty; } |
|
502 |
|
| CALL funccall newline{ $$.ty = gAny;} |
497 |
503 |
/*1 2 3 4 5 6 7 8 9 */ |
/*1 2 3 4 5 6 7 8 9 */ |
498 |
|
| IF expr THEN NEWLINE codeblock elsifseq elseseq ENDIF NEWLINE { |
|
|
504 |
|
| IF expr THEN newline codeblock elsifseq elseseq ENDIF newline { |
499 |
505 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
500 |
506 |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
501 |
507 |
} |
} |
502 |
|
| SET rid EQUALS expr NEWLINE { if (getVariable($2.str)->isarray) { |
|
|
508 |
|
| SET rid EQUALS expr newline { if (getVariable($2.str)->isarray) { |
503 |
509 |
char ebuf[1024]; |
char ebuf[1024]; |
504 |
510 |
snprintf(ebuf, 1024, "Index missing for array variable %s", $2.str); |
snprintf(ebuf, 1024, "Index missing for array variable %s", $2.str); |
505 |
511 |
yyerrorline(3, lineno - 1, ebuf); |
yyerrorline(3, lineno - 1, ebuf); |
|
... |
... |
statement: NEWLINE { $$.ty = gEmpty; } |
517 |
523 |
put(&initialized, $2.str, (void*)1); |
put(&initialized, $2.str, (void*)1); |
518 |
524 |
} |
} |
519 |
525 |
} |
} |
520 |
|
| SET rid LBRACKET expr RBRACKET EQUALS expr NEWLINE{ |
|
|
526 |
|
| SET rid LBRACKET expr RBRACKET EQUALS expr newline{ |
521 |
527 |
const struct typeandname *tan = getVariable($2.str); |
const struct typeandname *tan = getVariable($2.str); |
522 |
528 |
$$.ty = gAny; |
$$.ty = gAny; |
523 |
529 |
if (tan->ty != gAny) { |
if (tan->ty != gAny) { |
|
... |
... |
statement: NEWLINE { $$.ty = gEmpty; } |
532 |
538 |
validateGlobalAssignment($2.str); |
validateGlobalAssignment($2.str); |
533 |
539 |
} |
} |
534 |
540 |
} |
} |
535 |
|
| loopstart NEWLINE codeblock loopend NEWLINE {$$.ty = $3.ty;} |
|
536 |
|
| loopstart NEWLINE codeblock {$$.ty = $3.ty; yyerrorex(0, "Missing endloop");} |
|
537 |
|
| EXITWHEN expr NEWLINE { canconvert($2.ty, gBoolean, -1); if (!inloop) yyerrorline(0, lineno - 1, "Exitwhen outside of loop"); $$.ty = gAny;} |
|
538 |
|
| RETURN expr NEWLINE { |
|
|
541 |
|
| loopstart newline codeblock loopend newline {$$.ty = $3.ty;} |
|
542 |
|
| loopstart newline codeblock {$$.ty = $3.ty; yyerrorex(0, "Missing endloop");} |
|
543 |
|
| EXITWHEN expr newline { canconvert($2.ty, gBoolean, -1); if (!inloop) yyerrorline(0, lineno - 1, "Exitwhen outside of loop"); $$.ty = gAny;} |
|
544 |
|
| RETURN expr newline { |
539 |
545 |
$$.ty = mkretty($2.ty, 1); |
$$.ty = mkretty($2.ty, 1); |
540 |
546 |
if(retval == gNothing) |
if(retval == gNothing) |
541 |
547 |
yyerrorline(1, lineno - 1, "Cannot return value from function that returns nothing"); |
yyerrorline(1, lineno - 1, "Cannot return value from function that returns nothing"); |
542 |
548 |
else if (!returnbug) |
else if (!returnbug) |
543 |
549 |
canconvertreturn($2.ty, retval, 0); |
canconvertreturn($2.ty, retval, 0); |
544 |
550 |
} |
} |
545 |
|
| RETURN NEWLINE { |
|
|
551 |
|
| RETURN newline { |
546 |
552 |
if (retval != gNothing) |
if (retval != gNothing) |
547 |
553 |
yyerrorline(1, lineno - 1, "Return nothing in function that should return value"); |
yyerrorline(1, lineno - 1, "Return nothing in function that should return value"); |
548 |
554 |
$$.ty = mkretty(gAny, 1); |
$$.ty = mkretty(gAny, 1); |
549 |
555 |
} |
} |
550 |
556 |
| DEBUG statement {$$.ty = gAny;} |
| DEBUG statement {$$.ty = gAny;} |
551 |
557 |
/*1 2 3 4 5 6 7 */ |
/*1 2 3 4 5 6 7 */ |
552 |
|
| IF expr THEN NEWLINE codeblock elsifseq elseseq { |
|
|
558 |
|
| IF expr THEN newline codeblock elsifseq elseseq { |
553 |
559 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
554 |
560 |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
555 |
561 |
yyerrorex(0, "Missing endif"); |
yyerrorex(0, "Missing endif"); |
556 |
562 |
} |
} |
557 |
|
| IF expr NEWLINE { |
|
|
563 |
|
| IF expr newline { |
558 |
564 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
559 |
565 |
$$.ty = gAny; |
$$.ty = gAny; |
560 |
566 |
yyerrorex(0, "Missing then or non valid expression"); |
yyerrorex(0, "Missing then or non valid expression"); |
561 |
567 |
} |
} |
562 |
|
| SET funccall NEWLINE{$$.ty = gAny; yyerrorline(0, lineno - 1, "Call expected instead of set");} |
|
|
568 |
|
| SET funccall newline{$$.ty = gAny; yyerrorline(0, lineno - 1, "Call expected instead of set");} |
563 |
569 |
| lvardecl {yyerrorex(0, "Local declaration after first statement");} |
| lvardecl {yyerrorex(0, "Local declaration after first statement");} |
564 |
570 |
| error {$$.ty = gAny; } |
| error {$$.ty = gAny; } |
565 |
571 |
; |
; |
|
... |
... |
loopend: ENDLOOP {inloop--;} |
571 |
577 |
; |
; |
572 |
578 |
|
|
573 |
579 |
elseseq: /* empty */ { $$.ty = gAny; } |
elseseq: /* empty */ { $$.ty = gAny; } |
574 |
|
| ELSE NEWLINE codeblock { |
|
|
580 |
|
| ELSE newline codeblock { |
575 |
581 |
$$.ty = $3.ty; |
$$.ty = $3.ty; |
576 |
582 |
} |
} |
577 |
583 |
; |
; |
578 |
584 |
|
|
579 |
585 |
elsifseq: /* empty */ { $$.ty = mkretty(gEmpty, 1); } |
elsifseq: /* empty */ { $$.ty = mkretty(gEmpty, 1); } |
580 |
586 |
/* 1 2 3 4 5 6 */ |
/* 1 2 3 4 5 6 */ |
581 |
|
| ELSEIF expr THEN NEWLINE codeblock elsifseq { |
|
|
587 |
|
| ELSEIF expr THEN newline codeblock elsifseq { |
582 |
588 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
583 |
589 |
|
|
584 |
590 |
if(typeeq($6.ty, gEmpty)){ |
if(typeeq($6.ty, gEmpty)){ |
|
... |
... |
vartypedecl: type rid { |
769 |
775 |
|
|
770 |
776 |
localblock: endlocalsmarker |
localblock: endlocalsmarker |
771 |
777 |
| lvardecl localblock |
| lvardecl localblock |
772 |
|
| NEWLINE localblock |
|
|
778 |
|
| newline localblock |
773 |
779 |
; |
; |
774 |
780 |
|
|
775 |
781 |
endlocalsmarker: /* empty */ { fCurrent = 0; } |
endlocalsmarker: /* empty */ { fCurrent = 0; } |
|
... |
... |
lvardecl: LOCAL vardecl { } |
780 |
786 |
| typedef { yyerrorex(3,"Types can not be extended inside functions"); } |
| typedef { yyerrorex(3,"Types can not be extended inside functions"); } |
781 |
787 |
; |
; |
782 |
788 |
|
|
783 |
|
vardecl: vartypedecl NEWLINE { |
|
|
789 |
|
vardecl: vartypedecl newline { |
784 |
790 |
const struct typeandname *tan = getVariable($1.str); |
const struct typeandname *tan = getVariable($1.str); |
785 |
791 |
if (tan->isconst) { |
if (tan->isconst) { |
786 |
792 |
yyerrorline(3, lineno - 1, "Constants must be initialized"); |
yyerrorline(3, lineno - 1, "Constants must be initialized"); |
787 |
793 |
} |
} |
788 |
794 |
$$.ty = gNothing; |
$$.ty = gNothing; |
789 |
795 |
} |
} |
790 |
|
| vartypedecl EQUALS expr NEWLINE { |
|
|
796 |
|
| vartypedecl EQUALS expr newline { |
791 |
797 |
const struct typeandname *tan = getVariable($1.str); |
const struct typeandname *tan = getVariable($1.str); |
792 |
798 |
if (tan->isarray) { |
if (tan->isarray) { |
793 |
799 |
yyerrorex(3, "Arrays cannot be directly initialized"); |
yyerrorex(3, "Arrays cannot be directly initialized"); |
|
... |
... |
primtype: HANDLE { $$.ty = lookup(&types, yytext); } |
841 |
847 |
| CODE { $$.ty = lookup(&types, yytext); } |
| CODE { $$.ty = lookup(&types, yytext); } |
842 |
848 |
; |
; |
843 |
849 |
|
|
|
850 |
|
newline: NEWLINE |
|
851 |
|
| DERECTIVE |
|
852 |
|
; |