as_parser.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. #include "as_parser.h"
  2. #include "as_tokenizer.h"
  3. #include "utils.h"
  4. #include <stdio.h>
  5. #include <stdlib.h>
  6. #include <string.h>
  7. // BNF
  8. // ===
  9. //
  10. // <prog> ::= <stmts>
  11. // <stmts> ::= <stmt> newline | <stmt> newline <stmts>
  12. // <stmt> ::= <label> <instr> | <instr> | <label>
  13. // <instr> ::= <op> | <op> arg | <op> tag
  14. // <label> ::= tag ":"
  15. // <op> ::= "add" | "sub" | "mul" | "div" | "mod" | "eq"
  16. Prog parseProg(Allocator alct, TokenStream ts) {
  17. Prog p = allocate(alct, sizeof(struct prog));
  18. p->stmts = parseStmts(alct, ts);
  19. return p;
  20. }
  21. Stmts parseStmts(Allocator alct, TokenStream ts) {
  22. Stmts ss = allocate(alct, sizeof(struct stmts));
  23. ss->stmts = allocate(alct, sizeof(Stmt));
  24. ss->stmts[0] = NULL;
  25. int capacity = 0;
  26. int len = 0;
  27. while (peekToken(alct, ts)->type != ENDOFFILE) {
  28. Stmt s = parseStmt(alct, ts);
  29. if (s == NULL) continue;
  30. if (len == capacity) {
  31. size_t newCapacity = capacity * 2 + 1;
  32. void* newStmts = allocate(alct, sizeof(Stmt*) * newCapacity);
  33. memcpy(newStmts, ss->stmts, sizeof(Stmt*) * capacity);
  34. ss->stmts = newStmts;
  35. capacity = newCapacity;
  36. }
  37. // expect newline
  38. if (peekToken(alct, ts)->type == NEWLINE) {
  39. nextToken(alct, ts);
  40. } else {
  41. fprintf(stderr, "%d:%d Expect NEWLINE.\n", peekToken(alct, ts)->line, peekToken(alct, ts)->col);
  42. }
  43. ss->stmts[len] = s;
  44. len++;
  45. }
  46. ss->stmts[len] = NULL;
  47. return ss;
  48. }
  49. Label parseLabel(Allocator alct, TokenStream ts) {
  50. Token t = nextToken(alct, ts);
  51. if (t->type != TAG) {
  52. fprintf(stderr, "%d:%d Expect LABEL.\n", t->line, t->col);
  53. exit(-1);
  54. }
  55. Label l = allocate(alct, sizeof(Label));
  56. l->name = t->sval;
  57. t = nextToken(alct, ts);
  58. if (t->type != COLON) {
  59. fprintf(stderr, "%d:%d Expect COLON.\n", t->line, t->col);
  60. exit(-1);
  61. }
  62. return l;
  63. }
  64. Stmt parseStmt(Allocator alct, TokenStream ts) {
  65. Token t = peekToken(alct, ts);
  66. Stmt stmt = allocate(alct, sizeof(struct stmt));
  67. stmt->label = NULL;
  68. stmt->instr = NULL;
  69. if (t->type == TAG) {
  70. stmt->label = parseLabel(alct, ts);
  71. if (peekToken(alct, ts)->type == NEWLINE) {
  72. return stmt;
  73. }
  74. t = peekToken(alct, ts);
  75. }
  76. if (t->type == OP) {
  77. stmt->instr = parseInstr(alct, ts);
  78. if (peekToken(alct, ts)->type == NEWLINE) {
  79. return stmt;
  80. }
  81. }
  82. if (t->type == NEWLINE) {
  83. return NULL;
  84. }
  85. fprintf(stderr, "%d:%d Expect lable + instruction, lable, or instruction.\n", t->line, t->col);
  86. exit(-1);
  87. }
  88. enum op parseOp(Allocator alct, TokenStream ts) {
  89. Token t = nextToken(alct, ts);
  90. enum op op;
  91. if (t->type == OP) {
  92. op = str2op(t->sval);
  93. if (op == OPEND) {
  94. fprintf(stderr, "%d:%d Invalid OP.\n", t->line, t->col);
  95. exit(-1);
  96. }
  97. } else {
  98. fprintf(stderr, "%d:%d Expect OP.\n", t->line, t->col);
  99. exit(-1);
  100. }
  101. return op;
  102. }
  103. Instr parseInstr(Allocator alct, TokenStream ts) {
  104. Token t = peekToken(alct, ts);
  105. Instr i = allocate(alct, sizeof(struct instr));
  106. i->tagName = NULL;
  107. i->arg = NULL;
  108. i->op = OPEND;
  109. if (t->type == OP) {
  110. i->op = parseOp(alct, ts);
  111. t = peekToken(alct, ts);
  112. if (t->type == ARG) {
  113. Arg a = allocate(alct, sizeof(struct arg));
  114. a->ival = t->ival;
  115. a->fval = t->fval;
  116. i->arg = a;
  117. nextToken(alct, ts);
  118. } else if (t->type == TAG) {
  119. i->tagName = t->sval;
  120. nextToken(alct, ts);
  121. }
  122. }
  123. return i;
  124. }