mirror of
https://github.com/DarkPlacesEngine/gmqcc.git
synced 2025-03-21 02:10:53 +00:00
parsing typenames completely now with function parameters
This commit is contained in:
parent
5b7dfb7e4b
commit
1d55197097
2 changed files with 81 additions and 9 deletions
10
gmqcc.h
10
gmqcc.h
|
@ -717,6 +717,16 @@ void Tself##_##mem##_clear(Tself *self) \
|
|||
(owner)->mem##_alloc = 0; \
|
||||
}
|
||||
|
||||
#define MEM_VECTOR_MOVE(from, mem, to, tm) \
|
||||
{ \
|
||||
(to)->tm = (from)->mem; \
|
||||
(to)->tm##_count = (from)->mem##_count; \
|
||||
(to)->tm##_alloc = (from)->mem##_alloc; \
|
||||
(from)->mem = NULL; \
|
||||
(from)->mem##_count = 0; \
|
||||
(from)->mem##_alloc = 0; \
|
||||
}
|
||||
|
||||
#define MEM_VEC_FUNCTIONS(Tself, Twhat, mem) \
|
||||
_MEM_VEC_FUN_REMOVE(Tself, Twhat, mem) \
|
||||
_MEM_VEC_FUN_ADD(Tself, Twhat, mem)
|
||||
|
|
80
parser.c
80
parser.c
|
@ -60,15 +60,73 @@ ast_value* parser_find_global(parser_t *parser, const char *name)
|
|||
return NULL;
|
||||
}
|
||||
|
||||
typedef struct {
|
||||
MEM_VECTOR_MAKE(ast_value*, p);
|
||||
} paramlist_t;
|
||||
MEM_VEC_FUNCTIONS(paramlist_t, ast_value*, p)
|
||||
|
||||
ast_value *parser_parse_type(parser_t *parser)
|
||||
{
|
||||
paramlist_t params;
|
||||
ast_value *var;
|
||||
lex_ctx ctx = parser_ctx(parser);
|
||||
int vtype = parser_token(parser)->constval.t;
|
||||
|
||||
MEM_VECTOR_INIT(¶ms, p);
|
||||
|
||||
if (!parser_next(parser))
|
||||
return NULL;
|
||||
|
||||
if (parser->tok == '(') {
|
||||
while (true) {
|
||||
ast_value *param;
|
||||
|
||||
if (!parser_next(parser)) {
|
||||
MEM_VECTOR_CLEAR(¶ms, p);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
param = parser_parse_type(parser);
|
||||
if (!param) {
|
||||
MEM_VECTOR_CLEAR(¶ms, p);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (!paramlist_t_p_add(¶ms, param)) {
|
||||
MEM_VECTOR_CLEAR(¶ms, p);
|
||||
parseerror(parser, "Out of memory while parsing typename");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (parser->tok == ',')
|
||||
continue;
|
||||
if (parser->tok == ')')
|
||||
break;
|
||||
MEM_VECTOR_CLEAR(¶ms, p);
|
||||
parseerror(parser, "Unexpected token");
|
||||
return NULL;
|
||||
}
|
||||
if (!parser_next(parser)) {
|
||||
MEM_VECTOR_CLEAR(¶ms, p);
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
var = ast_value_new(ctx, "<unnamed>", vtype);
|
||||
if (!var) {
|
||||
MEM_VECTOR_CLEAR(¶ms, p);
|
||||
return NULL;
|
||||
}
|
||||
MEM_VECTOR_MOVE(¶ms, p, var, params);
|
||||
return var;
|
||||
}
|
||||
|
||||
bool parser_do(parser_t *parser)
|
||||
{
|
||||
if (parser->tok == TOKEN_TYPENAME)
|
||||
{
|
||||
ast_value *var;
|
||||
int vtype = parser->lex->tok->constval.t;
|
||||
|
||||
/* Declaring a variable */
|
||||
if (!parser_next(parser))
|
||||
ast_value *var = parser_parse_type(parser);
|
||||
if (!var)
|
||||
return false;
|
||||
|
||||
if (parser->tok != TOKEN_IDENT) {
|
||||
|
@ -76,14 +134,18 @@ bool parser_do(parser_t *parser)
|
|||
return false;
|
||||
}
|
||||
|
||||
var = parser_find_global(parser, parser_tokval(parser));
|
||||
|
||||
if (var) {
|
||||
if (parser_find_global(parser, parser_tokval(parser))) {
|
||||
ast_value_delete(var);
|
||||
parseerror(parser, "global already exists: %s\n", parser_tokval(parser));
|
||||
return false;
|
||||
}
|
||||
|
||||
var = ast_value_new(parser_ctx(parser), parser_tokval(parser), vtype);
|
||||
if (!ast_value_set_name(var, parser_tokval(parser))) {
|
||||
parseerror(parser, "failed to set variable name\n");
|
||||
ast_value_delete(var);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!parser_t_globals_add(parser, var))
|
||||
return false;
|
||||
|
||||
|
|
Loading…
Reference in a new issue