重构AST表达式枚举,将COMPOUND_LITERAL重命名为COMPOUND, 更新相关结构体定义以支持复合字面量的左右表达式向量表示。 添加lvalue表达式类型用于左值处理,实现完整的初始化器解析功能, 包括大括号初始化列表、成员访问和数组下标的处理逻辑。 更新表达式解析器为基于优先级的递归下降解析,修复变量声明中 初始化表达式的内存泄漏问题。 完善类型限定符和存储类说明符的重复检查机制,增强语法分析的准确性。
90 lines
2.9 KiB
C
90 lines
2.9 KiB
C
/**
|
|
* @file pprocessor.h
|
|
* @brief C语言预处理器核心数据结构与接口
|
|
*/
|
|
|
|
#ifndef __SCC_PPROC_H__
|
|
#define __SCC_PPROC_H__
|
|
|
|
#include "pproc_macro.h"
|
|
#include <scc_core.h>
|
|
#include <scc_core_ring.h>
|
|
#include <scc_lexer.h>
|
|
|
|
// 预处理器状态结构
|
|
|
|
// 条件编译状态栈
|
|
typedef struct {
|
|
cbool found_true;
|
|
cbool seen_else;
|
|
cbool active;
|
|
} scc_pproc_if_t;
|
|
typedef SCC_VEC(scc_pproc_if_t) scc_pproc_if_stack_t;
|
|
|
|
typedef struct {
|
|
scc_sstream_t sstream;
|
|
scc_lexer_t lexer;
|
|
scc_lexer_tok_ring_t *ring;
|
|
} scc_pproc_file_t;
|
|
typedef SCC_VEC(scc_pproc_file_t *) scc_pproc_file_stack_t;
|
|
|
|
typedef SCC_VEC(scc_lexer_tok_ring_t *) scc_pproc_ring_vec_t;
|
|
typedef SCC_VEC(scc_cstring_t) scc_pproc_cstr_vec_t;
|
|
|
|
typedef struct scc_pproc {
|
|
scc_lexer_tok_ring_t *org_ring;
|
|
scc_lexer_tok_ring_t *cur_ring;
|
|
scc_lexer_tok_ring_t expanded_ring;
|
|
scc_strpool_t strpool;
|
|
int at_line_start;
|
|
int enable;
|
|
|
|
scc_pproc_cstr_vec_t include_paths;
|
|
scc_pproc_macro_table_t macro_table;
|
|
scc_pproc_if_stack_t if_stack;
|
|
scc_pproc_file_stack_t file_stack;
|
|
|
|
scc_lexer_tok_ring_t ring;
|
|
int ring_ref_count;
|
|
cbool ring_need_comment;
|
|
cbool ring_need_empty;
|
|
|
|
struct {
|
|
int max_include_depth;
|
|
cbool keep_original_pos;
|
|
} config;
|
|
} scc_pproc_t;
|
|
|
|
void scc_pproc_init(scc_pproc_t *pp, scc_lexer_tok_ring_t *input);
|
|
scc_lexer_tok_ring_t *scc_pproc_to_ring(scc_pproc_t *pp, int ring_size,
|
|
cbool need_empty, cbool need_comment);
|
|
void scc_pproc_drop(scc_pproc_t *pp);
|
|
|
|
static inline void scc_pproc_add_include_path(scc_pproc_t *pp,
|
|
const scc_cstring_t *path) {
|
|
scc_vec_push(pp->include_paths, scc_cstring_copy(path));
|
|
}
|
|
|
|
static inline void scc_pproc_add_include_path_cstr(scc_pproc_t *pp,
|
|
const char *path) {
|
|
scc_vec_push(pp->include_paths, scc_cstring_from_cstr(path));
|
|
}
|
|
|
|
void scc_pproc_handle_directive(scc_pproc_t *pp);
|
|
|
|
cbool scc_pproc_parse_if_need_skip(scc_pproc_t *pp, scc_tok_type_t type);
|
|
cbool scc_pproc_parse_if_defined(scc_pproc_t *pp, scc_tok_type_t type,
|
|
const scc_lexer_tok_t *tok);
|
|
cbool scc_pproc_parse_if_condition(scc_pproc_t *pp, scc_tok_type_t type,
|
|
scc_lexer_tok_ring_t *tok_ring);
|
|
void scc_pproc_parse_include(scc_pproc_t *pp, scc_lexer_tok_t *include_tok,
|
|
scc_lexer_tok_ring_t *tok_ring);
|
|
|
|
void scc_pproc_parse_macro_arguments(scc_lexer_tok_ring_t *ring,
|
|
scc_lexer_tok_vec_t *args, int need_full);
|
|
void scc_pproc_parse_function_macro(scc_pproc_t *pp,
|
|
const scc_lexer_tok_t *ident);
|
|
void scc_pproc_parse_object_macro(scc_pproc_t *pp,
|
|
const scc_lexer_tok_t *ident);
|
|
#endif /* __SCC_PPROC_H__ */
|