|
|
@ -1,32 +1,212 @@
|
|
|
|
/*
|
|
|
|
/*
|
|
|
|
* RVBTCC By Yaossg
|
|
|
|
* RVBTCC By Yaossg
|
|
|
|
* A lightweight bootstrapping compiler in less than 2000 lines.
|
|
|
|
* A lightweight bootstrapping compiler in 2000 lines.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* It aims to demonstrate how to write a bootstrapping compiler in no time.
|
|
|
|
* It aims to demonstrate how to write a bootstrapping compiler in no time.
|
|
|
|
* Syntax is similar to C, output is RISC-V assembly.
|
|
|
|
* Syntax is similar to C, output is RISC-V assembly.
|
|
|
|
* Only dependent on some glibc functions for I/O.
|
|
|
|
* Only dependent on some glibc functions for I/O.
|
|
|
|
* Purely for educational purposes. Do not use in production.
|
|
|
|
* Purely for educational purposes. Do not use in production.
|
|
|
|
*
|
|
|
|
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
// libc dependency
|
|
|
|
// glibc dependency
|
|
|
|
|
|
|
|
|
|
|
|
extern void* stdin;
|
|
|
|
extern void* stdin;
|
|
|
|
extern void* stdout;
|
|
|
|
extern void* stdout;
|
|
|
|
extern void* stderr;
|
|
|
|
extern void* stderr;
|
|
|
|
|
|
|
|
|
|
|
|
int printf(const char* format, ...);
|
|
|
|
int printf(char* format, ...);
|
|
|
|
int getchar();
|
|
|
|
int getchar();
|
|
|
|
void exit(int status);
|
|
|
|
void exit(int status);
|
|
|
|
int fprintf(void* file, const char* format, ...);
|
|
|
|
int fprintf(void* file, char* format, ...);
|
|
|
|
int ungetc(int ch, void* file);
|
|
|
|
int ungetc(int ch, void* file);
|
|
|
|
|
|
|
|
|
|
|
|
void ungetchar(int ch) {
|
|
|
|
void ungetchar(int ch) {
|
|
|
|
ungetc(ch, stdin);
|
|
|
|
ungetc(ch, stdin);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// limitations
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
|
|
|
STRING_TABLE_SIZE = 65536,
|
|
|
|
|
|
|
|
STRING_LUT_SIZE = 4096,
|
|
|
|
|
|
|
|
ID_TABLE_SIZE = 65536,
|
|
|
|
|
|
|
|
ID_LUT_SIZE = 4096,
|
|
|
|
|
|
|
|
LOCAL_SIZE = 4096,
|
|
|
|
|
|
|
|
REG_SIZE = 4096,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// constants
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
|
|
|
TOKEN_EOF,
|
|
|
|
|
|
|
|
TOKEN_SEMICOLON,
|
|
|
|
|
|
|
|
TOKEN_ADD,
|
|
|
|
|
|
|
|
TOKEN_SUB,
|
|
|
|
|
|
|
|
TOKEN_MUL,
|
|
|
|
|
|
|
|
TOKEN_DIV,
|
|
|
|
|
|
|
|
TOKEN_REM,
|
|
|
|
|
|
|
|
TOKEN_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_COMMA,
|
|
|
|
|
|
|
|
TOKEN_LSHIFT,
|
|
|
|
|
|
|
|
TOKEN_RSHIFT,
|
|
|
|
|
|
|
|
TOKEN_AND,
|
|
|
|
|
|
|
|
TOKEN_OR,
|
|
|
|
|
|
|
|
TOKEN_XOR,
|
|
|
|
|
|
|
|
TOKEN_COMPL,
|
|
|
|
|
|
|
|
TOKEN_NOT,
|
|
|
|
|
|
|
|
TOKEN_LAND,
|
|
|
|
|
|
|
|
TOKEN_LOR,
|
|
|
|
|
|
|
|
TOKEN_ELLIPSIS,
|
|
|
|
|
|
|
|
TOKEN_INC,
|
|
|
|
|
|
|
|
TOKEN_DEC,
|
|
|
|
|
|
|
|
TOKEN_ADD_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_SUB_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_MUL_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_DIV_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_REM_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_AND_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_OR_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_XOR_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_LSHIFT_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_RSHIFT_ASSIGN,
|
|
|
|
|
|
|
|
TOKEN_QUESTION,
|
|
|
|
|
|
|
|
TOKEN_COLON,
|
|
|
|
|
|
|
|
TOKEN_EQ,
|
|
|
|
|
|
|
|
TOKEN_NE,
|
|
|
|
|
|
|
|
TOKEN_LT,
|
|
|
|
|
|
|
|
TOKEN_GT,
|
|
|
|
|
|
|
|
TOKEN_LE,
|
|
|
|
|
|
|
|
TOKEN_GE,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TOKEN_PAREN_LEFT = 50,
|
|
|
|
|
|
|
|
TOKEN_PAREN_RIGHT,
|
|
|
|
|
|
|
|
TOKEN_BRACKET_LEFT,
|
|
|
|
|
|
|
|
TOKEN_BRACKET_RIGHT,
|
|
|
|
|
|
|
|
TOKEN_BRACE_LEFT,
|
|
|
|
|
|
|
|
TOKEN_BRACE_RIGHT,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TOKEN_STRING = 99,
|
|
|
|
|
|
|
|
TOKEN_NUMBER,
|
|
|
|
|
|
|
|
TOKEN_ID,
|
|
|
|
|
|
|
|
TOKEN_IF,
|
|
|
|
|
|
|
|
TOKEN_ELSE,
|
|
|
|
|
|
|
|
TOKEN_WHILE,
|
|
|
|
|
|
|
|
TOKEN_FOR,
|
|
|
|
|
|
|
|
TOKEN_DO,
|
|
|
|
|
|
|
|
TOKEN_BREAK,
|
|
|
|
|
|
|
|
TOKEN_CONTINUE,
|
|
|
|
|
|
|
|
TOKEN_RETURN,
|
|
|
|
|
|
|
|
TOKEN_ENUM,
|
|
|
|
|
|
|
|
TOKEN_EXTERN,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TOKEN_VOID = 128,
|
|
|
|
|
|
|
|
TOKEN_INT,
|
|
|
|
|
|
|
|
TOKEN_CHAR,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
|
|
|
TYPE_VOID,
|
|
|
|
|
|
|
|
TYPE_INT,
|
|
|
|
|
|
|
|
TYPE_CHAR,
|
|
|
|
|
|
|
|
TYPE_VOID_PTR = 16,
|
|
|
|
|
|
|
|
TYPE_INT_PTR,
|
|
|
|
|
|
|
|
TYPE_CHAR_PTR,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TYPE_PTR_MASK = TYPE_VOID_PTR,
|
|
|
|
|
|
|
|
TYPE_TOKEN_MASK = TOKEN_VOID,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
|
|
|
KIND_TEMP,
|
|
|
|
|
|
|
|
KIND_SCALAR,
|
|
|
|
|
|
|
|
KIND_ARRAY,
|
|
|
|
|
|
|
|
KIND_FUNCTION,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
|
|
|
REG_ZERO,
|
|
|
|
|
|
|
|
REG_RA,
|
|
|
|
|
|
|
|
REG_SP,
|
|
|
|
|
|
|
|
REG_GP,
|
|
|
|
|
|
|
|
REG_TP,
|
|
|
|
|
|
|
|
REG_T0,
|
|
|
|
|
|
|
|
REG_T1,
|
|
|
|
|
|
|
|
REG_T2,
|
|
|
|
|
|
|
|
REG_FP,
|
|
|
|
|
|
|
|
REG_S1,
|
|
|
|
|
|
|
|
REG_A0,
|
|
|
|
|
|
|
|
REG_A1,
|
|
|
|
|
|
|
|
REG_A2,
|
|
|
|
|
|
|
|
REG_A3,
|
|
|
|
|
|
|
|
REG_A4,
|
|
|
|
|
|
|
|
REG_A5,
|
|
|
|
|
|
|
|
REG_A6,
|
|
|
|
|
|
|
|
REG_A7,
|
|
|
|
|
|
|
|
REG_S2,
|
|
|
|
|
|
|
|
REG_S3,
|
|
|
|
|
|
|
|
REG_S4,
|
|
|
|
|
|
|
|
REG_S5,
|
|
|
|
|
|
|
|
REG_S6,
|
|
|
|
|
|
|
|
REG_S7,
|
|
|
|
|
|
|
|
REG_S8,
|
|
|
|
|
|
|
|
REG_S9,
|
|
|
|
|
|
|
|
REG_S10,
|
|
|
|
|
|
|
|
REG_S11,
|
|
|
|
|
|
|
|
REG_T3,
|
|
|
|
|
|
|
|
REG_T4,
|
|
|
|
|
|
|
|
REG_T5,
|
|
|
|
|
|
|
|
REG_T6,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char* reg_name(int reg) {
|
|
|
|
|
|
|
|
// special begin
|
|
|
|
|
|
|
|
if (reg == REG_ZERO) return "zero";
|
|
|
|
|
|
|
|
if (reg == REG_RA) return "ra";
|
|
|
|
|
|
|
|
if (reg == REG_SP) return "sp";
|
|
|
|
|
|
|
|
if (reg == REG_GP) return "gp";
|
|
|
|
|
|
|
|
if (reg == REG_TP) return "tp";
|
|
|
|
|
|
|
|
if (reg == REG_T0) return "t0";
|
|
|
|
|
|
|
|
if (reg == REG_T1) return "t1";
|
|
|
|
|
|
|
|
if (reg == REG_T2) return "t2";
|
|
|
|
|
|
|
|
if (reg == REG_FP) return "fp";
|
|
|
|
|
|
|
|
if (reg == REG_S1) return "s1";
|
|
|
|
|
|
|
|
if (reg == REG_A0) return "a0";
|
|
|
|
|
|
|
|
if (reg == REG_A1) return "a1";
|
|
|
|
|
|
|
|
if (reg == REG_A2) return "a2";
|
|
|
|
|
|
|
|
if (reg == REG_A3) return "a3";
|
|
|
|
|
|
|
|
if (reg == REG_A4) return "a4";
|
|
|
|
|
|
|
|
if (reg == REG_A5) return "a5";
|
|
|
|
|
|
|
|
if (reg == REG_A6) return "a6";
|
|
|
|
|
|
|
|
if (reg == REG_A7) return "a7";
|
|
|
|
|
|
|
|
// allocation begin
|
|
|
|
|
|
|
|
if (reg == REG_S2) return "s2";
|
|
|
|
|
|
|
|
if (reg == REG_S3) return "s3";
|
|
|
|
|
|
|
|
if (reg == REG_S4) return "s4";
|
|
|
|
|
|
|
|
if (reg == REG_S5) return "s5";
|
|
|
|
|
|
|
|
if (reg == REG_S6) return "s6";
|
|
|
|
|
|
|
|
if (reg == REG_S7) return "s7";
|
|
|
|
|
|
|
|
if (reg == REG_S8) return "s8";
|
|
|
|
|
|
|
|
if (reg == REG_S9) return "s9";
|
|
|
|
|
|
|
|
if (reg == REG_S10) return "s10";
|
|
|
|
|
|
|
|
if (reg == REG_S11) return "s11";
|
|
|
|
|
|
|
|
if (reg == REG_T3) return "t3";
|
|
|
|
|
|
|
|
if (reg == REG_T4) return "t4";
|
|
|
|
|
|
|
|
if (reg == REG_T5) return "t5";
|
|
|
|
|
|
|
|
if (reg == REG_T6) return "t6";
|
|
|
|
|
|
|
|
// overflow begin
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// lexer
|
|
|
|
// lexer
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int streq(char* s1, char* s2) {
|
|
|
|
|
|
|
|
while (*s1 && *s2 && *s1 == *s2) {
|
|
|
|
|
|
|
|
s1++;
|
|
|
|
|
|
|
|
s2++;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return *s1 == *s2;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int is_digit(int ch) {
|
|
|
|
int is_digit(int ch) {
|
|
|
|
return '0' <= ch && ch <= '9';
|
|
|
|
return '0' <= ch && ch <= '9';
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -39,87 +219,6 @@ int is_id_cont(int ch) {
|
|
|
|
return is_id_start(ch) || is_digit(ch);
|
|
|
|
return is_id_start(ch) || is_digit(ch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int token_state;
|
|
|
|
|
|
|
|
int token_type;
|
|
|
|
|
|
|
|
int token_data;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int TOKEN_EOF = 0;
|
|
|
|
|
|
|
|
const int TOKEN_SEMICOLON = 1;
|
|
|
|
|
|
|
|
const int TOKEN_ADD = 2;
|
|
|
|
|
|
|
|
const int TOKEN_SUB = 3;
|
|
|
|
|
|
|
|
const int TOKEN_MUL = 4;
|
|
|
|
|
|
|
|
const int TOKEN_DIV = 5;
|
|
|
|
|
|
|
|
const int TOKEN_REM = 6;
|
|
|
|
|
|
|
|
const int TOKEN_ASSIGN = 7;
|
|
|
|
|
|
|
|
const int TOKEN_COMMA = 8;
|
|
|
|
|
|
|
|
const int TOKEN_DOT = 9;
|
|
|
|
|
|
|
|
const int TOKEN_LSHIFT = 10;
|
|
|
|
|
|
|
|
const int TOKEN_RSHIFT = 11;
|
|
|
|
|
|
|
|
const int TOKEN_AND = 12;
|
|
|
|
|
|
|
|
const int TOKEN_OR = 13;
|
|
|
|
|
|
|
|
const int TOKEN_XOR = 14;
|
|
|
|
|
|
|
|
const int TOKEN_COMPL = 15;
|
|
|
|
|
|
|
|
const int TOKEN_NOT = 16;
|
|
|
|
|
|
|
|
const int TOKEN_LAND = 17;
|
|
|
|
|
|
|
|
const int TOKEN_LOR = 18;
|
|
|
|
|
|
|
|
const int TOKEN_ELLIPSIS = 19;
|
|
|
|
|
|
|
|
const int TOKEN_INC = 20;
|
|
|
|
|
|
|
|
const int TOKEN_DEC = 21;
|
|
|
|
|
|
|
|
const int TOKEN_ADD_ASSIGN = 22;
|
|
|
|
|
|
|
|
const int TOKEN_SUB_ASSIGN = 23;
|
|
|
|
|
|
|
|
const int TOKEN_MUL_ASSIGN = 24;
|
|
|
|
|
|
|
|
const int TOKEN_DIV_ASSIGN = 25;
|
|
|
|
|
|
|
|
const int TOKEN_REM_ASSIGN = 26;
|
|
|
|
|
|
|
|
const int TOKEN_AND_ASSIGN = 27;
|
|
|
|
|
|
|
|
const int TOKEN_OR_ASSIGN = 28;
|
|
|
|
|
|
|
|
const int TOKEN_XOR_ASSIGN = 29;
|
|
|
|
|
|
|
|
const int TOKEN_LSHIFT_ASSIGN = 30;
|
|
|
|
|
|
|
|
const int TOKEN_RSHIFT_ASSIGN = 31;
|
|
|
|
|
|
|
|
const int TOKEN_QUESTION = 32;
|
|
|
|
|
|
|
|
const int TOKEN_COLON = 33;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int TOKEN_EQ = 40;
|
|
|
|
|
|
|
|
const int TOKEN_NE = 41;
|
|
|
|
|
|
|
|
const int TOKEN_LT = 42;
|
|
|
|
|
|
|
|
const int TOKEN_GT = 43;
|
|
|
|
|
|
|
|
const int TOKEN_LE = 44;
|
|
|
|
|
|
|
|
const int TOKEN_GE = 45;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int TOKEN_PAREN_LEFT = 50;
|
|
|
|
|
|
|
|
const int TOKEN_PAREN_RIGHT = 51;
|
|
|
|
|
|
|
|
const int TOKEN_BRACKET_LEFT = 52;
|
|
|
|
|
|
|
|
const int TOKEN_BRACKET_RIGHT = 53;
|
|
|
|
|
|
|
|
const int TOKEN_BRACE_LEFT = 54;
|
|
|
|
|
|
|
|
const int TOKEN_BRACE_RIGHT = 55;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int TOKEN_STRING = 99;
|
|
|
|
|
|
|
|
const int TOKEN_NUMBER = 100;
|
|
|
|
|
|
|
|
const int TOKEN_ID = 101;
|
|
|
|
|
|
|
|
const int TOKEN_IF = 102;
|
|
|
|
|
|
|
|
const int TOKEN_ELSE = 103;
|
|
|
|
|
|
|
|
const int TOKEN_WHILE = 104;
|
|
|
|
|
|
|
|
const int TOKEN_FOR = 105;
|
|
|
|
|
|
|
|
const int TOKEN_DO = 106;
|
|
|
|
|
|
|
|
const int TOKEN_BREAK = 107;
|
|
|
|
|
|
|
|
const int TOKEN_CONTINUE = 108;
|
|
|
|
|
|
|
|
const int TOKEN_RETURN = 109;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int TOKEN_EXTERN = 126;
|
|
|
|
|
|
|
|
const int TOKEN_CONST = 127;
|
|
|
|
|
|
|
|
const int TOKEN_VOID = 128;
|
|
|
|
|
|
|
|
const int TOKEN_INT = 129;
|
|
|
|
|
|
|
|
const int TOKEN_CHAR = 130;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int TYPE_VOID = 0;
|
|
|
|
|
|
|
|
const int TYPE_INT = 1;
|
|
|
|
|
|
|
|
const int TYPE_CHAR = 2;
|
|
|
|
|
|
|
|
const int TYPE_VOID_PTR = 16;
|
|
|
|
|
|
|
|
const int TYPE_INT_PTR = 17;
|
|
|
|
|
|
|
|
const int TYPE_CHAR_PTR = 18;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int TYPE_PTR_MASK = 16;
|
|
|
|
|
|
|
|
const int TYPE_TOKEN_MASK = 128;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int parse_int(int ch) {
|
|
|
|
int parse_int(int ch) {
|
|
|
|
int num = ch - '0';
|
|
|
|
int num = ch - '0';
|
|
|
|
while (is_digit(ch = getchar())) {
|
|
|
|
while (is_digit(ch = getchar())) {
|
|
|
@ -153,17 +252,13 @@ int get_escaped_char() {
|
|
|
|
return ch;
|
|
|
|
return ch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int streq(const char* s1, const char* s2) {
|
|
|
|
int token_state;
|
|
|
|
while (*s1 && *s2 && *s1 == *s2) {
|
|
|
|
int token_type;
|
|
|
|
s1++;
|
|
|
|
int token_data;
|
|
|
|
s2++;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return *s1 == *s2;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char string_table[65536];
|
|
|
|
char string_table[STRING_TABLE_SIZE];
|
|
|
|
int string_offset;
|
|
|
|
int string_offset;
|
|
|
|
int string_lut[4096];
|
|
|
|
int string_lut[STRING_LUT_SIZE];
|
|
|
|
int string_lut_size;
|
|
|
|
int string_lut_size;
|
|
|
|
int parse_string() {
|
|
|
|
int parse_string() {
|
|
|
|
int offset = string_offset;
|
|
|
|
int offset = string_offset;
|
|
|
@ -201,9 +296,9 @@ void dedup_string() {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char id_table[65536];
|
|
|
|
char id_table[ID_TABLE_SIZE];
|
|
|
|
int id_offset;
|
|
|
|
int id_offset;
|
|
|
|
int id_lut[4096];
|
|
|
|
int id_lut[ID_LUT_SIZE];
|
|
|
|
int id_lut_size;
|
|
|
|
int id_lut_size;
|
|
|
|
int parse_id(int ch) {
|
|
|
|
int parse_id(int ch) {
|
|
|
|
int offset = id_offset;
|
|
|
|
int offset = id_offset;
|
|
|
@ -255,8 +350,6 @@ void parse_id_like(int ch) {
|
|
|
|
token_type = TOKEN_RETURN;
|
|
|
|
token_type = TOKEN_RETURN;
|
|
|
|
} else if (streq(id, "void")) {
|
|
|
|
} else if (streq(id, "void")) {
|
|
|
|
token_type = TOKEN_VOID;
|
|
|
|
token_type = TOKEN_VOID;
|
|
|
|
} else if (streq(id, "const")) {
|
|
|
|
|
|
|
|
token_type = TOKEN_CONST;
|
|
|
|
|
|
|
|
} else if (streq(id, "char")) {
|
|
|
|
} else if (streq(id, "char")) {
|
|
|
|
token_type = TOKEN_CHAR;
|
|
|
|
token_type = TOKEN_CHAR;
|
|
|
|
} else if (streq(id, "for")) {
|
|
|
|
} else if (streq(id, "for")) {
|
|
|
@ -265,6 +358,8 @@ void parse_id_like(int ch) {
|
|
|
|
token_type = TOKEN_DO;
|
|
|
|
token_type = TOKEN_DO;
|
|
|
|
} else if (streq(id, "extern")) {
|
|
|
|
} else if (streq(id, "extern")) {
|
|
|
|
token_type = TOKEN_EXTERN;
|
|
|
|
token_type = TOKEN_EXTERN;
|
|
|
|
|
|
|
|
} else if (streq(id, "enum")) {
|
|
|
|
|
|
|
|
token_type = TOKEN_ENUM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (token_type != TOKEN_ID) {
|
|
|
|
if (token_type != TOKEN_ID) {
|
|
|
|
rewind_id(0);
|
|
|
|
rewind_id(0);
|
|
|
@ -488,7 +583,7 @@ void next_token() {
|
|
|
|
if (0) {
|
|
|
|
if (0) {
|
|
|
|
fprintf(stderr, "token: %d\n", token_type);
|
|
|
|
fprintf(stderr, "token: %d\n", token_type);
|
|
|
|
if (token_type == TOKEN_ID) {
|
|
|
|
if (token_type == TOKEN_ID) {
|
|
|
|
const char* name = id_table + id_lut[token_data];
|
|
|
|
char* name = id_table + id_lut[token_data];
|
|
|
|
fprintf(stderr, " id: %s\n", name);
|
|
|
|
fprintf(stderr, " id: %s\n", name);
|
|
|
|
} else if (token_type == TOKEN_NUMBER) {
|
|
|
|
} else if (token_type == TOKEN_NUMBER) {
|
|
|
|
fprintf(stderr, " number: %d\n", token_data);
|
|
|
|
fprintf(stderr, " number: %d\n", token_data);
|
|
|
@ -504,21 +599,12 @@ void expect_token(int expected_type) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ignore_const() {
|
|
|
|
|
|
|
|
if (token_type == TOKEN_CONST) {
|
|
|
|
|
|
|
|
next_token();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int parse_type() {
|
|
|
|
int parse_type() {
|
|
|
|
ignore_const();
|
|
|
|
|
|
|
|
if (token_type == TOKEN_INT || token_type == TOKEN_CHAR || token_type == TOKEN_VOID) {
|
|
|
|
if (token_type == TOKEN_INT || token_type == TOKEN_CHAR || token_type == TOKEN_VOID) {
|
|
|
|
int type = token_type & ~TYPE_TOKEN_MASK;
|
|
|
|
int type = token_type & ~TYPE_TOKEN_MASK;
|
|
|
|
next_token();
|
|
|
|
next_token();
|
|
|
|
ignore_const();
|
|
|
|
|
|
|
|
if (token_type == TOKEN_MUL) {
|
|
|
|
if (token_type == TOKEN_MUL) {
|
|
|
|
next_token();
|
|
|
|
next_token();
|
|
|
|
ignore_const();
|
|
|
|
|
|
|
|
type |= TYPE_PTR_MASK;
|
|
|
|
type |= TYPE_PTR_MASK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
unget_token();
|
|
|
|
unget_token();
|
|
|
@ -527,67 +613,47 @@ int parse_type() {
|
|
|
|
return -1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// asm
|
|
|
|
// assembly context
|
|
|
|
|
|
|
|
|
|
|
|
int local_table[4096]; // id -> local id
|
|
|
|
// use id as index
|
|
|
|
|
|
|
|
int local_table[ID_LUT_SIZE]; // id -> local id
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// use local id as index
|
|
|
|
int next_local_id = 1;
|
|
|
|
int next_local_id = 1;
|
|
|
|
int max_local_id = 1;
|
|
|
|
int max_local_id = 1;
|
|
|
|
|
|
|
|
int local_kind[LOCAL_SIZE];
|
|
|
|
|
|
|
|
int local_type[LOCAL_SIZE];
|
|
|
|
|
|
|
|
|
|
|
|
const int MARKER_TEMP = 0;
|
|
|
|
// use id as index
|
|
|
|
const int MARKER_SCALAR = 1;
|
|
|
|
int global_kind[ID_LUT_SIZE];
|
|
|
|
const int MARKER_ARRAY = 2;
|
|
|
|
int global_type[ID_LUT_SIZE];
|
|
|
|
const int MARKER_FUNCTION = 3;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int local_marker[4096];
|
|
|
|
// use reg id as index
|
|
|
|
int global_marker[4096];
|
|
|
|
int next_reg_id = REG_S2;
|
|
|
|
int local_type[4096];
|
|
|
|
int max_reg_id = REG_S2;
|
|
|
|
int global_type[4096];
|
|
|
|
int reg_type[REG_SIZE];
|
|
|
|
|
|
|
|
char indirection[REG_SIZE];
|
|
|
|
|
|
|
|
int overflow[REG_SIZE]; // reg -> local id
|
|
|
|
|
|
|
|
|
|
|
|
int reg_type[4096];
|
|
|
|
// use id as index
|
|
|
|
int next_reg_id = 18;
|
|
|
|
int const_table[ID_LUT_SIZE]; // id -> value
|
|
|
|
int max_reg_id = 18;
|
|
|
|
char is_const[ID_LUT_SIZE];
|
|
|
|
int indirection[4096];
|
|
|
|
|
|
|
|
int overflow[4096];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int const_table[4096]; // id -> value
|
|
|
|
int expect_const() {
|
|
|
|
int is_const[4096];
|
|
|
|
next_token();
|
|
|
|
|
|
|
|
if (token_type == TOKEN_NUMBER) {
|
|
|
|
const int REG_ZERO = 0;
|
|
|
|
return token_data;
|
|
|
|
const int REG_RA = 1;
|
|
|
|
}
|
|
|
|
const int REG_SP = 2;
|
|
|
|
if (token_type == TOKEN_ID && !local_table[token_data] && is_const[token_data]) {
|
|
|
|
const int REG_GP = 3;
|
|
|
|
return const_table[token_data];
|
|
|
|
const int REG_TP = 4;
|
|
|
|
}
|
|
|
|
const int REG_T0 = 5;
|
|
|
|
fprintf(stderr, "expecting a constant\n");
|
|
|
|
const int REG_T1 = 6;
|
|
|
|
exit(1);
|
|
|
|
const int REG_T2 = 7;
|
|
|
|
}
|
|
|
|
const int REG_FP = 8;
|
|
|
|
|
|
|
|
const int REG_S1 = 9;
|
|
|
|
|
|
|
|
const int REG_A0 = 10;
|
|
|
|
|
|
|
|
const int REG_A1 = 11;
|
|
|
|
|
|
|
|
const int REG_A2 = 12;
|
|
|
|
|
|
|
|
const int REG_A3 = 13;
|
|
|
|
|
|
|
|
const int REG_A4 = 14;
|
|
|
|
|
|
|
|
const int REG_A5 = 15;
|
|
|
|
|
|
|
|
const int REG_A6 = 16;
|
|
|
|
|
|
|
|
const int REG_A7 = 17;
|
|
|
|
|
|
|
|
const int REG_S2 = 18;
|
|
|
|
|
|
|
|
const int REG_S3 = 19;
|
|
|
|
|
|
|
|
const int REG_S4 = 20;
|
|
|
|
|
|
|
|
const int REG_S5 = 21;
|
|
|
|
|
|
|
|
const int REG_S6 = 22;
|
|
|
|
|
|
|
|
const int REG_S7 = 23;
|
|
|
|
|
|
|
|
const int REG_S8 = 24;
|
|
|
|
|
|
|
|
const int REG_S9 = 25;
|
|
|
|
|
|
|
|
const int REG_S10 = 26;
|
|
|
|
|
|
|
|
const int REG_S11 = 27;
|
|
|
|
|
|
|
|
const int REG_T3 = 28;
|
|
|
|
|
|
|
|
const int REG_T4 = 29;
|
|
|
|
|
|
|
|
const int REG_T5 = 30;
|
|
|
|
|
|
|
|
const int REG_T6 = 31;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void reset_reg() {
|
|
|
|
void reset_reg() {
|
|
|
|
next_reg_id = REG_S2;
|
|
|
|
next_reg_id = REG_S2;
|
|
|
|
for (int i = 0; i < 4096; ++i) {
|
|
|
|
for (int i = 0; i < REG_SIZE; ++i) {
|
|
|
|
reg_type[i] = TYPE_VOID;
|
|
|
|
reg_type[i] = TYPE_VOID;
|
|
|
|
indirection[i] = 0;
|
|
|
|
indirection[i] = 0;
|
|
|
|
overflow[i] = 0;
|
|
|
|
overflow[i] = 0;
|
|
|
@ -595,63 +661,25 @@ void reset_reg() {
|
|
|
|
reg_type[REG_ZERO] = TYPE_INT;
|
|
|
|
reg_type[REG_ZERO] = TYPE_INT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const char* reg_name(int reg) {
|
|
|
|
void reset_local_table() {
|
|
|
|
if (reg == 0) return "zero";
|
|
|
|
for (int i = 0; i < ID_LUT_SIZE; ++i) {
|
|
|
|
if (reg == 1) return "ra";
|
|
|
|
local_table[i] = 0;
|
|
|
|
if (reg == 2) return "sp";
|
|
|
|
}
|
|
|
|
if (reg == 3) return "gp";
|
|
|
|
|
|
|
|
if (reg == 4) return "tp";
|
|
|
|
|
|
|
|
if (reg == 5) return "t0";
|
|
|
|
|
|
|
|
if (reg == 6) return "t1";
|
|
|
|
|
|
|
|
if (reg == 7) return "t2";
|
|
|
|
|
|
|
|
if (reg == 8) return "fp";
|
|
|
|
|
|
|
|
// reserved begin
|
|
|
|
|
|
|
|
if (reg == 9) return "s1";
|
|
|
|
|
|
|
|
if (reg == 10) return "a0";
|
|
|
|
|
|
|
|
if (reg == 11) return "a1";
|
|
|
|
|
|
|
|
if (reg == 12) return "a2";
|
|
|
|
|
|
|
|
if (reg == 13) return "a3";
|
|
|
|
|
|
|
|
if (reg == 14) return "a4";
|
|
|
|
|
|
|
|
if (reg == 15) return "a5";
|
|
|
|
|
|
|
|
if (reg == 16) return "a6";
|
|
|
|
|
|
|
|
if (reg == 17) return "a7";
|
|
|
|
|
|
|
|
// allocation begin
|
|
|
|
|
|
|
|
if (reg == 18) return "s2";
|
|
|
|
|
|
|
|
if (reg == 19) return "s3";
|
|
|
|
|
|
|
|
if (reg == 20) return "s4";
|
|
|
|
|
|
|
|
if (reg == 21) return "s5";
|
|
|
|
|
|
|
|
if (reg == 22) return "s6";
|
|
|
|
|
|
|
|
if (reg == 23) return "s7";
|
|
|
|
|
|
|
|
if (reg == 24) return "s8";
|
|
|
|
|
|
|
|
if (reg == 25) return "s9";
|
|
|
|
|
|
|
|
if (reg == 26) return "s10";
|
|
|
|
|
|
|
|
if (reg == 27) return "s11";
|
|
|
|
|
|
|
|
if (reg == 28) return "t3";
|
|
|
|
|
|
|
|
if (reg == 29) return "t4";
|
|
|
|
|
|
|
|
if (reg == 30) return "t5";
|
|
|
|
|
|
|
|
if (reg == 31) return "t6";
|
|
|
|
|
|
|
|
// overflow begin
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int is_overflow(int reg) {
|
|
|
|
|
|
|
|
return reg > REG_T6;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void reset_local() {
|
|
|
|
void reset_local() {
|
|
|
|
next_local_id = 1;
|
|
|
|
next_local_id = 1;
|
|
|
|
max_local_id = 1;
|
|
|
|
max_local_id = 1;
|
|
|
|
max_reg_id = REG_S2;
|
|
|
|
max_reg_id = REG_S2;
|
|
|
|
for (int i = 0; i < 4096; ++i) {
|
|
|
|
for (int i = 0; i < LOCAL_SIZE; ++i) {
|
|
|
|
local_table[i] = 0;
|
|
|
|
local_kind[i] = KIND_TEMP;
|
|
|
|
local_marker[i] = MARKER_TEMP;
|
|
|
|
|
|
|
|
local_type[i] = TYPE_VOID;
|
|
|
|
local_type[i] = TYPE_VOID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
reset_reg();
|
|
|
|
reset_reg();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void reset_temp() {
|
|
|
|
void reset_temp() {
|
|
|
|
while (next_local_id > 1 && local_marker[next_local_id - 1] == MARKER_TEMP) {
|
|
|
|
while (next_local_id > 1 && local_kind[next_local_id - 1] == KIND_TEMP) {
|
|
|
|
--next_local_id;
|
|
|
|
--next_local_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
reset_reg();
|
|
|
|
reset_reg();
|
|
|
@ -669,7 +697,7 @@ int next_local_slot(int type) {
|
|
|
|
int declare_local(int id, int type) {
|
|
|
|
int declare_local(int id, int type) {
|
|
|
|
if (local_table[id] != 0) return local_table[id];
|
|
|
|
if (local_table[id] != 0) return local_table[id];
|
|
|
|
int slot = next_local_slot(type);
|
|
|
|
int slot = next_local_slot(type);
|
|
|
|
local_marker[slot] = MARKER_SCALAR;
|
|
|
|
local_kind[slot] = KIND_SCALAR;
|
|
|
|
return local_table[id] = slot;
|
|
|
|
return local_table[id] = slot;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -682,21 +710,25 @@ int declare_local_array(int id, int type, int size) {
|
|
|
|
int slot = next_local_slot(type);
|
|
|
|
int slot = next_local_slot(type);
|
|
|
|
int array_size = array_size_of(type, size);
|
|
|
|
int array_size = array_size_of(type, size);
|
|
|
|
int slot_size = (array_size + 7) / 8;
|
|
|
|
int slot_size = (array_size + 7) / 8;
|
|
|
|
local_marker[slot] = MARKER_ARRAY;
|
|
|
|
local_kind[slot] = KIND_ARRAY;
|
|
|
|
for (int i = 1; i < slot_size; ++i) local_marker[next_local_slot(type)] = MARKER_ARRAY;
|
|
|
|
for (int i = 1; i < slot_size; ++i) local_kind[next_local_slot(type)] = KIND_ARRAY;
|
|
|
|
return local_table[id] = slot;
|
|
|
|
return local_table[id] = slot;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void declare_global(int id, int marker, int type) {
|
|
|
|
void declare_global(int id, int kind, int type) {
|
|
|
|
global_marker[id] = marker;
|
|
|
|
global_kind[id] = kind;
|
|
|
|
global_type[id] = type;
|
|
|
|
global_type[id] = type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int is_overflow(int reg) {
|
|
|
|
|
|
|
|
return reg > REG_T6;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int next_reg(int type) {
|
|
|
|
int next_reg(int type) {
|
|
|
|
int reg = next_reg_id++;
|
|
|
|
int reg = next_reg_id++;
|
|
|
|
if (is_overflow(reg)) {
|
|
|
|
if (is_overflow(reg)) {
|
|
|
|
int slot = next_local_slot(type);
|
|
|
|
int slot = next_local_slot(type);
|
|
|
|
local_marker[slot] = MARKER_TEMP;
|
|
|
|
local_kind[slot] = KIND_TEMP;
|
|
|
|
overflow[reg] = slot;
|
|
|
|
overflow[reg] = slot;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
reg_type[reg] = type;
|
|
|
|
reg_type[reg] = type;
|
|
|
@ -713,7 +745,7 @@ int check_itype_immediate(int value) {
|
|
|
|
return value >= -2048 && value <= 2047;
|
|
|
|
return value >= -2048 && value <= 2047;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void asm_ld(const char* rd, int imm, const char* rs) {
|
|
|
|
void asm_ld(char* rd, int imm, char* rs) {
|
|
|
|
if (check_itype_immediate(imm)) {
|
|
|
|
if (check_itype_immediate(imm)) {
|
|
|
|
printf(" ld %s, %d(%s)\n", rd, imm, rs);
|
|
|
|
printf(" ld %s, %d(%s)\n", rd, imm, rs);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
@ -723,7 +755,7 @@ void asm_ld(const char* rd, int imm, const char* rs) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void asm_sd(const char* rs1, int imm, const char* rs2) {
|
|
|
|
void asm_sd(char* rs1, int imm, char* rs2) {
|
|
|
|
if (check_itype_immediate(imm)) {
|
|
|
|
if (check_itype_immediate(imm)) {
|
|
|
|
printf(" sd %s, %d(%s)\n", rs1, imm, rs2);
|
|
|
|
printf(" sd %s, %d(%s)\n", rs1, imm, rs2);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
@ -733,7 +765,7 @@ void asm_sd(const char* rs1, int imm, const char* rs2) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void asm_addi(const char* rd, const char* rs, int imm) {
|
|
|
|
void asm_addi(char* rd, char* rs, int imm) {
|
|
|
|
if (check_itype_immediate(imm)) {
|
|
|
|
if (check_itype_immediate(imm)) {
|
|
|
|
printf(" addi %s, %s, %d\n", rd, rs, imm);
|
|
|
|
printf(" addi %s, %s, %d\n", rd, rs, imm);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
@ -744,7 +776,7 @@ void asm_addi(const char* rd, const char* rs, int imm) {
|
|
|
|
|
|
|
|
|
|
|
|
// assembly helpers
|
|
|
|
// assembly helpers
|
|
|
|
|
|
|
|
|
|
|
|
const char* load_op_of_type(int type) {
|
|
|
|
char* load_op_of_type(int type) {
|
|
|
|
if (type & TYPE_PTR_MASK) {
|
|
|
|
if (type & TYPE_PTR_MASK) {
|
|
|
|
return "ld";
|
|
|
|
return "ld";
|
|
|
|
} else if (type == TYPE_CHAR) {
|
|
|
|
} else if (type == TYPE_CHAR) {
|
|
|
@ -754,7 +786,7 @@ const char* load_op_of_type(int type) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const char* store_op_of_type(int type) {
|
|
|
|
char* store_op_of_type(int type) {
|
|
|
|
if (type & TYPE_PTR_MASK) {
|
|
|
|
if (type & TYPE_PTR_MASK) {
|
|
|
|
return "sd";
|
|
|
|
return "sd";
|
|
|
|
} else if (type == TYPE_CHAR) {
|
|
|
|
} else if (type == TYPE_CHAR) {
|
|
|
@ -772,8 +804,8 @@ void load_local_address(int rd, int slot_id) {
|
|
|
|
|
|
|
|
|
|
|
|
// load a non-trivial register into trivial one
|
|
|
|
// load a non-trivial register into trivial one
|
|
|
|
void load(int rd, int rs) {
|
|
|
|
void load(int rd, int rs) {
|
|
|
|
const char* op = load_op_of_type(reg_type[rs]);
|
|
|
|
char* op = load_op_of_type(reg_type[rs]);
|
|
|
|
const char* rd_name = reg_name(rd);
|
|
|
|
char* rd_name = reg_name(rd);
|
|
|
|
if (is_overflow(rs)) {
|
|
|
|
if (is_overflow(rs)) {
|
|
|
|
load_local_address(rd, overflow[rs]);
|
|
|
|
load_local_address(rd, overflow[rs]);
|
|
|
|
if (indirection[rs]) {
|
|
|
|
if (indirection[rs]) {
|
|
|
@ -785,8 +817,8 @@ void load(int rd, int rs) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// store a trivial register into a non-trivial one
|
|
|
|
// store a trivial register into a non-trivial one
|
|
|
|
void store(const char* rs, int reg) {
|
|
|
|
void store(char* rs, int reg) {
|
|
|
|
const char* op = store_op_of_type(reg_type[reg]);
|
|
|
|
char* op = store_op_of_type(reg_type[reg]);
|
|
|
|
if (is_overflow(reg)) {
|
|
|
|
if (is_overflow(reg)) {
|
|
|
|
load_local_address(REG_T2, overflow[reg]);
|
|
|
|
load_local_address(REG_T2, overflow[reg]);
|
|
|
|
if (indirection[reg]) {
|
|
|
|
if (indirection[reg]) {
|
|
|
@ -801,7 +833,7 @@ int is_nontrivial(int reg) {
|
|
|
|
return is_overflow(reg) || indirection[reg];
|
|
|
|
return is_overflow(reg) || indirection[reg];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const char* trivialize(int rs, int t) {
|
|
|
|
char* trivialize(int rs, int t) {
|
|
|
|
if (is_nontrivial(rs)) {
|
|
|
|
if (is_nontrivial(rs)) {
|
|
|
|
load(t, rs);
|
|
|
|
load(t, rs);
|
|
|
|
return reg_name(t);
|
|
|
|
return reg_name(t);
|
|
|
@ -809,10 +841,10 @@ const char* trivialize(int rs, int t) {
|
|
|
|
return reg_name(rs);
|
|
|
|
return reg_name(rs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _asm_r(const char* op, int rd, int rs1) {
|
|
|
|
void _asm_r(char* op, int rd, int rs1) {
|
|
|
|
const char* rd_name = reg_name(rd);
|
|
|
|
char* rd_name = reg_name(rd);
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
const char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
printf(" %s %s, %s\n", op, rd_name, rs1_name);
|
|
|
|
printf(" %s %s, %s\n", op, rd_name, rs1_name);
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
store("t0", rd);
|
|
|
|
store("t0", rd);
|
|
|
@ -820,20 +852,20 @@ void _asm_r(const char* op, int rd, int rs1) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void asm_mv(int rd, int rs1) {
|
|
|
|
void asm_mv(int rd, int rs1) {
|
|
|
|
const char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
store(rs1_name, rd);
|
|
|
|
store(rs1_name, rd);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
const char* rd_name = reg_name(rd);
|
|
|
|
char* rd_name = reg_name(rd);
|
|
|
|
if (!streq(rd_name, rs1_name))
|
|
|
|
if (!streq(rd_name, rs1_name))
|
|
|
|
printf(" mv %s, %s\n", rd_name, rs1_name);
|
|
|
|
printf(" mv %s, %s\n", rd_name, rs1_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _asm_rr(const char* op, int rd, int rs1, int rs2) {
|
|
|
|
void _asm_rr(char* op, int rd, int rs1, int rs2) {
|
|
|
|
const char* rd_name = reg_name(rd);
|
|
|
|
char* rd_name = reg_name(rd);
|
|
|
|
const char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
const char* rs2_name = trivialize(rs2, REG_T1);
|
|
|
|
char* rs2_name = trivialize(rs2, REG_T1);
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
printf(" %s %s, %s, %s\n", op, rd_name, rs1_name, rs2_name);
|
|
|
|
printf(" %s %s, %s, %s\n", op, rd_name, rs1_name, rs2_name);
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
@ -841,23 +873,23 @@ void _asm_rr(const char* op, int rd, int rs1, int rs2) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _asm_ri(const char* op, int rd, int rs1, int imm) {
|
|
|
|
void _asm_ri(char* op, int rd, int rs1, int imm) {
|
|
|
|
const char* rd_name = reg_name(rd);
|
|
|
|
char* rd_name = reg_name(rd);
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
const char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
printf(" %s %s, %s, %d\n", op, rd_name, rs1_name, imm);
|
|
|
|
printf(" %s %s, %s, %d\n", op, rd_name, rs1_name, imm);
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
store("t0", rd);
|
|
|
|
store("t0", rd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void asm_branch(const char* op, int rs1, int label) {
|
|
|
|
void asm_branch(char* op, int rs1, int label) {
|
|
|
|
const char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
printf(" %s %s, L%d\n", op, rs1_name, label);
|
|
|
|
printf(" %s %s, L%d\n", op, rs1_name, label);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _asm_i(const char* op, int rd, const char* prefix1, const char* prefix2, int imm) {
|
|
|
|
void _asm_i(char* op, int rd, char* prefix1, char* prefix2, int imm) {
|
|
|
|
const char* rd_name = reg_name(rd);
|
|
|
|
char* rd_name = reg_name(rd);
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
if (is_nontrivial(rd)) rd_name = "t0";
|
|
|
|
printf(" %s %s, %s%s%d\n", op, rd_name, prefix1, prefix2, imm);
|
|
|
|
printf(" %s %s, %s%s%d\n", op, rd_name, prefix1, prefix2, imm);
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
|
if (is_nontrivial(rd)) {
|
|
|
@ -869,14 +901,14 @@ int is_not_reusable(int rs1, int expected_type) {
|
|
|
|
return indirection[rs1] || reg_type[rs1] != expected_type || rs1 == REG_ZERO;
|
|
|
|
return indirection[rs1] || reg_type[rs1] != expected_type || rs1 == REG_ZERO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int asm_r(int type, const char* op, int rs1) {
|
|
|
|
int asm_r(int type, char* op, int rs1) {
|
|
|
|
int rd = rs1;
|
|
|
|
int rd = rs1;
|
|
|
|
if (is_not_reusable(rs1, type)) rd = next_reg(type);
|
|
|
|
if (is_not_reusable(rs1, type)) rd = next_reg(type);
|
|
|
|
_asm_r(op, rd, rs1);
|
|
|
|
_asm_r(op, rd, rs1);
|
|
|
|
return rd;
|
|
|
|
return rd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int asm_rr(int type, const char* op, int rs1, int rs2) {
|
|
|
|
int asm_rr(int type, char* op, int rs1, int rs2) {
|
|
|
|
int rd = rs1;
|
|
|
|
int rd = rs1;
|
|
|
|
if (is_not_reusable(rs1, type)) rd = rs2;
|
|
|
|
if (is_not_reusable(rs1, type)) rd = rs2;
|
|
|
|
if (is_not_reusable(rs2, type)) rd = next_reg(type);
|
|
|
|
if (is_not_reusable(rs2, type)) rd = next_reg(type);
|
|
|
@ -885,17 +917,17 @@ int asm_rr(int type, const char* op, int rs1, int rs2) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void store_into_local(int rs1, int slot) {
|
|
|
|
void store_into_local(int rs1, int slot) {
|
|
|
|
const char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
char* rs1_name = trivialize(rs1, REG_T0);
|
|
|
|
load_local_address(REG_T2, slot);
|
|
|
|
load_local_address(REG_T2, slot);
|
|
|
|
printf(" %s %s, 0(t2)\n", store_op_of_type(local_type[slot]), rs1_name);
|
|
|
|
printf(" %s %s, 0(t2)\n", store_op_of_type(local_type[slot]), rs1_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int materialize_address(int rd, int type, int marker) {
|
|
|
|
int materialize_address(int rd, int type, int kind) {
|
|
|
|
if (marker == MARKER_ARRAY) {
|
|
|
|
if (kind == KIND_ARRAY) {
|
|
|
|
type |= TYPE_PTR_MASK;
|
|
|
|
type |= TYPE_PTR_MASK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
reg_type[rd] = type;
|
|
|
|
reg_type[rd] = type;
|
|
|
|
indirection[rd] = marker == MARKER_SCALAR;
|
|
|
|
indirection[rd] = kind == KIND_SCALAR;
|
|
|
|
return rd;
|
|
|
|
return rd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -907,7 +939,7 @@ int lookup_from_slot(int slot) {
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
load_local_address(rd, slot);
|
|
|
|
load_local_address(rd, slot);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return materialize_address(rd, local_type[slot], local_marker[slot]);
|
|
|
|
return materialize_address(rd, local_type[slot], local_kind[slot]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int load_imm(int imm) {
|
|
|
|
int load_imm(int imm) {
|
|
|
@ -924,21 +956,21 @@ int lookup(int id) {
|
|
|
|
if (is_const[id]) {
|
|
|
|
if (is_const[id]) {
|
|
|
|
return load_imm(const_table[id]);
|
|
|
|
return load_imm(const_table[id]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const char* name = id_table + id_lut[id];
|
|
|
|
char* name = id_table + id_lut[id];
|
|
|
|
if (global_marker[id]) {
|
|
|
|
if (global_kind[id]) {
|
|
|
|
if (global_marker[id] == MARKER_FUNCTION) {
|
|
|
|
if (global_kind[id] == KIND_FUNCTION) {
|
|
|
|
fprintf(stderr, "function name must not appear outside function call: %s\n", name);
|
|
|
|
fprintf(stderr, "function name must not appear outside function call: %s\n", name);
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int rd = next_reg(TYPE_VOID_PTR);
|
|
|
|
int rd = next_reg(TYPE_VOID_PTR);
|
|
|
|
_asm_i("la", rd, name, " # id: ", id);
|
|
|
|
_asm_i("la", rd, name, " # id: ", id);
|
|
|
|
return materialize_address(rd, global_type[id], global_marker[id]);
|
|
|
|
return materialize_address(rd, global_type[id], global_kind[id]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr, "unresolved identifier: %s\n", name);
|
|
|
|
fprintf(stderr, "unresolved identifier: %s\n", name);
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int asm_r_arith(const char* op, int rs1) {
|
|
|
|
int asm_r_arith(char* op, int rs1) {
|
|
|
|
if (reg_type[rs1] & TYPE_PTR_MASK) {
|
|
|
|
if (reg_type[rs1] & TYPE_PTR_MASK) {
|
|
|
|
fprintf(stderr, "pointer cannot be arithmetically operated by %s\n", op);
|
|
|
|
fprintf(stderr, "pointer cannot be arithmetically operated by %s\n", op);
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
@ -946,7 +978,7 @@ int asm_r_arith(const char* op, int rs1) {
|
|
|
|
return asm_r(TYPE_INT, op, rs1);
|
|
|
|
return asm_r(TYPE_INT, op, rs1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int asm_rr_arith(const char* op, int rs1, int rs2) {
|
|
|
|
int asm_rr_arith(char* op, int rs1, int rs2) {
|
|
|
|
if (reg_type[rs1] & TYPE_PTR_MASK || reg_type[rs2] & TYPE_PTR_MASK) {
|
|
|
|
if (reg_type[rs1] & TYPE_PTR_MASK || reg_type[rs2] & TYPE_PTR_MASK) {
|
|
|
|
fprintf(stderr, "pointer cannot be arithmetically operated by %s\n", op);
|
|
|
|
fprintf(stderr, "pointer cannot be arithmetically operated by %s\n", op);
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
@ -954,7 +986,7 @@ int asm_rr_arith(const char* op, int rs1, int rs2) {
|
|
|
|
return asm_rr(TYPE_INT, op, rs1, rs2);
|
|
|
|
return asm_rr(TYPE_INT, op, rs1, rs2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int asm_rr_cmp(const char* op, int rs1, int rs2) {
|
|
|
|
int asm_rr_cmp(char* op, int rs1, int rs2) {
|
|
|
|
// since NULL is virtually 0, it is considered a valid example of a pointer comparing with an integer
|
|
|
|
// since NULL is virtually 0, it is considered a valid example of a pointer comparing with an integer
|
|
|
|
return asm_rr(TYPE_INT, op, rs1, rs2);
|
|
|
|
return asm_rr(TYPE_INT, op, rs1, rs2);
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -1107,8 +1139,8 @@ int parse_expr();
|
|
|
|
int parse_assign_expr();
|
|
|
|
int parse_assign_expr();
|
|
|
|
|
|
|
|
|
|
|
|
int parse_function_call(int id) {
|
|
|
|
int parse_function_call(int id) {
|
|
|
|
const char* name = id_table + id_lut[id];
|
|
|
|
char* name = id_table + id_lut[id];
|
|
|
|
if (global_marker[id] != MARKER_FUNCTION) {
|
|
|
|
if (global_kind[id] != KIND_FUNCTION) {
|
|
|
|
fprintf(stderr, "not a function name: %s\n", name);
|
|
|
|
fprintf(stderr, "not a function name: %s\n", name);
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -1581,8 +1613,7 @@ void parse_local_variable(int type) {
|
|
|
|
fprintf(stderr, "array of pointers is not supported\n");
|
|
|
|
fprintf(stderr, "array of pointers is not supported\n");
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
expect_token(TOKEN_NUMBER);
|
|
|
|
int size = expect_const();
|
|
|
|
int size = token_data;
|
|
|
|
|
|
|
|
expect_token(TOKEN_BRACKET_RIGHT);
|
|
|
|
expect_token(TOKEN_BRACKET_RIGHT);
|
|
|
|
declare_local_array(id, type, size);
|
|
|
|
declare_local_array(id, type, size);
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -1731,7 +1762,7 @@ void parse_stmt() {
|
|
|
|
reset_temp();
|
|
|
|
reset_temp();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void parse_function(const char* name) {
|
|
|
|
void parse_function(char* name) {
|
|
|
|
reset_local();
|
|
|
|
reset_local();
|
|
|
|
int arg = 0;
|
|
|
|
int arg = 0;
|
|
|
|
int args[8];
|
|
|
|
int args[8];
|
|
|
@ -1777,6 +1808,7 @@ void parse_function(const char* name) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
next_token();
|
|
|
|
next_token();
|
|
|
|
if (token_type == TOKEN_SEMICOLON) {
|
|
|
|
if (token_type == TOKEN_SEMICOLON) {
|
|
|
|
|
|
|
|
reset_local_table();
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
unget_token();
|
|
|
|
unget_token();
|
|
|
@ -1832,27 +1864,26 @@ void parse_function(const char* name) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
asm_addi("sp", "sp", frame_size);
|
|
|
|
asm_addi("sp", "sp", frame_size);
|
|
|
|
printf(" ret\n");
|
|
|
|
printf(" ret\n");
|
|
|
|
|
|
|
|
reset_local_table();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void parse_global_variable(int id, const char* name, int type) {
|
|
|
|
void parse_global_variable(int id, char* name, int type) {
|
|
|
|
printf(".data\n");
|
|
|
|
printf(".data\n");
|
|
|
|
printf(".globl %s\n", name);
|
|
|
|
printf(".globl %s\n", name);
|
|
|
|
printf(".align 5\n");
|
|
|
|
printf(".align 5\n");
|
|
|
|
printf("%s:\n", name);
|
|
|
|
printf("%s:\n", name);
|
|
|
|
if (token_type == TOKEN_ASSIGN) {
|
|
|
|
if (token_type == TOKEN_ASSIGN) {
|
|
|
|
expect_token(TOKEN_NUMBER);
|
|
|
|
printf(" .dword %d\n", expect_const());
|
|
|
|
printf(" .dword %d\n", token_data);
|
|
|
|
|
|
|
|
} else if (token_type == TOKEN_BRACKET_LEFT) {
|
|
|
|
} else if (token_type == TOKEN_BRACKET_LEFT) {
|
|
|
|
if (type & TYPE_PTR_MASK) {
|
|
|
|
if (type & TYPE_PTR_MASK) {
|
|
|
|
fprintf(stderr, "array of pointers is not supported\n");
|
|
|
|
fprintf(stderr, "array of pointers is not supported\n");
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
expect_token(TOKEN_NUMBER);
|
|
|
|
int size = expect_const();
|
|
|
|
int size = token_data;
|
|
|
|
|
|
|
|
expect_token(TOKEN_BRACKET_RIGHT);
|
|
|
|
expect_token(TOKEN_BRACKET_RIGHT);
|
|
|
|
int array_size = array_size_of(type, size);
|
|
|
|
int array_size = array_size_of(type, size);
|
|
|
|
printf(" .zero %d\n", array_size);
|
|
|
|
printf(" .zero %d\n", array_size);
|
|
|
|
declare_global(id, MARKER_ARRAY, type);
|
|
|
|
declare_global(id, KIND_ARRAY, type);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
printf(" .zero %d\n", 8);
|
|
|
|
printf(" .zero %d\n", 8);
|
|
|
|
unget_token();
|
|
|
|
unget_token();
|
|
|
@ -1865,36 +1896,24 @@ void parse_global_declaration() {
|
|
|
|
external = 1;
|
|
|
|
external = 1;
|
|
|
|
next_token();
|
|
|
|
next_token();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int is_const_int = 1;
|
|
|
|
|
|
|
|
if (token_type != TOKEN_CONST) {
|
|
|
|
|
|
|
|
is_const_int = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int type = parse_type();
|
|
|
|
int type = parse_type();
|
|
|
|
if (type < 0) {
|
|
|
|
if (type < 0) {
|
|
|
|
fprintf(stderr, "expecting type for global declaration\n");
|
|
|
|
fprintf(stderr, "expecting type for global declaration\n");
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (type != TYPE_INT) {
|
|
|
|
|
|
|
|
is_const_int = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
expect_token(TOKEN_ID);
|
|
|
|
expect_token(TOKEN_ID);
|
|
|
|
int id = token_data;
|
|
|
|
int id = token_data;
|
|
|
|
char* name = id_table + id_lut[id];
|
|
|
|
char* name = id_table + id_lut[id];
|
|
|
|
next_token();
|
|
|
|
next_token();
|
|
|
|
if (!external && is_const_int && token_type == TOKEN_ASSIGN) {
|
|
|
|
if (token_type == TOKEN_PAREN_LEFT) {
|
|
|
|
expect_token(TOKEN_NUMBER);
|
|
|
|
declare_global(id, KIND_FUNCTION, type);
|
|
|
|
const_table[id] = token_data;
|
|
|
|
|
|
|
|
is_const[id] = 1;
|
|
|
|
|
|
|
|
expect_token(TOKEN_SEMICOLON);
|
|
|
|
|
|
|
|
} else if (token_type == TOKEN_PAREN_LEFT) {
|
|
|
|
|
|
|
|
declare_global(id, MARKER_FUNCTION, type);
|
|
|
|
|
|
|
|
parse_function(name);
|
|
|
|
parse_function(name);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
if (type == TYPE_VOID) {
|
|
|
|
if (type == TYPE_VOID) {
|
|
|
|
fprintf(stderr, "variable cannot be of void type\n");
|
|
|
|
fprintf(stderr, "variable cannot be of void type\n");
|
|
|
|
exit(1);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
declare_global(id, MARKER_SCALAR, type);
|
|
|
|
declare_global(id, KIND_SCALAR, type);
|
|
|
|
if (external) {
|
|
|
|
if (external) {
|
|
|
|
unget_token();
|
|
|
|
unget_token();
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
@ -1904,11 +1923,49 @@ void parse_global_declaration() {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void parse_enum() {
|
|
|
|
|
|
|
|
expect_token(TOKEN_BRACE_LEFT);
|
|
|
|
|
|
|
|
int value = 0;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
|
|
|
|
next_token();
|
|
|
|
|
|
|
|
if (token_type == TOKEN_BRACE_RIGHT) {
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (token_type != TOKEN_ID) {
|
|
|
|
|
|
|
|
fprintf(stderr, "expecting identifier in enum\n");
|
|
|
|
|
|
|
|
exit(1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int id = token_data;
|
|
|
|
|
|
|
|
next_token();
|
|
|
|
|
|
|
|
if (token_type == TOKEN_ASSIGN) {
|
|
|
|
|
|
|
|
value = expect_const();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
unget_token();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
const_table[id] = value++;
|
|
|
|
|
|
|
|
is_const[id] = 1;
|
|
|
|
|
|
|
|
next_token();
|
|
|
|
|
|
|
|
if (token_type == TOKEN_COMMA) {
|
|
|
|
|
|
|
|
// continue;
|
|
|
|
|
|
|
|
} else if (token_type == TOKEN_BRACE_RIGHT) {
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
fprintf(stderr, "expecting ',' or '}'\n");
|
|
|
|
|
|
|
|
exit(1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
expect_token(TOKEN_SEMICOLON);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void parse_top_level() {
|
|
|
|
void parse_top_level() {
|
|
|
|
next_token();
|
|
|
|
next_token();
|
|
|
|
if (token_type == TOKEN_EOF)
|
|
|
|
if (token_type == TOKEN_EOF)
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
parse_global_declaration();
|
|
|
|
if (token_type == TOKEN_ENUM) {
|
|
|
|
|
|
|
|
parse_enum();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
parse_global_declaration();
|
|
|
|
|
|
|
|
}
|
|
|
|
parse_top_level();
|
|
|
|
parse_top_level();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|