File blocks.c added (mode: 100644) (index 0000000..960f22c) |
|
1 |
|
#include <stdint.h> |
|
2 |
|
#include <stdio.h> |
|
3 |
|
|
|
4 |
|
#include "blocks.h" |
|
5 |
|
#include "misc.h" |
|
6 |
|
|
|
7 |
|
|
|
8 |
|
static struct block_start { |
|
9 |
|
int lineno; |
|
10 |
|
enum block_type type; |
|
11 |
|
|
|
12 |
|
}; |
|
13 |
|
|
|
14 |
|
static struct block_start *blocks = NULL; |
|
15 |
|
static size_t size = 0; |
|
16 |
|
static size_t capacity; |
|
17 |
|
|
|
18 |
|
static char* names[] = { "function", "loop", "if" }; |
|
19 |
|
|
|
20 |
|
|
|
21 |
|
void block_push(int lineno, enum block_type type){ |
|
22 |
|
if(! blocks){ |
|
23 |
|
capacity = 7; |
|
24 |
|
blocks = malloc(sizeof(struct block_start) * capacity); |
|
25 |
|
}else if(size >= capacity){ |
|
26 |
|
capacity *= 2; |
|
27 |
|
realloc(blocks, capacity * sizeof(struct block_start)); |
|
28 |
|
} |
|
29 |
|
|
|
30 |
|
blocks[size].type = type; |
|
31 |
|
blocks[size].lineno = lineno; |
|
32 |
|
|
|
33 |
|
size++; |
|
34 |
|
} |
|
35 |
|
|
|
36 |
|
bool block_pop(enum block_type type, char *buf, size_t len){ |
|
37 |
|
if(size == 0){ |
|
38 |
|
// Standalone end* found |
|
39 |
|
snprintf(buf, len, "[1] Standalone %s (no corresponding opening block)", names[type]); |
|
40 |
|
return false; |
|
41 |
|
} |
|
42 |
|
|
|
43 |
|
|
|
44 |
|
if(blocks[size-1].type != type){ |
|
45 |
|
// Missing end* for blocks[size-1].type |
|
46 |
|
snprintf(buf, len, "[2] Missing end%s for block openend in line %d", names[blocks[size-1].type], blocks[size-1].lineno); |
|
47 |
|
return false; |
|
48 |
|
} |
|
49 |
|
|
|
50 |
|
size--; |
|
51 |
|
return true; |
|
52 |
|
} |
|
53 |
|
|
|
54 |
|
|
|
55 |
|
void block_missing_error(int x, char *msg, size_t len){ |
|
56 |
|
if(size == 0){ |
|
57 |
|
return; |
|
58 |
|
} |
|
59 |
|
snprintf(msg, len, "[%d] [3] Missing end%s for block opened in line %d.", x, names[blocks[size-1].type], blocks[size-1].lineno); |
|
60 |
|
size--; |
|
61 |
|
} |
File grammar.y changed (mode: 100644) (index 939157c..e47aacf) |
9 |
9 |
#include <string.h> |
#include <string.h> |
10 |
10 |
#include "token.yy.h" |
#include "token.yy.h" |
11 |
11 |
#include "misc.h" |
#include "misc.h" |
|
12 |
|
#include "blocks.h" |
12 |
13 |
|
|
13 |
14 |
|
|
14 |
15 |
#define YYSTYPE union node |
#define YYSTYPE union node |
|
... |
... |
funcdefncore: funcbegin localblock codeblock funcend { |
338 |
339 |
fnannotations = pjass_flags; |
fnannotations = pjass_flags; |
339 |
340 |
} |
} |
340 |
341 |
| funcbegin localblock codeblock { |
| funcbegin localblock codeblock { |
341 |
|
yyerrorex(syntaxerror, "Missing endfunction"); |
|
|
342 |
|
|
|
343 |
|
char msg[1024]; |
|
344 |
|
block_missing_error(2, msg, 1024); |
|
345 |
|
yyerrorex(syntaxerror, msg); |
|
346 |
|
|
342 |
347 |
ht_clear(¶ms); |
ht_clear(¶ms); |
343 |
348 |
ht_clear(&locals); |
ht_clear(&locals); |
344 |
349 |
ht_clear(&initialized); |
ht_clear(&initialized); |
|
... |
... |
funcend: ENDFUNCTION { |
355 |
360 |
inblock = 0; |
inblock = 0; |
356 |
361 |
inconstant = 0; |
inconstant = 0; |
357 |
362 |
infunction = 0; |
infunction = 0; |
|
363 |
|
|
|
364 |
|
char msg[1024]; |
|
365 |
|
|
|
366 |
|
if(! block_pop(Function, msg, 1024)){ |
|
367 |
|
yyerrorex(syntaxerror, msg); |
|
368 |
|
} |
358 |
369 |
} |
} |
359 |
370 |
; |
; |
360 |
371 |
|
|
|
... |
... |
funcbegin: FUNCTION rid TAKES optparam_list returnorreturns opttype { |
367 |
378 |
infunction = 1; |
infunction = 1; |
368 |
379 |
$$ = checkfunctionheader($2.str, $4.pl, $6.ty); |
$$ = checkfunctionheader($2.str, $4.pl, $6.ty); |
369 |
380 |
$$.fd->isconst = 0; |
$$.fd->isconst = 0; |
|
381 |
|
block_push(lineno, Function); |
370 |
382 |
} |
} |
371 |
383 |
| CONSTANT FUNCTION rid TAKES optparam_list returnorreturns opttype { |
| CONSTANT FUNCTION rid TAKES optparam_list returnorreturns opttype { |
372 |
384 |
inconstant = 1; |
inconstant = 1; |
373 |
385 |
infunction = 1; |
infunction = 1; |
374 |
386 |
$$ = checkfunctionheader($3.str, $5.pl, $7.ty); |
$$ = checkfunctionheader($3.str, $5.pl, $7.ty); |
375 |
387 |
$$.fd->isconst = 1; |
$$.fd->isconst = 1; |
|
388 |
|
block_push(lineno, Function); |
376 |
389 |
} |
} |
377 |
390 |
; |
; |
378 |
391 |
|
|
|
... |
... |
codeblock: /* empty */ { $$.ty = gEmpty; } |
388 |
401 |
statement: newline { $$.ty = gEmpty; } |
statement: newline { $$.ty = gEmpty; } |
389 |
402 |
| CALL funccall newline{ $$.ty = gAny;} |
| CALL funccall newline{ $$.ty = gAny;} |
390 |
403 |
/*1 2 3 4 5 6 7 8 9 */ |
/*1 2 3 4 5 6 7 8 9 */ |
391 |
|
| IF expr THEN newline codeblock elsifseq elseseq ENDIF newline { |
|
|
404 |
|
| ifstart expr THEN newline codeblock elsifseq elseseq ifend newline { |
392 |
405 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
393 |
406 |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
394 |
407 |
} |
} |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
440 |
453 |
} |
} |
441 |
454 |
} |
} |
442 |
455 |
| loopstart newline codeblock loopend newline {$$.ty = $3.ty;} |
| loopstart newline codeblock loopend newline {$$.ty = $3.ty;} |
443 |
|
| loopstart newline codeblock {$$.ty = $3.ty; yyerrorex(syntaxerror, "Missing endloop");} |
|
|
456 |
|
| loopstart newline codeblock { |
|
457 |
|
$$.ty = $3.ty; |
|
458 |
|
|
|
459 |
|
char msg[1024]; |
|
460 |
|
block_missing_error(0, msg, 1024); |
|
461 |
|
yyerrorex(syntaxerror, msg); |
|
462 |
|
|
|
463 |
|
} |
444 |
464 |
| EXITWHEN expr newline { canconvert($2.ty, gBoolean, -1); if (!inloop) yyerrorline(syntaxerror, lineno - 1, "Exitwhen outside of loop"); $$.ty = gAny;} |
| EXITWHEN expr newline { canconvert($2.ty, gBoolean, -1); if (!inloop) yyerrorline(syntaxerror, lineno - 1, "Exitwhen outside of loop"); $$.ty = gAny;} |
445 |
465 |
| RETURN expr newline { |
| RETURN expr newline { |
446 |
466 |
$$.ty = mkretty($2.ty, 1); |
$$.ty = mkretty($2.ty, 1); |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
456 |
476 |
} |
} |
457 |
477 |
| DEBUG statement {$$.ty = gAny;} |
| DEBUG statement {$$.ty = gAny;} |
458 |
478 |
/*1 2 3 4 5 6 7 */ |
/*1 2 3 4 5 6 7 */ |
459 |
|
| IF expr THEN newline codeblock elsifseq elseseq { |
|
|
479 |
|
| ifstart expr THEN newline codeblock elsifseq elseseq { |
460 |
480 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
461 |
481 |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
$$.ty = combinetype($5.ty, combinetype($6.ty, $7.ty)); |
462 |
|
yyerrorex(syntaxerror, "Missing endif"); |
|
|
482 |
|
|
|
483 |
|
char msg[1024]; |
|
484 |
|
block_missing_error(1, msg, 1024); |
|
485 |
|
yyerrorex(syntaxerror, msg); |
463 |
486 |
} |
} |
464 |
|
| IF expr newline { |
|
|
487 |
|
| ifstart expr newline { |
465 |
488 |
canconvert($2.ty, gBoolean, -1); |
canconvert($2.ty, gBoolean, -1); |
466 |
489 |
$$.ty = gAny; |
$$.ty = gAny; |
467 |
490 |
yyerrorex(syntaxerror, "Missing then or non valid expression"); |
yyerrorex(syntaxerror, "Missing then or non valid expression"); |
|
... |
... |
statement: newline { $$.ty = gEmpty; } |
474 |
497 |
| error {$$.ty = gAny; } |
| error {$$.ty = gAny; } |
475 |
498 |
; |
; |
476 |
499 |
|
|
477 |
|
loopstart: LOOP {inloop++;} |
|
478 |
|
; |
|
|
500 |
|
loopstart: LOOP { |
|
501 |
|
inloop++; |
|
502 |
|
block_push(lineno, Loop); |
|
503 |
|
}; |
479 |
504 |
|
|
480 |
|
loopend: ENDLOOP {inloop--;} |
|
481 |
|
; |
|
|
505 |
|
loopend: ENDLOOP { |
|
506 |
|
inloop--; |
|
507 |
|
|
|
508 |
|
char msg[1024]; |
|
509 |
|
if(! block_pop(Loop, msg, 1024)){ |
|
510 |
|
yyerrorex(syntaxerror, msg); |
|
511 |
|
} |
|
512 |
|
}; |
|
513 |
|
|
|
514 |
|
ifstart: IF { |
|
515 |
|
block_push(lineno, If); |
|
516 |
|
}; |
|
517 |
|
|
|
518 |
|
ifend: ENDIF { |
|
519 |
|
char msg[1024]; |
|
520 |
|
if(! block_pop(If, msg, 1024)){ |
|
521 |
|
yyerrorex(syntaxerror, msg); |
|
522 |
|
} |
|
523 |
|
}; |
482 |
524 |
|
|
483 |
525 |
elseseq: /* empty */ { $$.ty = gAny; } |
elseseq: /* empty */ { $$.ty = gAny; } |
484 |
526 |
| ELSE newline codeblock { |
| ELSE newline codeblock { |