2001-11-01 22:48:10 +08:00
|
|
|
/*
|
|
|
|
* TCC - Tiny C Compiler
|
|
|
|
*
|
|
|
|
* Copyright (c) 2001 Fabrice Bellard
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*/
|
2001-11-06 09:20:20 +08:00
|
|
|
#include "tcclib.h"
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
#define INCLUDE_PATH "/usr/include"
|
|
|
|
|
|
|
|
#define TEXT_SIZE 50000
|
|
|
|
#define DATA_SIZE 50000
|
|
|
|
#define SYM_TABLE_SIZE 10000
|
|
|
|
#define MACRO_STACK_SIZE 32
|
|
|
|
#define INCLUDE_STACK_SIZE 32
|
|
|
|
|
|
|
|
#define NB_REGS 3
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
/* symbol management */
|
|
|
|
typedef struct Sym {
|
|
|
|
int v; /* symbol token */
|
|
|
|
int t; /* associated type */
|
|
|
|
int c; /* associated number */
|
|
|
|
struct Sym *next; /* next related symbol */
|
|
|
|
struct Sym *prev; /* prev symbol in stack */
|
|
|
|
} Sym;
|
|
|
|
|
|
|
|
#define SYM_STRUCT 0x40000000 /* struct/union/enum symbol space */
|
|
|
|
#define SYM_FIELD 0x20000000 /* struct/union field symbol space */
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
typedef struct {
|
|
|
|
FILE *file; /* stdio file */
|
|
|
|
char *filename;
|
|
|
|
int line_num;
|
|
|
|
} IncludeFile;
|
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
/* loc : local variable index
|
2001-10-28 07:48:39 +08:00
|
|
|
glo : global variable index
|
|
|
|
ind : output code ptr
|
|
|
|
rsym: return symbol
|
|
|
|
prog: output code
|
2001-11-06 09:20:20 +08:00
|
|
|
anon_sym: anonymous symbol index
|
2001-10-28 07:48:39 +08:00
|
|
|
*/
|
2001-11-06 09:20:20 +08:00
|
|
|
FILE *file;
|
|
|
|
int tok, tok1, tokc, rsym, anon_sym,
|
|
|
|
prog, ind, loc, glo, vt, *vstack, *vstack_ptr,
|
|
|
|
vc, line_num;
|
2001-10-30 07:23:01 +08:00
|
|
|
char *idtable, *idptr, *filename;
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *define_stack, *global_stack, *local_stack, *label_stack;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
char *macro_stack[MACRO_STACK_SIZE], **macro_stack_ptr, *macro_ptr;
|
|
|
|
IncludeFile include_stack[INCLUDE_STACK_SIZE], *include_stack_ptr;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
/* The current value can be: */
|
|
|
|
#define VT_VALMASK 0x000f
|
|
|
|
#define VT_CONST 0x000a /* constant in vc
|
|
|
|
(must be first non register value) */
|
|
|
|
#define VT_LLOCAL 0x000b /* lvalue, offset on stack */
|
|
|
|
#define VT_LOCAL 0x000c /* offset on stack */
|
|
|
|
#define VT_CMP 0x000d /* the value is stored in processor flags (in vc) */
|
|
|
|
#define VT_JMP 0x000e /* value is the consequence of jmp true */
|
|
|
|
#define VT_JMPI 0x000f /* value is the consequence of jmp false */
|
|
|
|
#define VT_LVAL 0x0010 /* var is an lvalue */
|
2001-10-28 07:48:39 +08:00
|
|
|
#define VT_LVALN -17 /* ~VT_LVAL */
|
2001-11-06 09:20:20 +08:00
|
|
|
#define VT_FORWARD 0x0020 /* value is forward reference
|
|
|
|
(only used for functions) */
|
|
|
|
|
|
|
|
#define VT_VOID 0x00040
|
|
|
|
#define VT_BYTE 0x00080 /* byte type */
|
|
|
|
#define VT_PTR 0x00100 /* pointer increment */
|
|
|
|
#define VT_UNSIGNED 0x00200 /* unsigned type */
|
|
|
|
#define VT_ARRAY 0x00400 /* array type (only used in parsing) */
|
|
|
|
#define VT_ENUM 0x00800 /* enum definition */
|
2001-10-30 07:23:01 +08:00
|
|
|
#define VT_FUNC 0x01000 /* function type */
|
2001-11-06 09:20:20 +08:00
|
|
|
#define VT_STRUCT 0x002000 /* struct/union definition */
|
|
|
|
#define VT_TYPEDEF 0x004000 /* typedef definition */
|
2001-11-01 22:48:10 +08:00
|
|
|
#define VT_EXTERN 0x008000 /* extern definition */
|
|
|
|
#define VT_STATIC 0x010000 /* static variable */
|
2001-11-06 09:20:20 +08:00
|
|
|
#define VT_STRUCT_SHIFT 17 /* structure/enum name shift (12 bits lefts) */
|
|
|
|
|
|
|
|
#define VT_TYPE 0xffffffc0 /* type mask */
|
|
|
|
#define VT_TYPEN 0x0000003f /* ~VT_TYPE */
|
|
|
|
#define VT_FUNCN -4097 /* ~VT_FUNC */
|
|
|
|
|
2001-10-30 07:23:01 +08:00
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
/* Special infos */
|
|
|
|
|
|
|
|
/* token values */
|
2001-10-30 07:23:01 +08:00
|
|
|
#define TOK_INT 256
|
|
|
|
#define TOK_VOID 257
|
|
|
|
#define TOK_CHAR 258
|
|
|
|
#define TOK_IF 259
|
|
|
|
#define TOK_ELSE 260
|
|
|
|
#define TOK_WHILE 261
|
|
|
|
#define TOK_BREAK 262
|
|
|
|
#define TOK_RETURN 263
|
|
|
|
#define TOK_DEFINE 264
|
|
|
|
#define TOK_MAIN 265
|
|
|
|
#define TOK_FOR 266
|
|
|
|
#define TOK_EXTERN 267
|
|
|
|
#define TOK_STATIC 268
|
|
|
|
#define TOK_UNSIGNED 269
|
|
|
|
#define TOK_GOTO 270
|
|
|
|
#define TOK_DO 271
|
|
|
|
#define TOK_CONTINUE 272
|
|
|
|
#define TOK_SWITCH 273
|
|
|
|
#define TOK_CASE 274
|
|
|
|
|
|
|
|
/* ignored types Must have contiguous values */
|
2001-10-31 04:27:54 +08:00
|
|
|
#define TOK_CONST 275
|
|
|
|
#define TOK_VOLATILE 276
|
|
|
|
#define TOK_LONG 277
|
|
|
|
#define TOK_REGISTER 278
|
|
|
|
#define TOK_SIGNED 279
|
2001-11-06 09:20:20 +08:00
|
|
|
#define TOK_AUTO 280
|
|
|
|
#define TOK_INLINE 281
|
2001-10-30 07:23:01 +08:00
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
#define TOK_FLOAT 282 /* unsupported */
|
|
|
|
#define TOK_DOUBLE 283 /* unsupported */
|
2001-11-01 22:48:10 +08:00
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
#define TOK_STRUCT 284
|
|
|
|
#define TOK_UNION 285
|
|
|
|
#define TOK_TYPEDEF 286
|
|
|
|
#define TOK_DEFAULT 287
|
|
|
|
#define TOK_ENUM 288
|
|
|
|
#define TOK_SIZEOF 289
|
|
|
|
#define TOK_INCLUDE 290
|
2001-10-28 07:48:39 +08:00
|
|
|
|
|
|
|
#define TOK_EQ 0x94 /* warning: depend on asm code */
|
|
|
|
#define TOK_NE 0x95 /* warning: depend on asm code */
|
|
|
|
#define TOK_LT 0x9c /* warning: depend on asm code */
|
|
|
|
#define TOK_GE 0x9d /* warning: depend on asm code */
|
|
|
|
#define TOK_LE 0x9e /* warning: depend on asm code */
|
|
|
|
#define TOK_GT 0x9f /* warning: depend on asm code */
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
#define TOK_LAND 0xa0
|
|
|
|
#define TOK_LOR 0xa1
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
#define TOK_DEC 0xa2
|
|
|
|
#define TOK_MID 0xa3 /* inc/dec, to void constant */
|
|
|
|
#define TOK_INC 0xa4
|
|
|
|
#define TOK_ARROW 0xa7
|
2001-11-06 09:20:20 +08:00
|
|
|
#define TOK_DOTS 0xa8 /* three dots */
|
|
|
|
#define TOK_SHR 0xa9 /* unsigned shift right */
|
|
|
|
#define TOK_UDIV 0xb0 /* unsigned division */
|
|
|
|
#define TOK_UMOD 0xb1 /* unsigned modulo */
|
|
|
|
#define TOK_PDIV 0xb2 /* fast division with undefined rounding for pointers */
|
|
|
|
#define TOK_NUM 0xb3 /* number in tokc */
|
|
|
|
|
|
|
|
#define TOK_SHL 0x01 /* shift left */
|
|
|
|
#define TOK_SAR 0x02 /* signed shift right */
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-10-29 06:24:16 +08:00
|
|
|
/* assignement operators : normal operator or 0x80 */
|
|
|
|
#define TOK_A_MOD 0xa5
|
|
|
|
#define TOK_A_AND 0xa6
|
|
|
|
#define TOK_A_MUL 0xaa
|
|
|
|
#define TOK_A_ADD 0xab
|
|
|
|
#define TOK_A_SUB 0xad
|
|
|
|
#define TOK_A_DIV 0xaf
|
|
|
|
#define TOK_A_XOR 0xde
|
|
|
|
#define TOK_A_OR 0xfc
|
|
|
|
#define TOK_A_SHL 0x81
|
2001-11-06 09:20:20 +08:00
|
|
|
#define TOK_A_SAR 0x82
|
2001-10-29 06:24:16 +08:00
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
void sum();
|
|
|
|
void next();
|
|
|
|
void expr_eq();
|
|
|
|
void expr();
|
|
|
|
void decl();
|
2001-11-06 09:20:20 +08:00
|
|
|
int gv();
|
|
|
|
void move_reg();
|
|
|
|
void save_reg();
|
2001-10-28 07:48:39 +08:00
|
|
|
|
|
|
|
int isid(c)
|
|
|
|
{
|
|
|
|
return (c >= 'a' & c <= 'z') |
|
|
|
|
(c >= 'A' & c <= 'Z') |
|
|
|
|
c == '_';
|
|
|
|
}
|
|
|
|
|
|
|
|
int isnum(c)
|
|
|
|
{
|
|
|
|
return c >= '0' & c <= '9';
|
|
|
|
}
|
|
|
|
|
2001-10-28 23:20:12 +08:00
|
|
|
#ifndef TINY
|
|
|
|
/* XXX: use stderr ? */
|
|
|
|
void error(char *msg)
|
|
|
|
{
|
|
|
|
printf("%s:%d: %s\n", filename, line_num, msg);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
void expect(char *msg)
|
|
|
|
{
|
|
|
|
printf("%s:%d: %s expected\n", filename, line_num, msg);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2001-10-28 23:20:12 +08:00
|
|
|
void warning(char *msg)
|
|
|
|
{
|
|
|
|
printf("%s:%d: warning: %s\n", filename, line_num, msg);
|
|
|
|
}
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
void skip(c)
|
|
|
|
{
|
|
|
|
if (tok != c) {
|
2001-10-28 23:20:12 +08:00
|
|
|
printf("%s:%d: '%c' expected\n", filename, line_num, c);
|
2001-10-28 07:48:39 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
next();
|
|
|
|
}
|
2001-10-28 23:20:12 +08:00
|
|
|
|
|
|
|
void test_lvalue()
|
|
|
|
{
|
|
|
|
if (!(vt & VT_LVAL))
|
2001-11-01 22:48:10 +08:00
|
|
|
expect("lvalue");
|
2001-10-28 23:20:12 +08:00
|
|
|
}
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
#else
|
2001-10-28 23:20:12 +08:00
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
#define skip(c) next()
|
2001-10-28 23:20:12 +08:00
|
|
|
#define test_lvalue()
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
#endif
|
|
|
|
|
2001-10-30 07:23:01 +08:00
|
|
|
char *get_tok_str(int v)
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
char *p;
|
|
|
|
p = idtable;
|
|
|
|
t = 256;
|
|
|
|
while (t != v) {
|
|
|
|
if (p >= idptr)
|
|
|
|
return 0;
|
|
|
|
while (*p++);
|
|
|
|
t++;
|
|
|
|
}
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
/* find a symbol and return its associated structure. 's' is the top
|
|
|
|
of the symbol stack */
|
|
|
|
Sym *sym_find1(Sym *s, int v)
|
|
|
|
{
|
|
|
|
while (s) {
|
|
|
|
if (s->v == v)
|
|
|
|
return s;
|
|
|
|
s = s->prev;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Sym *sym_push1(Sym **ps, int v, int t, int c)
|
|
|
|
{
|
|
|
|
Sym *s;
|
|
|
|
s = malloc(sizeof(Sym));
|
|
|
|
if (!s)
|
|
|
|
error("memory full");
|
|
|
|
s->v = v;
|
|
|
|
s->t = t;
|
|
|
|
s->c = c;
|
|
|
|
s->next = 0;
|
|
|
|
s->prev = *ps;
|
|
|
|
*ps = s;
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find a symbol in the right symbol space */
|
|
|
|
Sym *sym_find(int v)
|
|
|
|
{
|
|
|
|
Sym *s;
|
|
|
|
s = sym_find1(local_stack, v);
|
|
|
|
if (!s)
|
|
|
|
s = sym_find1(global_stack, v);
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* push a given symbol on the symbol stack */
|
|
|
|
Sym *sym_push(int v, int t, int c)
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
// printf("sym_push: %x %s type=%x\n", v, get_tok_str(v), t);
|
2001-11-01 22:48:10 +08:00
|
|
|
if (local_stack)
|
|
|
|
return sym_push1(&local_stack, v, t, c);
|
|
|
|
else
|
|
|
|
return sym_push1(&global_stack, v, t, c);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* pop symbols until top reaches 'b' */
|
|
|
|
void sym_pop(Sym **ps, Sym *b)
|
|
|
|
{
|
|
|
|
Sym *s, *ss;
|
|
|
|
|
|
|
|
s = *ps;
|
|
|
|
while(s != b) {
|
|
|
|
ss = s->prev;
|
2001-11-06 09:20:20 +08:00
|
|
|
// printf("sym_pop: %x %s type=%x\n", s->v, get_tok_str(s->v), s->t);
|
2001-11-01 22:48:10 +08:00
|
|
|
free(s);
|
|
|
|
s = ss;
|
|
|
|
}
|
|
|
|
*ps = b;
|
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
int ch, ch1;
|
|
|
|
|
|
|
|
/* read next char from current input file */
|
|
|
|
void inp()
|
|
|
|
{
|
|
|
|
int c;
|
|
|
|
|
|
|
|
redo:
|
|
|
|
ch1 = fgetc(file);
|
|
|
|
if (ch1 == -1) {
|
|
|
|
if (include_stack_ptr == include_stack)
|
|
|
|
return;
|
|
|
|
/* pop include stack */
|
|
|
|
fclose(file);
|
|
|
|
free(filename);
|
|
|
|
include_stack_ptr--;
|
|
|
|
file = include_stack_ptr->file;
|
|
|
|
filename = include_stack_ptr->filename;
|
|
|
|
line_num = include_stack_ptr->line_num;
|
|
|
|
goto redo;
|
|
|
|
}
|
|
|
|
if (ch1 == '\n')
|
|
|
|
line_num++;
|
|
|
|
// printf("ch1=%c\n", ch1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* input with '\\n' handling and macro subtitution if in macro state */
|
|
|
|
void minp()
|
|
|
|
{
|
|
|
|
if (macro_ptr != 0) {
|
|
|
|
ch = *macro_ptr++;
|
|
|
|
/* end of macro ? */
|
|
|
|
if (ch == '\0') {
|
|
|
|
macro_ptr = *--macro_stack_ptr;
|
|
|
|
ch = (int)*--macro_stack_ptr;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
redo:
|
|
|
|
ch = ch1;
|
|
|
|
inp();
|
|
|
|
if (ch == '\\' && ch1 == '\n') {
|
|
|
|
inp();
|
|
|
|
goto redo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* same as minp, but also skip comments */
|
|
|
|
void cinp()
|
|
|
|
{
|
|
|
|
int c;
|
|
|
|
|
|
|
|
if (ch1 == '/') {
|
|
|
|
inp();
|
|
|
|
if (ch1 == '/') {
|
|
|
|
/* single line C++ comments */
|
|
|
|
inp();
|
|
|
|
while (ch1 != '\n' && ch1 != -1)
|
|
|
|
inp();
|
|
|
|
inp();
|
|
|
|
ch = ' '; /* return space */
|
|
|
|
} else if (ch1 == '*') {
|
|
|
|
/* C comments */
|
|
|
|
inp();
|
|
|
|
while (ch1 != -1) {
|
|
|
|
c = ch1;
|
|
|
|
inp();
|
|
|
|
if (c == '*' && ch1 == '/') {
|
|
|
|
inp();
|
|
|
|
ch = ' '; /* return space */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ch = '/';
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
minp();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void skip_spaces()
|
|
|
|
{
|
|
|
|
while (ch == ' ' || ch == '\t')
|
|
|
|
cinp();
|
|
|
|
}
|
|
|
|
|
|
|
|
void preprocess()
|
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
int size, n, c;
|
|
|
|
char buf[1024], *q, *p;
|
|
|
|
char buf1[1024];
|
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
cinp();
|
|
|
|
next(); /* XXX: should pass parameter to avoid macro subst */
|
|
|
|
if (tok == TOK_DEFINE) {
|
|
|
|
next(); /* XXX: should pass parameter to avoid macro subst */
|
|
|
|
skip_spaces();
|
|
|
|
/* now 'tok' is the macro symbol */
|
|
|
|
str = NULL;
|
|
|
|
size = 0;
|
|
|
|
n = 0;
|
|
|
|
while (1) {
|
|
|
|
if ((n + 1) >= size) {
|
|
|
|
size += 128;
|
|
|
|
str = realloc(str, size);
|
|
|
|
if (!str)
|
|
|
|
error("memory full");
|
|
|
|
}
|
|
|
|
if (ch == -1 || ch == '\n') {
|
|
|
|
str[n++] = ' '; /* a space is inserted after each macro */
|
|
|
|
str[n++] = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
str[n++] = ch;
|
|
|
|
cinp();
|
|
|
|
}
|
|
|
|
sym_push1(&define_stack, tok, 0, (int)str);
|
|
|
|
} else if (tok == TOK_INCLUDE) {
|
|
|
|
skip_spaces();
|
|
|
|
if (ch == '<') {
|
|
|
|
c = '>';
|
|
|
|
goto read_name;
|
|
|
|
} else if (ch == '\"') {
|
|
|
|
c = ch;
|
|
|
|
read_name:
|
|
|
|
minp();
|
|
|
|
q = buf;
|
|
|
|
while (ch != c && ch != '\n' && ch != -1) {
|
|
|
|
if ((q - buf) < sizeof(buf) - 1)
|
|
|
|
*q++ = ch;
|
|
|
|
minp();
|
|
|
|
}
|
|
|
|
*q = '\0';
|
|
|
|
if (include_stack_ptr >= include_stack + INCLUDE_STACK_SIZE)
|
|
|
|
error("memory full");
|
|
|
|
if (c == '\"') {
|
|
|
|
/* first search in current dir if "header.h" */
|
|
|
|
/* XXX: buffer overflow */
|
|
|
|
size = 0;
|
|
|
|
p = strrchr(filename, '/');
|
|
|
|
if (p)
|
|
|
|
size = p + 1 - filename;
|
|
|
|
memcpy(buf1, filename, size);
|
|
|
|
buf1[size] = '\0';
|
|
|
|
strcat(buf1, buf);
|
|
|
|
f = fopen(buf1, "r");
|
|
|
|
if (f)
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
/* now search in standard include path */
|
|
|
|
strcpy(buf1, INCLUDE_PATH);
|
|
|
|
strcat(buf1, "/");
|
|
|
|
strcat(buf1, buf);
|
|
|
|
f = fopen(buf1, "r");
|
|
|
|
if (!f)
|
|
|
|
error("include file not found");
|
|
|
|
found:
|
|
|
|
/* push current file in stack */
|
|
|
|
include_stack_ptr->file = file;
|
|
|
|
include_stack_ptr->filename = filename;
|
|
|
|
include_stack_ptr->line_num = line_num;
|
|
|
|
include_stack_ptr++;
|
|
|
|
file = f;
|
|
|
|
filename = strdup(buf1);
|
|
|
|
line_num = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
/* ignore other preprocess commands or #! for C scripts */
|
|
|
|
while (ch != '\n' && ch != -1)
|
|
|
|
cinp();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* read a number in base b */
|
|
|
|
int getn(b)
|
|
|
|
{
|
|
|
|
int n, t;
|
|
|
|
n = 0;
|
|
|
|
while (1) {
|
|
|
|
if (ch >= 'a' & ch <= 'f')
|
|
|
|
t = ch - 'a' + 10;
|
|
|
|
else if (ch >= 'A' & ch <= 'F')
|
|
|
|
t = ch - 'A' + 10;
|
|
|
|
else if (isnum(ch))
|
|
|
|
t = ch - '0';
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
if (t < 0 | t >= b)
|
|
|
|
break;
|
|
|
|
n = n * b + t;
|
|
|
|
cinp();
|
|
|
|
}
|
|
|
|
return n;
|
|
|
|
}
|
2001-11-01 22:48:10 +08:00
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
void next()
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int v, b;
|
2001-10-28 07:48:39 +08:00
|
|
|
char *q, *p;
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *s;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-10-31 06:43:19 +08:00
|
|
|
/* special 'ungettok' case for label parsing */
|
|
|
|
if (tok1) {
|
|
|
|
tok = tok1;
|
|
|
|
tok1 = 0;
|
|
|
|
return;
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
redo:
|
|
|
|
/* skip spaces */
|
2001-10-28 07:48:39 +08:00
|
|
|
while(1) {
|
2001-11-06 09:20:20 +08:00
|
|
|
while (ch == '\n') {
|
|
|
|
cinp();
|
|
|
|
while (ch == ' ' || ch == 9)
|
|
|
|
cinp();
|
|
|
|
if (ch == '#') {
|
|
|
|
/* preprocessor command if # at start of line after
|
|
|
|
spaces */
|
|
|
|
preprocess();
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-10-28 23:20:12 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
if (ch != ' ' && ch != 9)
|
2001-10-28 07:48:39 +08:00
|
|
|
break;
|
2001-11-06 09:20:20 +08:00
|
|
|
cinp();
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
if (isid(ch)) {
|
2001-10-28 07:48:39 +08:00
|
|
|
q = idptr;
|
2001-11-06 09:20:20 +08:00
|
|
|
while(isid(ch) | isnum(ch)) {
|
|
|
|
*q++ = ch;
|
|
|
|
cinp();
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
*q++ = '\0';
|
|
|
|
p = idtable;
|
|
|
|
tok = 256;
|
|
|
|
while (p < idptr) {
|
|
|
|
if (strcmp(p, idptr) == 0)
|
|
|
|
break;
|
|
|
|
while (*p++);
|
|
|
|
tok++;
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
// printf("id=%s\n", idptr);
|
2001-10-28 07:48:39 +08:00
|
|
|
/* if not found, add symbol */
|
|
|
|
if (p == idptr)
|
|
|
|
idptr = q;
|
2001-11-06 09:20:20 +08:00
|
|
|
/* if symbol is a define, prepare substitution */
|
2001-11-01 22:48:10 +08:00
|
|
|
if (s = sym_find1(define_stack, tok)) {
|
2001-11-06 09:20:20 +08:00
|
|
|
if ((macro_stack_ptr - macro_stack) >= MACRO_STACK_SIZE)
|
|
|
|
error("too many nested macros");
|
|
|
|
*macro_stack_ptr++ = (char *)ch;
|
|
|
|
*macro_stack_ptr++ = macro_ptr;
|
|
|
|
macro_ptr = (char *)s->c;
|
|
|
|
cinp();
|
|
|
|
goto redo;
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
} else if (isnum(ch)) {
|
|
|
|
/* number */
|
|
|
|
b = 10;
|
|
|
|
if (ch == '0') {
|
|
|
|
cinp();
|
|
|
|
b = 8;
|
|
|
|
if (ch == 'x') {
|
|
|
|
cinp();
|
|
|
|
b = 16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tokc = getn(b);
|
|
|
|
tok = TOK_NUM;
|
2001-10-28 07:48:39 +08:00
|
|
|
} else {
|
2001-10-28 10:51:52 +08:00
|
|
|
#ifdef TINY
|
|
|
|
q = "<=\236>=\235!=\225++\244--\242==\224";
|
|
|
|
#else
|
2001-11-06 09:20:20 +08:00
|
|
|
q = "<=\236>=\235!=\225&&\240||\241++\244--\242==\224<<\1>>\2+=\253-=\255*=\252/=\257%=\245&=\246^=\336|=\374->\247..\250";
|
2001-10-28 10:51:52 +08:00
|
|
|
#endif
|
2001-10-28 07:48:39 +08:00
|
|
|
/* two chars */
|
2001-11-06 09:20:20 +08:00
|
|
|
tok = ch;
|
|
|
|
cinp();
|
2001-10-28 07:48:39 +08:00
|
|
|
while (*q) {
|
2001-11-06 09:20:20 +08:00
|
|
|
if (*q == tok & q[1] == ch) {
|
|
|
|
cinp();
|
2001-10-28 07:48:39 +08:00
|
|
|
tok = q[2] & 0xff;
|
2001-11-06 09:20:20 +08:00
|
|
|
/* three chars tests */
|
|
|
|
if (tok == TOK_SHL | tok == TOK_SAR) {
|
|
|
|
if (ch == '=') {
|
2001-10-29 06:24:16 +08:00
|
|
|
tok = tok | 0x80;
|
2001-11-06 09:20:20 +08:00
|
|
|
cinp();
|
|
|
|
}
|
|
|
|
} else if (tok == TOK_DOTS) {
|
|
|
|
if (ch != '.')
|
|
|
|
error("parse error");
|
|
|
|
cinp();
|
2001-10-29 06:24:16 +08:00
|
|
|
}
|
2001-10-28 07:48:39 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
q = q + 3;
|
|
|
|
}
|
|
|
|
/* single char substitutions */
|
2001-11-06 09:20:20 +08:00
|
|
|
if (tok == '<')
|
2001-10-28 07:48:39 +08:00
|
|
|
tok = TOK_LT;
|
2001-11-06 09:20:20 +08:00
|
|
|
else if (tok == '>')
|
2001-10-28 07:48:39 +08:00
|
|
|
tok = TOK_GT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
void swap(int *p, int *q)
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
t = *p;
|
|
|
|
*p = *q;
|
|
|
|
*q = t;
|
|
|
|
}
|
|
|
|
|
|
|
|
void vset(t, v)
|
|
|
|
{
|
|
|
|
vt = t;
|
|
|
|
vc = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************/
|
|
|
|
/* X86 code generator */
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
void g(c)
|
|
|
|
{
|
|
|
|
*(char *)ind++ = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void o(c)
|
|
|
|
{
|
|
|
|
while (c) {
|
|
|
|
g(c);
|
|
|
|
c = c / 256;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* output a symbol and patch all calls to it */
|
2001-10-31 04:27:54 +08:00
|
|
|
void gsym_addr(t, a)
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
|
|
|
int n;
|
|
|
|
while (t) {
|
|
|
|
n = *(int *)t; /* next value */
|
2001-10-31 04:27:54 +08:00
|
|
|
*(int *)t = a - t - 4;
|
2001-10-28 07:48:39 +08:00
|
|
|
t = n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-10-31 04:27:54 +08:00
|
|
|
void gsym(t)
|
|
|
|
{
|
|
|
|
gsym_addr(t, ind);
|
|
|
|
}
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
/* psym is used to put an instruction with a data field which is a
|
|
|
|
reference to a symbol. It is in fact the same as oad ! */
|
|
|
|
#define psym oad
|
|
|
|
|
|
|
|
/* instruction + 4 bytes data. Return the address of the data */
|
|
|
|
int oad(c, s)
|
|
|
|
{
|
|
|
|
o(c);
|
|
|
|
*(int *)ind = s;
|
|
|
|
s = ind;
|
|
|
|
ind = ind + 4;
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2001-10-30 07:23:01 +08:00
|
|
|
/* XXX: generate correct pointer for forward references to functions */
|
2001-11-06 09:20:20 +08:00
|
|
|
/* r = (ft, fc) */
|
|
|
|
void load(r, ft, fc)
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int v, t;
|
|
|
|
|
|
|
|
v = ft & VT_VALMASK;
|
|
|
|
if (ft & VT_LVAL) {
|
|
|
|
if (v == VT_LLOCAL) {
|
|
|
|
load(r, VT_LOCAL | VT_LVAL, fc);
|
|
|
|
v = r;
|
|
|
|
}
|
|
|
|
if ((ft & VT_TYPE) == VT_BYTE)
|
|
|
|
o(0xbe0f); /* movsbl */
|
2001-10-28 07:48:39 +08:00
|
|
|
else
|
2001-11-06 09:20:20 +08:00
|
|
|
o(0x8b); /* movl */
|
|
|
|
if (v == VT_CONST) {
|
|
|
|
oad(0x05 + r * 8, fc); /* 0xXX, r */
|
|
|
|
} else if (v == VT_LOCAL) {
|
|
|
|
oad(0x85 + r * 8, fc); /* xx(%ebp), r */
|
|
|
|
} else {
|
|
|
|
g(0x00 + r * 8 + v); /* (v), r */
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
} else {
|
|
|
|
if (v == VT_CONST) {
|
|
|
|
oad(0xb8 + r, fc); /* mov $xx, r */
|
|
|
|
} else if (v == VT_LOCAL) {
|
|
|
|
o(0x8d);
|
|
|
|
oad(0x85 + r * 8, fc); /* lea xxx(%ebp), r */
|
|
|
|
} else if (v == VT_CMP) {
|
|
|
|
oad(0xb8 + r, 0); /* mov $0, r */
|
|
|
|
o(0x0f); /* setxx %br */
|
|
|
|
o(fc);
|
|
|
|
o(0xc0 + r);
|
|
|
|
} else if (v == VT_JMP || v == VT_JMPI) {
|
|
|
|
t = v & 1;
|
|
|
|
oad(0xb8 + r, t); /* mov $1, r */
|
2001-10-28 07:48:39 +08:00
|
|
|
oad(0xe9, 5); /* jmp after */
|
2001-11-06 09:20:20 +08:00
|
|
|
gsym(fc);
|
|
|
|
oad(0xb8 + r, t ^ 1); /* mov $0, r */
|
|
|
|
} else if (v != r) {
|
|
|
|
o(0x89);
|
|
|
|
o(0xc0 + r + v * 8); /* mov v, r */
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* (ft, fc) = r */
|
|
|
|
/* WARNING: r must not be allocated on the stack */
|
|
|
|
void store(r, ft, fc)
|
|
|
|
{
|
|
|
|
int fr, b;
|
|
|
|
|
|
|
|
fr = ft & VT_VALMASK;
|
|
|
|
b = (ft & VT_TYPE) == VT_BYTE;
|
|
|
|
o(0x89 - b);
|
|
|
|
if (fr == VT_CONST) {
|
|
|
|
oad(0x05 + r * 8, fc); /* mov r,xxx */
|
|
|
|
} else if (fr == VT_LOCAL) {
|
|
|
|
oad(0x85 + r * 8, fc); /* mov r,xxx(%ebp) */
|
|
|
|
} else if (ft & VT_LVAL) {
|
|
|
|
g(fr + r * 8); /* mov r, (fr) */
|
|
|
|
} else if (fr != r) {
|
|
|
|
o(0xc0 + fr + r * 8); /* mov r, fr */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int gjmp(t)
|
|
|
|
{
|
|
|
|
return psym(0xe9, t);
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* generate a test. set 'inv' to invert test */
|
|
|
|
int gtst(inv, t)
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int v, *p;
|
|
|
|
v = vt & VT_VALMASK;
|
|
|
|
if (v == VT_CMP) {
|
2001-10-28 07:48:39 +08:00
|
|
|
/* fast case : can jump directly since flags are set */
|
|
|
|
g(0x0f);
|
|
|
|
t = psym((vc - 16) ^ inv, t);
|
2001-11-06 09:20:20 +08:00
|
|
|
} else if (v == VT_JMP || v == VT_JMPI) {
|
2001-10-28 07:48:39 +08:00
|
|
|
/* && or || optimization */
|
2001-11-06 09:20:20 +08:00
|
|
|
if ((v & 1) == inv) {
|
|
|
|
/* insert vc jump list in t */
|
|
|
|
p = &vc;
|
|
|
|
while (*p != 0)
|
|
|
|
p = (int *)*p;
|
|
|
|
*p = t;
|
2001-10-28 07:48:39 +08:00
|
|
|
t = vc;
|
2001-11-06 09:20:20 +08:00
|
|
|
} else {
|
|
|
|
t = gjmp(t);
|
2001-10-28 07:48:39 +08:00
|
|
|
gsym(vc);
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
} else if ((vt & (VT_VALMASK | VT_LVAL)) == VT_CONST) {
|
2001-10-28 07:48:39 +08:00
|
|
|
/* constant jmp optimization */
|
|
|
|
if ((vc != 0) != inv)
|
2001-11-06 09:20:20 +08:00
|
|
|
t = gjmp(t);
|
|
|
|
} else {
|
|
|
|
v = gv();
|
|
|
|
o(0x85);
|
|
|
|
o(0xc0 + v * 9);
|
2001-10-28 07:48:39 +08:00
|
|
|
g(0x0f);
|
|
|
|
t = psym(0x85 ^ inv, t);
|
|
|
|
}
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
/* generate a binary operation 'v = r op fr' instruction and modifies
|
|
|
|
(vt,vc) if needed */
|
|
|
|
void gen_op1(op, r, fr)
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
if (op == '+') {
|
|
|
|
o(0x01);
|
|
|
|
o(0xc0 + r + fr * 8);
|
|
|
|
} else if (op == '-') {
|
|
|
|
o(0x29);
|
|
|
|
o(0xc0 + r + fr * 8);
|
|
|
|
} else if (op == '&') {
|
|
|
|
o(0x21);
|
|
|
|
o(0xc0 + r + fr * 8);
|
|
|
|
} else if (op == '^') {
|
|
|
|
o(0x31);
|
|
|
|
o(0xc0 + r + fr * 8);
|
|
|
|
} else if (op == '|') {
|
|
|
|
o(0x09);
|
|
|
|
o(0xc0 + r + fr * 8);
|
|
|
|
} else if (op == '*') {
|
|
|
|
o(0xaf0f); /* imul fr, r */
|
|
|
|
o(0xc0 + fr + r * 8);
|
|
|
|
} else if (op == TOK_SHL | op == TOK_SHR | op == TOK_SAR) {
|
|
|
|
/* op2 is %ecx */
|
|
|
|
if (fr != 1) {
|
|
|
|
if (r == 1) {
|
|
|
|
r = fr;
|
|
|
|
fr = 1;
|
|
|
|
o(0x87); /* xchg r, %ecx */
|
|
|
|
o(0xc1 + r * 8);
|
|
|
|
} else
|
|
|
|
move_reg(1, fr);
|
|
|
|
}
|
|
|
|
o(0xd3); /* shl/shr/sar %cl, r */
|
|
|
|
if (op == TOK_SHL)
|
|
|
|
o(0xe0 + r);
|
|
|
|
else if (op == TOK_SHR)
|
|
|
|
o(0xe8 + r);
|
|
|
|
else
|
|
|
|
o(0xf8 + r);
|
|
|
|
vt = (vt & VT_TYPE) | r;
|
|
|
|
} else if (op == '/' | op == TOK_UDIV | op == TOK_PDIV |
|
|
|
|
op == '%' | op == TOK_UMOD) {
|
|
|
|
save_reg(2); /* save edx */
|
|
|
|
t = save_reg_forced(fr); /* save fr and get op2 location */
|
|
|
|
move_reg(0, r); /* op1 is %eax */
|
|
|
|
if (op == TOK_UDIV | op == TOK_UMOD) {
|
|
|
|
o(0xf7d231); /* xor %edx, %edx, div t(%ebp), %eax */
|
|
|
|
oad(0xb5, t);
|
|
|
|
} else {
|
|
|
|
o(0xf799); /* cltd, idiv t(%ebp), %eax */
|
|
|
|
oad(0xbd, t);
|
|
|
|
}
|
|
|
|
if (op == '%' | op == TOK_UMOD)
|
|
|
|
r = 2;
|
|
|
|
else
|
|
|
|
r = 0;
|
|
|
|
vt = (vt & VT_TYPE) | r;
|
|
|
|
} else {
|
|
|
|
o(0x39);
|
|
|
|
o(0xc0 + r + fr * 8); /* cmp fr, r */
|
|
|
|
vset(VT_CMP, op);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* end of X86 code generator */
|
|
|
|
/*************************************************************/
|
|
|
|
|
|
|
|
int save_reg_forced(r)
|
|
|
|
{
|
|
|
|
int i, l, *p, t;
|
|
|
|
/* store register */
|
|
|
|
loc = (loc - 4) & -3;
|
|
|
|
store(r, VT_LOCAL, loc);
|
|
|
|
l = loc;
|
|
|
|
|
|
|
|
/* modify all stack values */
|
|
|
|
for(p=vstack;p<vstack_ptr;p+=2) {
|
|
|
|
i = p[0] & VT_VALMASK;
|
|
|
|
if (i == r) {
|
|
|
|
if (p[0] & VT_LVAL)
|
|
|
|
t = VT_LLOCAL;
|
|
|
|
else
|
|
|
|
t = VT_LOCAL;
|
|
|
|
p[0] = (p[0] & VT_TYPE) | VT_LVAL | t;
|
|
|
|
p[1] = l;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* save r to memory. and mark it as being free */
|
|
|
|
void save_reg(r)
|
|
|
|
{
|
|
|
|
int i, *p;
|
|
|
|
|
|
|
|
/* modify all stack values */
|
|
|
|
for(p=vstack;p<vstack_ptr;p+=2) {
|
|
|
|
i = p[0] & VT_VALMASK;
|
|
|
|
if (i == r) {
|
|
|
|
save_reg_forced(r);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find a free register. If none, save one register */
|
|
|
|
int get_reg()
|
|
|
|
{
|
|
|
|
int r, i, *p;
|
|
|
|
|
|
|
|
/* find a free register */
|
|
|
|
for(r=0;r<NB_REGS;r++) {
|
|
|
|
for(p=vstack;p<vstack_ptr;p+=2) {
|
|
|
|
i = p[0] & VT_VALMASK;
|
|
|
|
if (i == r)
|
|
|
|
goto notfound;
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
notfound: ;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* no register left : free the first one on the stack (very
|
|
|
|
important to start from the bottom to ensure that we don't
|
|
|
|
spill registers used in gen_op()) */
|
|
|
|
for(p=vstack;p<vstack_ptr;p+=2) {
|
|
|
|
r = p[0] & VT_VALMASK;
|
|
|
|
if (r < VT_CONST) {
|
|
|
|
save_reg(r);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
void save_regs()
|
|
|
|
{
|
|
|
|
int r, *p;
|
|
|
|
for(p=vstack;p<vstack_ptr;p+=2) {
|
|
|
|
r = p[0] & VT_VALMASK;
|
|
|
|
if (r < VT_CONST) {
|
|
|
|
save_reg(r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* move register 's' to 'r', and flush previous value of r to memory
|
|
|
|
if needed */
|
|
|
|
void move_reg(r, s)
|
|
|
|
{
|
|
|
|
if (r != s) {
|
|
|
|
save_reg(r);
|
|
|
|
load(r, s, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* convert a stack entry in register */
|
|
|
|
int gvp(int *p)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
r = p[0] & VT_VALMASK;
|
|
|
|
if (r >= VT_CONST || (p[0] & VT_LVAL))
|
|
|
|
r = get_reg();
|
|
|
|
load(r, p[0], p[1]);
|
|
|
|
p[0] = (p[0] & VT_TYPE) | r;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
void vpush()
|
|
|
|
{
|
|
|
|
*vstack_ptr++ = vt;
|
|
|
|
*vstack_ptr++ = vc;
|
|
|
|
/* cannot let cpu flags if other instruction are generated */
|
|
|
|
if ((vt & VT_VALMASK) == VT_CMP)
|
|
|
|
gvp(vstack_ptr - 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void vpop(int *ft, int *fc)
|
|
|
|
{
|
|
|
|
*fc = *--vstack_ptr;
|
|
|
|
*ft = *--vstack_ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* generate a value in a register from vt and vc */
|
|
|
|
int gv()
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
vpush();
|
|
|
|
r = gvp(vstack_ptr - 2);
|
|
|
|
vpop(&vt, &vc);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle constant optimizations and various machine independant opt */
|
|
|
|
void gen_opc(op)
|
|
|
|
{
|
|
|
|
int fr, ft, fc, r, c1, c2, n;
|
|
|
|
|
|
|
|
vpop(&ft, &fc);
|
|
|
|
vpop(&vt, &vc);
|
|
|
|
c1 = (vt & (VT_VALMASK | VT_LVAL)) == VT_CONST;
|
|
|
|
c2 = (ft & (VT_VALMASK | VT_LVAL)) == VT_CONST;
|
|
|
|
if (c1 && c2) {
|
|
|
|
switch(op) {
|
|
|
|
case '+': vc += fc; break;
|
|
|
|
case '-': vc -= fc; break;
|
|
|
|
case '&': vc &= fc; break;
|
|
|
|
case '^': vc ^= fc; break;
|
|
|
|
case '|': vc |= fc; break;
|
|
|
|
case '*': vc *= fc; break;
|
|
|
|
case TOK_PDIV:
|
|
|
|
case '/': vc /= fc; break; /* XXX: zero case ? */
|
|
|
|
case '%': vc %= fc; break; /* XXX: zero case ? */
|
|
|
|
case TOK_UDIV: vc = (unsigned)vc / fc; break; /* XXX: zero case ? */
|
|
|
|
case TOK_UMOD: vc = (unsigned)vc % fc; break; /* XXX: zero case ? */
|
|
|
|
case TOK_SHL: vc <<= fc; break;
|
|
|
|
case TOK_SHR: vc = (unsigned)vc >> fc; break;
|
|
|
|
case TOK_SAR: vc >>= fc; break;
|
|
|
|
default:
|
|
|
|
goto general_case;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* if commutative ops, put c2 as constant */
|
|
|
|
if (c1 && (op == '+' || op == '&' || op == '^' ||
|
|
|
|
op == '|' || op == '*')) {
|
|
|
|
swap(&vt, &ft);
|
|
|
|
swap(&vc, &fc);
|
|
|
|
swap(&c1, &c2);
|
|
|
|
}
|
|
|
|
if (c2 && (((op == '*' || op == '/' || op == TOK_UDIV ||
|
|
|
|
op == TOK_PDIV) &&
|
|
|
|
fc == 1) ||
|
|
|
|
((op == '+' || op == '-' || op == '|' || op == '^' ||
|
|
|
|
op == TOK_SHL || op == TOK_SHR || op == TOK_SAR) &&
|
|
|
|
fc == 0) ||
|
|
|
|
(op == '&' &&
|
|
|
|
fc == -1))) {
|
|
|
|
} else if (c2 && (op == '*' || op == TOK_PDIV || op == TOK_UDIV)) {
|
|
|
|
/* try to use shifts instead of muls or divs */
|
|
|
|
if (fc > 0 && (fc & (fc - 1)) == 0) {
|
|
|
|
n = -1;
|
|
|
|
while (fc) {
|
|
|
|
fc >>= 1;
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
fc = n;
|
|
|
|
if (op == '*')
|
|
|
|
op = TOK_SHL;
|
|
|
|
else if (op == TOK_PDIV)
|
|
|
|
op = TOK_SAR;
|
|
|
|
else
|
|
|
|
op = TOK_SHR;
|
|
|
|
}
|
|
|
|
goto general_case;
|
|
|
|
} else {
|
|
|
|
general_case:
|
|
|
|
vpush();
|
|
|
|
vt = ft;
|
|
|
|
vc = fc;
|
|
|
|
vpush();
|
|
|
|
r = gvp(vstack_ptr - 4);
|
|
|
|
fr = gvp(vstack_ptr - 2);
|
|
|
|
vpop(&ft, &fc);
|
|
|
|
vpop(&vt, &vc);
|
|
|
|
/* call low level op generator */
|
|
|
|
gen_op1(op, r, fr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int pointed_size(t)
|
|
|
|
{
|
|
|
|
return type_size(pointed_type(t), &t);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* generic gen_op: handles types problems */
|
|
|
|
void gen_op(op)
|
|
|
|
{
|
|
|
|
int u, t1, t2;
|
|
|
|
|
|
|
|
vpush();
|
|
|
|
t1 = vstack_ptr[-4];
|
|
|
|
t2 = vstack_ptr[-2];
|
|
|
|
if (op == '+' | op == '-') {
|
|
|
|
if ((t1 & VT_PTR) && (t2 & VT_PTR)) {
|
|
|
|
if (op != '-')
|
|
|
|
error("invalid type");
|
|
|
|
/* XXX: check that types are compatible */
|
|
|
|
u = pointed_size(t1);
|
|
|
|
gen_opc(op);
|
|
|
|
vpush();
|
|
|
|
vstack_ptr[-2] &= ~VT_TYPE; /* set to integer */
|
|
|
|
vset(VT_CONST, u);
|
|
|
|
gen_op(TOK_PDIV);
|
|
|
|
} else if ((t1 | t2) & VT_PTR) {
|
|
|
|
if (t2 & VT_PTR) {
|
|
|
|
swap(vstack_ptr - 4, vstack_ptr - 2);
|
|
|
|
swap(vstack_ptr - 3, vstack_ptr - 1);
|
|
|
|
swap(&t1, &t2);
|
|
|
|
}
|
|
|
|
/* stack-4 contains pointer, stack-2 value to add */
|
|
|
|
vset(VT_CONST, pointed_size(vstack_ptr[-4]));
|
|
|
|
gen_op('*');
|
|
|
|
vpush();
|
|
|
|
gen_opc(op);
|
|
|
|
/* put again type if gen_opc() swaped operands */
|
|
|
|
vt = (vt & VT_TYPEN) | (t1 & VT_TYPE);
|
|
|
|
} else {
|
|
|
|
gen_opc(op);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((t1 | t2) & VT_UNSIGNED) {
|
|
|
|
if (op == TOK_SAR)
|
|
|
|
op = TOK_SHR;
|
|
|
|
else if (op == '/')
|
|
|
|
op = TOK_UDIV;
|
|
|
|
else if (op == '%')
|
|
|
|
op = TOK_UMOD;
|
|
|
|
}
|
|
|
|
gen_opc(op);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
/* return type size. Put alignment at 'a' */
|
|
|
|
int type_size(int t, int *a)
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *s;
|
|
|
|
|
|
|
|
/* int, enum or pointer */
|
2001-11-06 09:20:20 +08:00
|
|
|
if (t & VT_STRUCT) {
|
2001-11-01 22:48:10 +08:00
|
|
|
/* struct/union */
|
|
|
|
s = sym_find(((unsigned)t >> VT_STRUCT_SHIFT) | SYM_STRUCT);
|
|
|
|
*a = 4; /* XXX: cannot store it yet. Doing that is safe */
|
|
|
|
return s->c;
|
2001-11-06 09:20:20 +08:00
|
|
|
} else if (t & VT_ARRAY) {
|
|
|
|
s = sym_find(((unsigned)t >> VT_STRUCT_SHIFT));
|
|
|
|
return type_size(s->t, a) * s->c;
|
|
|
|
} else if ((t & VT_PTR) |
|
|
|
|
(t & VT_TYPE) == 0 |
|
|
|
|
(t & VT_ENUM)) {
|
|
|
|
*a = 4;
|
|
|
|
return 4;
|
2001-11-01 22:48:10 +08:00
|
|
|
} else {
|
|
|
|
*a = 1;
|
2001-10-28 07:48:39 +08:00
|
|
|
return 1;
|
2001-11-01 22:48:10 +08:00
|
|
|
}
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
/* return the pointed type of t */
|
|
|
|
int pointed_type(int t)
|
2001-10-30 07:23:01 +08:00
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
Sym *s;
|
|
|
|
s = sym_find(((unsigned)t >> VT_STRUCT_SHIFT));
|
|
|
|
return s->t | (t & VT_TYPEN);
|
2001-10-30 07:23:01 +08:00
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
int mk_pointer(int t)
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int p;
|
|
|
|
p = anon_sym++;
|
|
|
|
sym_push(p, t, -1);
|
|
|
|
return VT_PTR | (p << VT_STRUCT_SHIFT) | (t & VT_TYPEN);
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
/* store value in lvalue pushed on stack */
|
|
|
|
void vstore()
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int ft, fc, r, t;
|
|
|
|
|
|
|
|
r = gv(); /* generate value */
|
|
|
|
vpush();
|
|
|
|
ft = vstack_ptr[-4];
|
|
|
|
fc = vstack_ptr[-3];
|
|
|
|
if ((ft & VT_VALMASK) == VT_LLOCAL) {
|
|
|
|
t = get_reg();
|
|
|
|
load(t, VT_LOCAL | VT_LVAL, fc);
|
|
|
|
ft = (ft & ~VT_VALMASK) | t;
|
2001-10-28 09:15:41 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
store(r, ft, fc);
|
|
|
|
vstack_ptr -= 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* post defines POST/PRE add. c is the token ++ or -- */
|
|
|
|
void inc(post, c)
|
|
|
|
{
|
|
|
|
int r, r1;
|
|
|
|
test_lvalue();
|
|
|
|
if (post)
|
|
|
|
vpush(); /* room for returned value */
|
|
|
|
vpush(); /* save lvalue */
|
|
|
|
r = gv();
|
|
|
|
vpush(); /* save value */
|
|
|
|
if (post) {
|
|
|
|
/* duplicate value */
|
|
|
|
r1 = get_reg();
|
|
|
|
load(r1, r, 0); /* move r to r1 */
|
|
|
|
vstack_ptr[-6] = (vt & VT_TYPE) | r1;
|
|
|
|
vstack_ptr[-5] = 0;
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
/* add constant */
|
|
|
|
vset(VT_CONST, c - TOK_MID);
|
|
|
|
gen_op('+');
|
|
|
|
vstore(); /* store value */
|
|
|
|
if (post)
|
|
|
|
vpop(&vt, &vc);
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
|
2001-10-31 06:43:19 +08:00
|
|
|
int expr_const()
|
|
|
|
{
|
|
|
|
expr_eq();
|
|
|
|
if ((vt & (VT_CONST | VT_LVAL)) != VT_CONST)
|
2001-11-01 22:48:10 +08:00
|
|
|
expect("constant");
|
2001-10-31 06:43:19 +08:00
|
|
|
return vc;
|
|
|
|
}
|
|
|
|
|
2001-11-01 22:48:10 +08:00
|
|
|
/* enum/struct/union declaration */
|
|
|
|
int struct_decl(u)
|
|
|
|
{
|
|
|
|
int a, t, b, v, size, align, maxalign, c;
|
|
|
|
Sym *slast, *s, *ss;
|
|
|
|
|
|
|
|
a = tok; /* save decl type */
|
|
|
|
next();
|
|
|
|
if (tok != '{') {
|
|
|
|
v = tok;
|
|
|
|
next();
|
|
|
|
/* struct already defined ? return it */
|
|
|
|
/* XXX: check consistency */
|
|
|
|
if (s = sym_find(v | SYM_STRUCT)) {
|
|
|
|
if (s->t != a)
|
|
|
|
error("invalid type");
|
|
|
|
u = u | (v << VT_STRUCT_SHIFT);
|
|
|
|
return u;
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
} else {
|
|
|
|
v = anon_sym++;
|
2001-11-01 22:48:10 +08:00
|
|
|
}
|
|
|
|
s = sym_push(v | SYM_STRUCT, a, 0);
|
|
|
|
/* put struct/union/enum name in type */
|
|
|
|
u = u | (v << VT_STRUCT_SHIFT);
|
|
|
|
|
|
|
|
if (tok == '{') {
|
|
|
|
next();
|
|
|
|
/* cannot be empty */
|
|
|
|
c = 0;
|
|
|
|
maxalign = 0;
|
|
|
|
slast = 0;
|
|
|
|
while (1) {
|
|
|
|
if (a == TOK_ENUM) {
|
|
|
|
v = tok;
|
|
|
|
next();
|
|
|
|
if (tok == '=') {
|
|
|
|
next();
|
|
|
|
c = expr_const();
|
|
|
|
}
|
|
|
|
sym_push(v, VT_CONST, c);
|
|
|
|
if (tok == ',')
|
|
|
|
next();
|
|
|
|
c++;
|
|
|
|
} else {
|
|
|
|
b = ist();
|
|
|
|
while (1) {
|
2001-11-06 09:20:20 +08:00
|
|
|
t = typ(&v, b);
|
2001-11-01 22:48:10 +08:00
|
|
|
if (t & (VT_FUNC | VT_TYPEDEF))
|
|
|
|
error("invalid type");
|
|
|
|
/* XXX: align & correct type size */
|
|
|
|
v |= SYM_FIELD;
|
|
|
|
size = type_size(t, &align);
|
|
|
|
if (a == TOK_STRUCT) {
|
|
|
|
c = (c + align - 1) & -align;
|
|
|
|
ss = sym_push(v, t, c);
|
|
|
|
c += size;
|
|
|
|
} else {
|
|
|
|
ss = sym_push(v, t, 0);
|
|
|
|
if (size > c)
|
|
|
|
c = size;
|
|
|
|
}
|
|
|
|
if (align > maxalign)
|
|
|
|
maxalign = align;
|
|
|
|
ss->next = slast;
|
|
|
|
slast = ss;
|
|
|
|
if (tok == ';' || tok == -1)
|
|
|
|
break;
|
|
|
|
skip(',');
|
|
|
|
}
|
|
|
|
skip(';');
|
|
|
|
}
|
|
|
|
if (tok == '}')
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
skip('}');
|
|
|
|
s->next = slast;
|
|
|
|
/* size for struct/union, dummy for enum */
|
|
|
|
s->c = (c + maxalign - 1) & -maxalign;
|
|
|
|
}
|
|
|
|
return u;
|
|
|
|
}
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
/* return 0 if no type declaration. otherwise, return the basic type
|
|
|
|
and skip it.
|
|
|
|
XXX: A '2' is ored to ensure non zero return if int type.
|
|
|
|
*/
|
|
|
|
int ist()
|
|
|
|
{
|
2001-10-31 06:43:19 +08:00
|
|
|
int t, n, v;
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *s;
|
2001-10-31 06:43:19 +08:00
|
|
|
|
2001-10-30 07:23:01 +08:00
|
|
|
t = 0;
|
|
|
|
while(1) {
|
2001-11-01 22:48:10 +08:00
|
|
|
#ifndef TINY
|
2001-10-31 06:43:19 +08:00
|
|
|
if (tok == TOK_ENUM) {
|
2001-11-06 09:20:20 +08:00
|
|
|
t |= struct_decl(VT_ENUM);
|
2001-11-01 22:48:10 +08:00
|
|
|
} else if (tok == TOK_STRUCT || tok == TOK_UNION) {
|
2001-11-06 09:20:20 +08:00
|
|
|
t |= struct_decl(VT_STRUCT);
|
2001-11-01 22:48:10 +08:00
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
if (tok == TOK_CHAR) {
|
2001-10-31 06:43:19 +08:00
|
|
|
t |= VT_BYTE;
|
2001-11-06 09:20:20 +08:00
|
|
|
} else if (tok == TOK_VOID) {
|
|
|
|
t |= VT_VOID;
|
2001-10-31 06:43:19 +08:00
|
|
|
} else if (tok == TOK_INT |
|
2001-11-06 09:20:20 +08:00
|
|
|
(tok >= TOK_CONST & tok <= TOK_INLINE)) {
|
2001-10-31 06:43:19 +08:00
|
|
|
/* ignored types */
|
2001-11-01 22:48:10 +08:00
|
|
|
} else if (tok == TOK_FLOAT & tok == TOK_DOUBLE) {
|
|
|
|
error("floats not supported");
|
2001-10-31 06:43:19 +08:00
|
|
|
} else if (tok == TOK_EXTERN) {
|
|
|
|
t |= VT_EXTERN;
|
|
|
|
} else if (tok == TOK_STATIC) {
|
|
|
|
t |= VT_STATIC;
|
|
|
|
} else if (tok == TOK_UNSIGNED) {
|
|
|
|
t |= VT_UNSIGNED;
|
|
|
|
} else if (tok == TOK_TYPEDEF) {
|
|
|
|
t |= VT_TYPEDEF;
|
|
|
|
} else {
|
2001-11-01 22:48:10 +08:00
|
|
|
s = sym_find(tok);
|
|
|
|
if (!s || !(s->t & VT_TYPEDEF))
|
2001-10-31 06:43:19 +08:00
|
|
|
break;
|
2001-11-01 22:48:10 +08:00
|
|
|
t = s->t & ~VT_TYPEDEF;
|
2001-10-31 06:43:19 +08:00
|
|
|
}
|
|
|
|
next();
|
2001-10-30 07:23:01 +08:00
|
|
|
}
|
|
|
|
t |= 2;
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-10-30 07:23:01 +08:00
|
|
|
return t;
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
int post_type(u, t)
|
|
|
|
{
|
|
|
|
int p, n, pt, l;
|
|
|
|
|
|
|
|
if (tok == '(') {
|
|
|
|
/* function declaration */
|
|
|
|
next();
|
|
|
|
/* push a dummy symbol to force local symbol stack usage */
|
|
|
|
sym_push1(&local_stack, 0, 0, 0);
|
|
|
|
p = 4;
|
|
|
|
l = 0;
|
|
|
|
while (tok != ')') {
|
|
|
|
/* read param name and compute offset */
|
|
|
|
if (l != 2) {
|
|
|
|
if (!(pt = ist())) {
|
|
|
|
if (l) {
|
|
|
|
error("invalid type");
|
|
|
|
} else {
|
|
|
|
l = 2;
|
|
|
|
goto old_proto;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (pt & VT_VOID && tok == ')')
|
|
|
|
break;
|
|
|
|
l = 1;
|
|
|
|
pt = typ(&n, pt); /* XXX: should accept
|
|
|
|
both arg/non arg if v == 0 */
|
|
|
|
} else {
|
|
|
|
old_proto:
|
|
|
|
n = tok;
|
|
|
|
pt = 0; /* int type */
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
/* array must be transformed to pointer according to ANSI C */
|
|
|
|
pt &= ~VT_ARRAY;
|
|
|
|
p = p + 4;
|
|
|
|
sym_push(n, VT_LOCAL | VT_LVAL | pt, p);
|
|
|
|
if (tok == ',') {
|
|
|
|
next();
|
|
|
|
if (l == 1 && tok == TOK_DOTS) {
|
|
|
|
next();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
skip(')');
|
|
|
|
t = post_type(0, t); /* XXX: may be incorrect */
|
|
|
|
/* transform function pointer to 'char *' */
|
|
|
|
if (u)
|
|
|
|
t = u + VT_BYTE;
|
|
|
|
else
|
|
|
|
t = t | VT_FUNC;
|
|
|
|
} else if (tok == '[') {
|
|
|
|
/* array definition */
|
|
|
|
next();
|
|
|
|
n = -1;
|
|
|
|
if (tok != ']') {
|
|
|
|
n = expr_const();
|
|
|
|
if (n < 0)
|
|
|
|
error("invalid array size");
|
|
|
|
}
|
|
|
|
skip(']');
|
|
|
|
/* parse next post type */
|
|
|
|
t = post_type(u, t);
|
|
|
|
|
|
|
|
/* we push a anonymous symbol which will contain the array
|
|
|
|
element type */
|
|
|
|
p = anon_sym++;
|
|
|
|
sym_push(p, t, n);
|
|
|
|
t = VT_ARRAY | VT_PTR | (p << VT_STRUCT_SHIFT);
|
|
|
|
}
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
/* Read a type declaration (except basic type), and return the
|
|
|
|
type. If v is true, then also put variable name in 'vc' */
|
2001-11-06 09:20:20 +08:00
|
|
|
int typ(int *v, int t)
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int u;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
|
|
|
t = t & -3; /* suppress the ored '2' */
|
|
|
|
while (tok == '*') {
|
|
|
|
next();
|
2001-11-06 09:20:20 +08:00
|
|
|
t = mk_pointer(t);
|
|
|
|
}
|
2001-10-28 07:48:39 +08:00
|
|
|
|
|
|
|
/* recursive type */
|
|
|
|
/* XXX: incorrect if abstract type for functions (e.g. 'int ()') */
|
|
|
|
if (tok == '(') {
|
|
|
|
next();
|
2001-11-06 09:20:20 +08:00
|
|
|
u = typ(v, 0);
|
2001-10-28 07:48:39 +08:00
|
|
|
skip(')');
|
|
|
|
} else {
|
|
|
|
u = 0;
|
|
|
|
/* type identifier */
|
|
|
|
if (v) {
|
2001-10-30 07:23:01 +08:00
|
|
|
*v = tok;
|
2001-10-28 07:48:39 +08:00
|
|
|
next();
|
|
|
|
}
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
return post_type(u, t);
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
|
2001-10-30 07:23:01 +08:00
|
|
|
/* define a new external reference to a function 'v' of type 'u' */
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *external_func(v, u)
|
2001-10-30 07:23:01 +08:00
|
|
|
{
|
|
|
|
int t, n;
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *s;
|
|
|
|
s = sym_find(v);
|
|
|
|
if (!s) {
|
2001-10-30 07:23:01 +08:00
|
|
|
n = dlsym(0, get_tok_str(v));
|
|
|
|
if (n == 0) {
|
2001-11-01 22:48:10 +08:00
|
|
|
/* used to generate symbol list */
|
|
|
|
s = sym_push1(&global_stack,
|
|
|
|
v, u | VT_CONST | VT_LVAL | VT_FORWARD, 0);
|
2001-10-30 07:23:01 +08:00
|
|
|
} else {
|
2001-11-01 22:48:10 +08:00
|
|
|
/* int f() */
|
|
|
|
s = sym_push1(&global_stack,
|
|
|
|
v, u | VT_CONST | VT_LVAL, n);
|
2001-10-30 07:23:01 +08:00
|
|
|
}
|
|
|
|
}
|
2001-11-01 22:48:10 +08:00
|
|
|
return s;
|
2001-10-30 07:23:01 +08:00
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
/* read a character for string or char constant and eval escape codes */
|
|
|
|
int getq()
|
2001-10-28 09:15:41 +08:00
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int c;
|
|
|
|
|
|
|
|
c = ch;
|
|
|
|
minp();
|
|
|
|
if (c == '\\') {
|
|
|
|
if (isnum(ch)) {
|
|
|
|
return getn(8);
|
|
|
|
} else {
|
|
|
|
if (ch == 'n')
|
|
|
|
c = '\n';
|
|
|
|
else if (ch == 'r')
|
|
|
|
c = '\r';
|
|
|
|
else if (ch == 't')
|
|
|
|
c = '\t';
|
|
|
|
else
|
|
|
|
c = ch;
|
|
|
|
minp();
|
|
|
|
}
|
2001-10-28 09:15:41 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
return c;
|
2001-10-28 09:15:41 +08:00
|
|
|
}
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
void indir()
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
if (vt & VT_LVAL)
|
|
|
|
gv();
|
|
|
|
if (!(vt & VT_PTR))
|
|
|
|
expect("pointer");
|
|
|
|
vt = pointed_type(vt);
|
|
|
|
if (!(vt & VT_ARRAY)) /* an array is never an lvalue */
|
|
|
|
vt |= VT_LVAL;
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void unary()
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int n, t, ft, fc, p, r;
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *s;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
if (tok == TOK_NUM) {
|
|
|
|
vset(VT_CONST, tokc);
|
2001-10-28 09:15:41 +08:00
|
|
|
next();
|
2001-10-28 12:08:38 +08:00
|
|
|
} else
|
|
|
|
if (tok == '\'') {
|
2001-11-06 09:20:20 +08:00
|
|
|
vset(VT_CONST, getq());
|
2001-10-28 07:48:39 +08:00
|
|
|
next(); /* skip char */
|
|
|
|
skip('\'');
|
2001-11-06 09:20:20 +08:00
|
|
|
} else if (tok == '\"') {
|
|
|
|
/* generate (char *) type */
|
|
|
|
vset(VT_CONST | mk_pointer(VT_TYPE), glo);
|
2001-10-28 10:51:52 +08:00
|
|
|
while (tok == '\"') {
|
2001-11-06 09:20:20 +08:00
|
|
|
while (ch != '\"') {
|
|
|
|
if (ch == -1)
|
|
|
|
error("unterminated string");
|
|
|
|
*(char *)glo++ = getq();
|
2001-10-28 10:51:52 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
minp();
|
2001-10-28 10:51:52 +08:00
|
|
|
next();
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-10-30 07:23:01 +08:00
|
|
|
*(char *)glo++ = 0;
|
2001-10-28 07:48:39 +08:00
|
|
|
} else {
|
|
|
|
t = tok;
|
|
|
|
next();
|
|
|
|
if (t == '(') {
|
|
|
|
/* cast ? */
|
|
|
|
if (t = ist()) {
|
2001-11-06 09:20:20 +08:00
|
|
|
ft = typ(0, t);
|
2001-10-28 07:48:39 +08:00
|
|
|
skip(')');
|
|
|
|
unary();
|
|
|
|
vt = (vt & VT_TYPEN) | ft;
|
|
|
|
} else {
|
|
|
|
expr();
|
|
|
|
skip(')');
|
|
|
|
}
|
|
|
|
} else if (t == '*') {
|
|
|
|
unary();
|
2001-11-06 09:20:20 +08:00
|
|
|
indir();
|
2001-10-28 07:48:39 +08:00
|
|
|
} else if (t == '&') {
|
|
|
|
unary();
|
2001-10-28 23:20:12 +08:00
|
|
|
test_lvalue();
|
2001-11-06 09:20:20 +08:00
|
|
|
vt = mk_pointer(vt & VT_LVALN);
|
2001-10-28 07:48:39 +08:00
|
|
|
} else
|
|
|
|
if (t == '!') {
|
|
|
|
unary();
|
2001-11-06 09:20:20 +08:00
|
|
|
if ((vt & VT_VALMASK) == VT_CMP)
|
2001-10-28 07:48:39 +08:00
|
|
|
vc = vc ^ 1;
|
|
|
|
else
|
|
|
|
vset(VT_JMP, gtst(1, 0));
|
|
|
|
} else
|
|
|
|
if (t == '~') {
|
|
|
|
unary();
|
2001-11-06 09:20:20 +08:00
|
|
|
vpush();
|
|
|
|
vset(VT_CONST, -1);
|
|
|
|
gen_op('^');
|
2001-10-28 07:48:39 +08:00
|
|
|
} else
|
2001-10-28 12:08:38 +08:00
|
|
|
if (t == '+') {
|
|
|
|
unary();
|
|
|
|
} else
|
2001-11-06 09:20:20 +08:00
|
|
|
if (t == TOK_SIZEOF) {
|
|
|
|
/* XXX: some code can be generated */
|
|
|
|
if (tok == '(') {
|
|
|
|
next();
|
|
|
|
if (t = ist())
|
|
|
|
vt = typ(0, t);
|
|
|
|
else
|
|
|
|
expr();
|
|
|
|
skip(')');
|
|
|
|
} else {
|
|
|
|
unary();
|
|
|
|
}
|
|
|
|
vset(VT_CONST, type_size(vt, &t));
|
|
|
|
} else
|
2001-10-28 07:48:39 +08:00
|
|
|
if (t == TOK_INC | t == TOK_DEC) {
|
|
|
|
unary();
|
2001-11-06 09:20:20 +08:00
|
|
|
inc(0, t);
|
2001-10-28 07:48:39 +08:00
|
|
|
} else if (t == '-') {
|
2001-11-06 09:20:20 +08:00
|
|
|
vset(VT_CONST, 0);
|
|
|
|
vpush();
|
2001-10-28 07:48:39 +08:00
|
|
|
unary();
|
2001-11-06 09:20:20 +08:00
|
|
|
gen_op('-');
|
2001-10-28 12:08:38 +08:00
|
|
|
} else
|
|
|
|
{
|
2001-11-01 22:48:10 +08:00
|
|
|
s = sym_find(t);
|
|
|
|
if (!s) {
|
2001-10-30 07:23:01 +08:00
|
|
|
if (tok != '(')
|
|
|
|
error("undefined symbol");
|
|
|
|
/* for simple function calls, we tolerate undeclared
|
|
|
|
external reference */
|
2001-11-01 22:48:10 +08:00
|
|
|
s = external_func(t, VT_FUNC); /* int() function */
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
2001-11-01 22:48:10 +08:00
|
|
|
vset(s->t, s->c);
|
|
|
|
/* if forward reference, we must point to s->c */
|
2001-10-30 07:23:01 +08:00
|
|
|
if (vt & VT_FORWARD)
|
2001-11-01 22:48:10 +08:00
|
|
|
vc = (int)&s->c;
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* post operations */
|
2001-11-01 22:48:10 +08:00
|
|
|
while (1) {
|
|
|
|
if (tok == TOK_INC | tok == TOK_DEC) {
|
2001-11-06 09:20:20 +08:00
|
|
|
inc(1, tok);
|
2001-11-01 22:48:10 +08:00
|
|
|
next();
|
|
|
|
} else if (tok == '.' | tok == TOK_ARROW) {
|
|
|
|
/* field */
|
2001-11-06 09:20:20 +08:00
|
|
|
if (tok == TOK_ARROW)
|
|
|
|
indir();
|
|
|
|
test_lvalue();
|
|
|
|
vt &= VT_LVALN;
|
2001-11-01 22:48:10 +08:00
|
|
|
next();
|
|
|
|
/* expect pointer on structure */
|
2001-11-06 09:20:20 +08:00
|
|
|
if (!(vt & VT_STRUCT))
|
2001-11-01 22:48:10 +08:00
|
|
|
expect("struct or union");
|
|
|
|
s = sym_find(((unsigned)vt >> VT_STRUCT_SHIFT) | SYM_STRUCT);
|
|
|
|
/* find field */
|
|
|
|
tok |= SYM_FIELD;
|
|
|
|
while (s = s->next) {
|
|
|
|
if (s->v == tok)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!s)
|
|
|
|
error("field not found");
|
|
|
|
/* add field offset to pointer */
|
2001-11-06 09:20:20 +08:00
|
|
|
vt = vt & VT_TYPEN; /* change type to int */
|
|
|
|
vpush();
|
|
|
|
vset(VT_CONST, s->c);
|
|
|
|
gen_op('+');
|
2001-11-01 22:48:10 +08:00
|
|
|
/* change type to field type, and set to lvalue */
|
|
|
|
vt = (vt & VT_TYPEN) | VT_LVAL | s->t;
|
|
|
|
next();
|
|
|
|
} else if (tok == '[') {
|
2001-11-06 09:20:20 +08:00
|
|
|
next();
|
|
|
|
vpush();
|
|
|
|
expr();
|
|
|
|
gen_op('+');
|
|
|
|
indir();
|
2001-11-01 22:48:10 +08:00
|
|
|
skip(']');
|
|
|
|
} else if (tok == '(') {
|
|
|
|
/* function call */
|
2001-11-06 09:20:20 +08:00
|
|
|
save_regs(); /* save used temporary registers */
|
2001-11-01 22:48:10 +08:00
|
|
|
/* lvalue is implied */
|
|
|
|
vt = vt & VT_LVALN;
|
2001-11-06 09:20:20 +08:00
|
|
|
if ((vt & VT_VALMASK) != VT_CONST) {
|
2001-11-01 22:48:10 +08:00
|
|
|
/* evaluate function address */
|
2001-11-06 09:20:20 +08:00
|
|
|
r = gv();
|
|
|
|
o(0x50 + r); /* push r */
|
2001-11-01 22:48:10 +08:00
|
|
|
}
|
|
|
|
ft = vt;
|
|
|
|
fc = vc;
|
|
|
|
next();
|
|
|
|
t = 0;
|
|
|
|
while (tok != ')') {
|
|
|
|
t = t + 4;
|
|
|
|
expr_eq();
|
2001-11-06 09:20:20 +08:00
|
|
|
r = gv();
|
|
|
|
o(0x50 + r); /* push r */
|
2001-11-01 22:48:10 +08:00
|
|
|
if (tok == ',')
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
skip(')');
|
|
|
|
/* horrible, but needed : convert to native ordering (could
|
|
|
|
parse parameters in reverse order, but would cost more
|
|
|
|
code) */
|
|
|
|
n = 0;
|
|
|
|
p = t - 4;
|
|
|
|
while (n < p) {
|
|
|
|
oad(0x24848b, p); /* mov x(%esp,1), %eax */
|
|
|
|
oad(0x248487, n); /* xchg x(%esp,1), %eax */
|
|
|
|
oad(0x248489, p); /* mov %eax, x(%esp,1) */
|
|
|
|
n = n + 4;
|
|
|
|
p = p - 4;
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
if ((ft & VT_VALMASK) == VT_CONST) {
|
2001-11-01 22:48:10 +08:00
|
|
|
/* forward reference */
|
|
|
|
if (ft & VT_FORWARD) {
|
|
|
|
*(int *)fc = psym(0xe8, *(int *)fc);
|
|
|
|
} else
|
|
|
|
oad(0xe8, fc - ind - 5);
|
2001-11-06 09:20:20 +08:00
|
|
|
/* return value is %eax, and take type from function proto */
|
|
|
|
vt = 0 | (ft & VT_TYPE & VT_FUNCN);
|
2001-11-01 22:48:10 +08:00
|
|
|
} else {
|
|
|
|
oad(0x2494ff, t); /* call *xxx(%esp) */
|
|
|
|
t = t + 4;
|
2001-11-06 09:20:20 +08:00
|
|
|
/* return value is %eax, integer */
|
|
|
|
vt = 0;
|
2001-11-01 22:48:10 +08:00
|
|
|
}
|
|
|
|
if (t)
|
|
|
|
oad(0xc481, t);
|
2001-10-28 07:48:39 +08:00
|
|
|
} else {
|
2001-11-01 22:48:10 +08:00
|
|
|
break;
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void uneq()
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int t;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
|
|
|
unary();
|
2001-10-29 06:24:16 +08:00
|
|
|
if (tok == '=' |
|
2001-11-06 09:20:20 +08:00
|
|
|
(tok >= TOK_A_MOD & tok <= TOK_A_DIV) |
|
2001-10-29 06:24:16 +08:00
|
|
|
tok == TOK_A_XOR | tok == TOK_A_OR |
|
2001-11-06 09:20:20 +08:00
|
|
|
tok == TOK_A_SHL | tok == TOK_A_SAR) {
|
2001-10-28 23:20:12 +08:00
|
|
|
test_lvalue();
|
2001-11-06 09:20:20 +08:00
|
|
|
vpush();
|
|
|
|
t = tok;
|
|
|
|
next();
|
|
|
|
if (t == '=') {
|
2001-10-29 06:24:16 +08:00
|
|
|
expr_eq();
|
2001-11-06 09:20:20 +08:00
|
|
|
/* XXX: be more precise */
|
|
|
|
if ((vt & VT_PTR) != (vstack_ptr[-2] & VT_PTR))
|
2001-10-29 06:24:16 +08:00
|
|
|
warning("incompatible type");
|
2001-11-06 09:20:20 +08:00
|
|
|
} else {
|
|
|
|
vpush();
|
|
|
|
expr_eq();
|
|
|
|
gen_op(t & 0x7f);
|
|
|
|
}
|
|
|
|
vstore();
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void sum(l)
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int ft, fc, t;
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
if (l == 0)
|
|
|
|
uneq();
|
|
|
|
else {
|
2001-10-28 09:15:41 +08:00
|
|
|
sum(--l);
|
|
|
|
while ((l == 0 & (tok == '*' | tok == '/' | tok == '%')) |
|
|
|
|
(l == 1 & (tok == '+' | tok == '-')) |
|
2001-11-06 09:20:20 +08:00
|
|
|
(l == 2 & (tok == TOK_SHL | tok == TOK_SAR)) |
|
2001-10-28 09:15:41 +08:00
|
|
|
(l == 3 & (tok >= TOK_LT & tok <= TOK_GT)) |
|
|
|
|
(l == 4 & (tok == TOK_EQ | tok == TOK_NE)) |
|
|
|
|
(l == 5 & tok == '&') |
|
|
|
|
(l == 6 & tok == '^') |
|
|
|
|
(l == 7 & tok == '|')) {
|
2001-11-06 09:20:20 +08:00
|
|
|
vpush();
|
|
|
|
t = tok;
|
|
|
|
next();
|
|
|
|
sum(l);
|
|
|
|
gen_op(t);
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef TINY
|
|
|
|
void expr()
|
|
|
|
{
|
|
|
|
sum(8);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void eand()
|
|
|
|
{
|
|
|
|
int t;
|
|
|
|
|
|
|
|
sum(8);
|
|
|
|
t = 0;
|
|
|
|
while (1) {
|
|
|
|
if (tok != TOK_LAND) {
|
|
|
|
if (t) {
|
|
|
|
t = gtst(1, t);
|
2001-11-06 09:20:20 +08:00
|
|
|
vset(VT_JMPI, t);
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
t = gtst(1, t);
|
|
|
|
next();
|
|
|
|
sum(8);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-10-28 10:51:52 +08:00
|
|
|
void eor()
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
|
|
|
int t, u;
|
|
|
|
|
|
|
|
eand();
|
|
|
|
t = 0;
|
|
|
|
while (1) {
|
|
|
|
if (tok != TOK_LOR) {
|
|
|
|
if (t) {
|
|
|
|
t = gtst(0, t);
|
|
|
|
vset(VT_JMP, t);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
t = gtst(0, t);
|
|
|
|
next();
|
|
|
|
eand();
|
|
|
|
}
|
|
|
|
}
|
2001-10-28 10:51:52 +08:00
|
|
|
|
|
|
|
void expr_eq()
|
|
|
|
{
|
|
|
|
int t, u;
|
|
|
|
|
|
|
|
eor();
|
|
|
|
if (tok == '?') {
|
|
|
|
next();
|
|
|
|
t = gtst(1, 0);
|
|
|
|
expr();
|
|
|
|
gv();
|
|
|
|
skip(':');
|
2001-11-06 09:20:20 +08:00
|
|
|
u = gjmp(0);
|
2001-10-28 10:51:52 +08:00
|
|
|
gsym(t);
|
|
|
|
expr_eq();
|
|
|
|
gv();
|
|
|
|
gsym(u);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void expr()
|
|
|
|
{
|
|
|
|
while (1) {
|
|
|
|
expr_eq();
|
|
|
|
if (tok != ',')
|
|
|
|
break;
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
#endif
|
|
|
|
|
2001-11-06 09:20:20 +08:00
|
|
|
void block(int *bsym, int *csym, int *case_sym, int *def_sym, int case_reg)
|
2001-10-28 07:48:39 +08:00
|
|
|
{
|
2001-10-31 04:27:54 +08:00
|
|
|
int a, b, c, d;
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *s;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
|
|
|
if (tok == TOK_IF) {
|
|
|
|
/* if test */
|
|
|
|
next();
|
|
|
|
skip('(');
|
|
|
|
expr();
|
|
|
|
skip(')');
|
|
|
|
a = gtst(1, 0);
|
2001-11-06 09:20:20 +08:00
|
|
|
block(bsym, csym, case_sym, def_sym, case_reg);
|
2001-10-28 07:48:39 +08:00
|
|
|
c = tok;
|
|
|
|
if (c == TOK_ELSE) {
|
|
|
|
next();
|
2001-11-06 09:20:20 +08:00
|
|
|
d = gjmp(0);
|
2001-10-28 07:48:39 +08:00
|
|
|
gsym(a);
|
2001-11-06 09:20:20 +08:00
|
|
|
block(bsym, csym, case_sym, def_sym, case_reg);
|
2001-10-28 07:48:39 +08:00
|
|
|
gsym(d); /* patch else jmp */
|
|
|
|
} else
|
|
|
|
gsym(a);
|
|
|
|
} else if (tok == TOK_WHILE) {
|
|
|
|
next();
|
|
|
|
d = ind;
|
|
|
|
skip('(');
|
|
|
|
expr();
|
|
|
|
skip(')');
|
2001-10-31 04:27:54 +08:00
|
|
|
a = gtst(1, 0);
|
|
|
|
b = 0;
|
2001-11-06 09:20:20 +08:00
|
|
|
block(&a, &b, case_sym, def_sym, case_reg);
|
2001-10-28 07:48:39 +08:00
|
|
|
oad(0xe9, d - ind - 5); /* jmp */
|
2001-10-31 04:27:54 +08:00
|
|
|
gsym(a);
|
|
|
|
gsym_addr(b, d);
|
2001-10-28 07:48:39 +08:00
|
|
|
} else if (tok == '{') {
|
|
|
|
next();
|
|
|
|
/* declarations */
|
2001-11-01 22:48:10 +08:00
|
|
|
s = local_stack;
|
2001-10-28 07:48:39 +08:00
|
|
|
decl(VT_LOCAL);
|
|
|
|
while (tok != '}')
|
2001-11-06 09:20:20 +08:00
|
|
|
block(bsym, csym, case_sym, def_sym, case_reg);
|
2001-11-01 22:48:10 +08:00
|
|
|
/* pop locally defined symbols */
|
|
|
|
sym_pop(&local_stack, s);
|
2001-10-28 07:48:39 +08:00
|
|
|
next();
|
|
|
|
} else if (tok == TOK_RETURN) {
|
|
|
|
next();
|
|
|
|
if (tok != ';') {
|
|
|
|
expr();
|
2001-11-06 09:20:20 +08:00
|
|
|
move_reg(0, gv());
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
skip(';');
|
2001-11-06 09:20:20 +08:00
|
|
|
rsym = gjmp(rsym); /* jmp */
|
2001-10-28 07:48:39 +08:00
|
|
|
} else if (tok == TOK_BREAK) {
|
|
|
|
/* compute jump */
|
2001-10-31 04:27:54 +08:00
|
|
|
if (!bsym)
|
|
|
|
error("cannot break");
|
2001-11-06 09:20:20 +08:00
|
|
|
*bsym = gjmp(*bsym);
|
2001-10-31 04:27:54 +08:00
|
|
|
next();
|
|
|
|
skip(';');
|
|
|
|
} else if (tok == TOK_CONTINUE) {
|
|
|
|
/* compute jump */
|
|
|
|
if (!csym)
|
|
|
|
error("cannot continue");
|
2001-11-06 09:20:20 +08:00
|
|
|
*csym = gjmp(*csym);
|
2001-10-28 07:48:39 +08:00
|
|
|
next();
|
|
|
|
skip(';');
|
2001-10-28 10:51:52 +08:00
|
|
|
} else
|
|
|
|
#ifndef TINY
|
|
|
|
if (tok == TOK_FOR) {
|
|
|
|
int e;
|
|
|
|
next();
|
|
|
|
skip('(');
|
|
|
|
if (tok != ';')
|
|
|
|
expr();
|
|
|
|
skip(';');
|
|
|
|
d = ind;
|
|
|
|
c = ind;
|
|
|
|
a = 0;
|
2001-10-31 04:27:54 +08:00
|
|
|
b = 0;
|
2001-10-28 10:51:52 +08:00
|
|
|
if (tok != ';') {
|
|
|
|
expr();
|
|
|
|
a = gtst(1, 0);
|
|
|
|
}
|
|
|
|
skip(';');
|
|
|
|
if (tok != ')') {
|
2001-11-06 09:20:20 +08:00
|
|
|
e = gjmp(0);
|
2001-10-28 10:51:52 +08:00
|
|
|
c = ind;
|
|
|
|
expr();
|
|
|
|
oad(0xe9, d - ind - 5); /* jmp */
|
|
|
|
gsym(e);
|
|
|
|
}
|
|
|
|
skip(')');
|
2001-11-06 09:20:20 +08:00
|
|
|
block(&a, &b, case_sym, def_sym, case_reg);
|
2001-10-28 10:51:52 +08:00
|
|
|
oad(0xe9, c - ind - 5); /* jmp */
|
2001-10-31 04:27:54 +08:00
|
|
|
gsym(a);
|
|
|
|
gsym_addr(b, c);
|
2001-10-28 10:51:52 +08:00
|
|
|
} else
|
2001-10-31 04:27:54 +08:00
|
|
|
if (tok == TOK_DO) {
|
|
|
|
next();
|
|
|
|
a = 0;
|
|
|
|
b = 0;
|
|
|
|
d = ind;
|
2001-11-06 09:20:20 +08:00
|
|
|
block(&a, &b, case_sym, def_sym, case_reg);
|
2001-10-31 04:27:54 +08:00
|
|
|
skip(TOK_WHILE);
|
|
|
|
skip('(');
|
|
|
|
gsym(b);
|
|
|
|
expr();
|
|
|
|
c = gtst(0, 0);
|
|
|
|
gsym_addr(c, d);
|
|
|
|
skip(')');
|
|
|
|
gsym(a);
|
|
|
|
} else
|
2001-10-31 06:43:19 +08:00
|
|
|
if (tok == TOK_SWITCH) {
|
|
|
|
next();
|
|
|
|
skip('(');
|
|
|
|
expr();
|
2001-11-06 09:20:20 +08:00
|
|
|
case_reg = gv();
|
2001-10-31 06:43:19 +08:00
|
|
|
skip(')');
|
|
|
|
a = 0;
|
|
|
|
b = 0;
|
|
|
|
c = 0;
|
2001-11-06 09:20:20 +08:00
|
|
|
block(&a, csym, &b, &c, case_reg);
|
2001-10-31 06:43:19 +08:00
|
|
|
/* if no default, jmp after switch */
|
|
|
|
if (c == 0)
|
|
|
|
c = ind;
|
|
|
|
/* default label */
|
|
|
|
gsym_addr(b, c);
|
|
|
|
/* break label */
|
|
|
|
gsym(a);
|
|
|
|
} else
|
|
|
|
if (tok == TOK_CASE) {
|
|
|
|
next();
|
|
|
|
a = expr_const();
|
|
|
|
if (!case_sym)
|
2001-11-01 22:48:10 +08:00
|
|
|
expect("switch");
|
2001-10-31 06:43:19 +08:00
|
|
|
gsym(*case_sym);
|
2001-11-06 09:20:20 +08:00
|
|
|
vset(case_reg, 0);
|
|
|
|
vpush();
|
|
|
|
vset(VT_CONST, a);
|
|
|
|
gen_op(TOK_EQ);
|
|
|
|
*case_sym = gtst(1, 0);
|
2001-10-31 06:43:19 +08:00
|
|
|
skip(':');
|
2001-11-06 09:20:20 +08:00
|
|
|
block(bsym, csym, case_sym, def_sym, case_reg);
|
2001-10-31 06:43:19 +08:00
|
|
|
} else
|
|
|
|
if (tok == TOK_DEFAULT) {
|
|
|
|
next();
|
|
|
|
skip(':');
|
|
|
|
if (!def_sym)
|
2001-11-01 22:48:10 +08:00
|
|
|
expect("switch");
|
2001-10-31 06:43:19 +08:00
|
|
|
if (*def_sym)
|
|
|
|
error("too many 'default'");
|
|
|
|
*def_sym = ind;
|
2001-11-06 09:20:20 +08:00
|
|
|
block(bsym, csym, case_sym, def_sym, case_reg);
|
2001-10-31 06:43:19 +08:00
|
|
|
} else
|
|
|
|
if (tok == TOK_GOTO) {
|
|
|
|
next();
|
2001-11-01 22:48:10 +08:00
|
|
|
s = sym_find1(label_stack, tok);
|
|
|
|
/* put forward definition if needed */
|
|
|
|
if (!s)
|
|
|
|
s = sym_push1(&label_stack, tok, VT_FORWARD, 0);
|
2001-10-31 06:43:19 +08:00
|
|
|
/* label already defined */
|
2001-11-01 22:48:10 +08:00
|
|
|
if (s->t & VT_FORWARD)
|
2001-11-06 09:20:20 +08:00
|
|
|
s->c = gjmp(s->c); /* jmp xxx */
|
2001-10-31 06:43:19 +08:00
|
|
|
else
|
2001-11-01 22:48:10 +08:00
|
|
|
oad(0xe9, s->c - ind - 5); /* jmp xxx */
|
2001-10-31 06:43:19 +08:00
|
|
|
next();
|
|
|
|
skip(';');
|
|
|
|
} else
|
2001-10-28 10:51:52 +08:00
|
|
|
#endif
|
|
|
|
{
|
2001-10-31 06:43:19 +08:00
|
|
|
b = tok;
|
|
|
|
next();
|
|
|
|
if (tok == ':') {
|
|
|
|
next();
|
|
|
|
/* label case */
|
2001-11-01 22:48:10 +08:00
|
|
|
s = sym_find1(label_stack, b);
|
|
|
|
if (s) {
|
|
|
|
if (!(s->t & VT_FORWARD))
|
2001-10-31 06:43:19 +08:00
|
|
|
error("multiple defined label");
|
2001-11-01 22:48:10 +08:00
|
|
|
gsym(s->c);
|
|
|
|
s->c = ind;
|
|
|
|
s->t = 0;
|
|
|
|
} else {
|
|
|
|
sym_push1(&label_stack, b, 0, ind);
|
2001-10-31 06:43:19 +08:00
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
block(bsym, csym, case_sym, def_sym, case_reg);
|
2001-10-31 06:43:19 +08:00
|
|
|
} else {
|
|
|
|
/* expression case: go backward of one token */
|
|
|
|
/* XXX: currently incorrect if number/string/char */
|
|
|
|
tok1 = tok;
|
|
|
|
tok = b;
|
|
|
|
if (tok != ';') {
|
|
|
|
expr();
|
|
|
|
}
|
|
|
|
skip(';');
|
|
|
|
}
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-10-30 07:23:01 +08:00
|
|
|
/* 'l' is VT_LOCAL or VT_CONST to define default storage type */
|
2001-10-28 07:48:39 +08:00
|
|
|
void decl(l)
|
|
|
|
{
|
2001-11-06 09:20:20 +08:00
|
|
|
int *a, t, b, size, align, v, u, n;
|
|
|
|
Sym *sym;
|
2001-10-28 07:48:39 +08:00
|
|
|
|
|
|
|
while (b = ist()) {
|
2001-11-01 22:48:10 +08:00
|
|
|
if ((b & (VT_ENUM | VT_STRUCT)) && tok == ';') {
|
2001-10-31 06:43:19 +08:00
|
|
|
/* we accept no variable after */
|
|
|
|
next();
|
|
|
|
continue;
|
|
|
|
}
|
2001-10-28 07:48:39 +08:00
|
|
|
while (1) { /* iterate thru each declaration */
|
2001-11-06 09:20:20 +08:00
|
|
|
t = typ(&v, b);
|
2001-10-28 07:48:39 +08:00
|
|
|
if (tok == '{') {
|
2001-10-30 07:23:01 +08:00
|
|
|
/* patch forward references */
|
2001-11-01 22:48:10 +08:00
|
|
|
if ((sym = sym_find(v)) && (sym->t & VT_FORWARD)) {
|
|
|
|
gsym(sym->c);
|
|
|
|
sym->c = ind;
|
|
|
|
sym->t = VT_CONST | VT_LVAL | t;
|
|
|
|
} else {
|
|
|
|
/* put function address */
|
|
|
|
sym_push1(&global_stack, v, VT_CONST | VT_LVAL | t, ind);
|
|
|
|
}
|
2001-10-28 07:48:39 +08:00
|
|
|
loc = 0;
|
|
|
|
o(0xe58955); /* push %ebp, mov %esp, %ebp */
|
2001-11-01 22:48:10 +08:00
|
|
|
a = (int *)oad(0xec81, 0); /* sub $xxx, %esp */
|
2001-10-28 07:48:39 +08:00
|
|
|
rsym = 0;
|
2001-11-06 09:20:20 +08:00
|
|
|
block(0, 0, 0, 0, 0);
|
2001-10-28 07:48:39 +08:00
|
|
|
gsym(rsym);
|
|
|
|
o(0xc3c9); /* leave, ret */
|
2001-10-30 07:23:01 +08:00
|
|
|
*a = (-loc + 3) & -4; /* align local size to word &
|
|
|
|
save local variables */
|
2001-11-01 22:48:10 +08:00
|
|
|
sym_pop(&label_stack, 0); /* reset label stack */
|
|
|
|
sym_pop(&local_stack, 0); /* reset local stack */
|
2001-10-28 07:48:39 +08:00
|
|
|
break;
|
|
|
|
} else {
|
2001-11-06 09:20:20 +08:00
|
|
|
if (b & VT_TYPEDEF) {
|
2001-11-01 22:48:10 +08:00
|
|
|
/* save typedefed type */
|
2001-11-06 09:20:20 +08:00
|
|
|
sym_push(v, t | VT_TYPEDEF, 0);
|
2001-10-31 06:43:19 +08:00
|
|
|
} else if (t & VT_FUNC) {
|
2001-11-06 09:20:20 +08:00
|
|
|
/* XXX: incorrect to flush, but needed while
|
|
|
|
waiting for function prototypes */
|
2001-10-30 07:23:01 +08:00
|
|
|
/* external function definition */
|
|
|
|
external_func(v, t);
|
2001-10-28 07:48:39 +08:00
|
|
|
} else {
|
2001-10-30 07:23:01 +08:00
|
|
|
/* not lvalue if array */
|
|
|
|
if (!(t & VT_ARRAY))
|
|
|
|
t |= VT_LVAL;
|
|
|
|
if (t & VT_EXTERN) {
|
|
|
|
/* external variable */
|
2001-11-01 22:48:10 +08:00
|
|
|
/* XXX: factorize with external function def */
|
2001-10-30 07:23:01 +08:00
|
|
|
n = dlsym(NULL, get_tok_str(v));
|
|
|
|
if (!n)
|
|
|
|
error("unknown external variable");
|
2001-11-01 22:48:10 +08:00
|
|
|
sym_push(v, VT_CONST | t, n);
|
2001-10-30 07:23:01 +08:00
|
|
|
} else {
|
|
|
|
u = l;
|
|
|
|
if (t & VT_STATIC)
|
|
|
|
u = VT_CONST;
|
2001-11-01 22:48:10 +08:00
|
|
|
u |= t;
|
2001-11-06 09:20:20 +08:00
|
|
|
size = type_size(t, &align);
|
|
|
|
if (size < 0)
|
|
|
|
error("invalid size");
|
|
|
|
if ((u & VT_VALMASK) == VT_LOCAL) {
|
2001-10-30 07:23:01 +08:00
|
|
|
/* allocate space down on the stack */
|
2001-11-06 09:20:20 +08:00
|
|
|
loc = (loc - size) & -align;
|
2001-11-01 22:48:10 +08:00
|
|
|
sym_push(v, u, loc);
|
2001-10-30 07:23:01 +08:00
|
|
|
} else {
|
|
|
|
/* allocate space up in the data space */
|
|
|
|
glo = (glo + align - 1) & -align;
|
2001-11-01 22:48:10 +08:00
|
|
|
sym_push(v, u, glo);
|
2001-11-06 09:20:20 +08:00
|
|
|
glo += size;
|
2001-10-30 07:23:01 +08:00
|
|
|
}
|
|
|
|
}
|
2001-10-28 07:48:39 +08:00
|
|
|
}
|
|
|
|
if (tok != ',') {
|
|
|
|
skip(';');
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int c, char **v)
|
|
|
|
{
|
2001-11-01 22:48:10 +08:00
|
|
|
Sym *s;
|
2001-10-28 07:48:39 +08:00
|
|
|
int (*t)();
|
|
|
|
if (c < 2) {
|
2001-11-06 09:20:20 +08:00
|
|
|
printf("usage: tcc source ...\n");
|
2001-10-28 07:48:39 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
v++;
|
2001-10-28 23:20:12 +08:00
|
|
|
filename = *v;
|
2001-11-06 09:20:20 +08:00
|
|
|
line_num = 1;
|
2001-10-28 23:20:12 +08:00
|
|
|
file = fopen(filename, "r");
|
|
|
|
if (!file) {
|
|
|
|
perror(filename);
|
|
|
|
exit(1);
|
|
|
|
}
|
2001-11-06 09:20:20 +08:00
|
|
|
include_stack_ptr = include_stack;
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
idtable = malloc(SYM_TABLE_SIZE);
|
2001-10-28 10:51:52 +08:00
|
|
|
memcpy(idtable,
|
2001-11-06 09:20:20 +08:00
|
|
|
"int\0void\0char\0if\0else\0while\0break\0return\0define\0main\0for\0extern\0static\0unsigned\0goto\0do\0continue\0switch\0case\0const\0volatile\0long\0register\0signed\0auto\0inline\0float\0double\0struct\0union\0typedef\0default\0enum\0sizeof\0include", 219);
|
|
|
|
idptr = idtable + 219;
|
|
|
|
|
2001-10-28 07:48:39 +08:00
|
|
|
glo = malloc(DATA_SIZE);
|
2001-11-06 09:20:20 +08:00
|
|
|
memset((void *)glo, 0, DATA_SIZE);
|
2001-10-28 07:48:39 +08:00
|
|
|
prog = malloc(TEXT_SIZE);
|
2001-11-06 09:20:20 +08:00
|
|
|
vstack = malloc(256);
|
|
|
|
vstack_ptr = vstack;
|
2001-10-28 07:48:39 +08:00
|
|
|
macro_stack_ptr = macro_stack;
|
2001-11-06 09:20:20 +08:00
|
|
|
anon_sym = 1 << (31 - VT_STRUCT_SHIFT);
|
2001-10-28 07:48:39 +08:00
|
|
|
ind = prog;
|
2001-11-06 09:20:20 +08:00
|
|
|
inp();
|
|
|
|
ch = '\n'; /* needed to parse correctly first preprocessor command */
|
2001-10-28 07:48:39 +08:00
|
|
|
next();
|
|
|
|
decl(VT_CONST);
|
|
|
|
#ifdef TEST
|
|
|
|
{
|
|
|
|
FILE *f;
|
|
|
|
f = fopen(v[1], "w");
|
|
|
|
fwrite((void *)prog, 1, ind - prog, f);
|
|
|
|
fclose(f);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#else
|
2001-11-01 22:48:10 +08:00
|
|
|
s = sym_find(TOK_MAIN);
|
|
|
|
if (!s)
|
2001-10-28 23:20:12 +08:00
|
|
|
error("main() not defined");
|
2001-11-01 22:48:10 +08:00
|
|
|
t = s->c;
|
2001-10-28 07:48:39 +08:00
|
|
|
return (*t)(c - 1, v);
|
|
|
|
#endif
|
|
|
|
}
|