- 实现了 # 和 ## 预处理器操作符的功能 - 添加了 token 深拷贝和移动函数以支持宏展开 - 修改预处理器展开逻辑以正确处理宏参数替换 - 增加了宏参数分割时对空白字符的处理 fix: 修复预处理器宏展开中的内存管理和逻辑错误 - 修正了宏展开集合的数据结构初始化方式 - 修复了函数式宏调用时括号匹配的判断逻辑 - 改进了宏参数解析过程中空白字符的处理 - 解决了 token 在宏展开过程中的所有权管理问题 chore: 为 justfile 添加文件统计命令并优化构建配置 - 新增 count-file 命令用于统计代码文件数量 - 调整了输出文件的默认命名规则 - 优化了词法分析器 token 释放时的字段重置逻辑
158 lines
4.6 KiB
C
158 lines
4.6 KiB
C
#include <pproc_macro.h>
|
|
|
|
// 创建宏对象
|
|
scc_pproc_macro_t *scc_pproc_macro_new(const scc_cstring_t *name,
|
|
scc_pproc_macro_type_t type) {
|
|
scc_pproc_macro_t *macro = scc_malloc(sizeof(scc_pproc_macro_t));
|
|
if (!macro) {
|
|
LOG_ERROR("Failed to allocate memory for macro");
|
|
return null;
|
|
}
|
|
|
|
macro->name = scc_cstring_copy(name);
|
|
macro->type = type;
|
|
scc_vec_init(macro->params);
|
|
scc_vec_init(macro->replaces);
|
|
|
|
return macro;
|
|
}
|
|
|
|
// 销毁宏对象
|
|
void scc_pproc_macro_drop(scc_pproc_macro_t *macro) {
|
|
if (!macro)
|
|
return;
|
|
|
|
// 释放参数列表
|
|
for (usize i = 0; i < macro->params.size; ++i) {
|
|
scc_lexer_tok_drop(&scc_vec_at(macro->params, i));
|
|
}
|
|
scc_vec_free(macro->params);
|
|
|
|
// 释放替换列表
|
|
for (usize i = 0; i < macro->replaces.size; ++i) {
|
|
scc_lexer_tok_drop(&scc_vec_at(macro->replaces, i));
|
|
}
|
|
scc_vec_free(macro->replaces);
|
|
|
|
scc_cstring_free(¯o->name);
|
|
|
|
scc_free(macro);
|
|
}
|
|
|
|
// 添加对象宏
|
|
cbool scc_pproc_add_object_macro(scc_pproc_macro_table_t *macros,
|
|
const scc_cstring_t *name,
|
|
const scc_lexer_tok_vec_t *replacement) {
|
|
if (!macros || !name || !replacement)
|
|
return false;
|
|
|
|
scc_pproc_macro_t *macro = scc_pproc_macro_new(name, SCC_PP_MACRO_OBJECT);
|
|
if (!macro)
|
|
return false;
|
|
|
|
macro->replaces = *replacement;
|
|
|
|
// 检查是否已存在同名宏
|
|
scc_pproc_macro_t *existing =
|
|
scc_hashtable_get(¯os->table, ¯o->name);
|
|
if (existing) {
|
|
LOG_WARN("Redefining macro: %s", scc_cstring_as_cstr(¯o->name));
|
|
scc_pproc_macro_drop(existing);
|
|
}
|
|
|
|
scc_hashtable_set(¯os->table, ¯o->name, macro);
|
|
return true;
|
|
}
|
|
|
|
// 添加函数宏
|
|
cbool scc_pproc_add_function_macro(scc_pproc_macro_table_t *macros,
|
|
const scc_cstring_t *name,
|
|
const scc_lexer_tok_vec_t *params,
|
|
const scc_lexer_tok_vec_t *replacement) {
|
|
if (!macros || !name || !params || !replacement)
|
|
return false;
|
|
|
|
scc_pproc_macro_t *macro = scc_pproc_macro_new(name, SCC_PP_MACRO_FUNCTION);
|
|
if (!macro)
|
|
return false;
|
|
|
|
// 复制参数列表
|
|
macro->params = *params;
|
|
macro->replaces = *replacement;
|
|
|
|
// 检查是否已存在同名宏
|
|
scc_pproc_macro_t *existing =
|
|
scc_hashtable_get(¯os->table, ¯o->name);
|
|
if (existing) {
|
|
LOG_WARN("Redefining macro: %s", scc_cstring_as_cstr(¯o->name));
|
|
scc_pproc_macro_drop(existing);
|
|
}
|
|
|
|
scc_hashtable_set(¯os->table, ¯o->name, macro);
|
|
return true;
|
|
}
|
|
|
|
/// marco_table
|
|
|
|
scc_pproc_macro_t *scc_pproc_macro_table_set(scc_pproc_macro_table_t *pp,
|
|
scc_pproc_macro_t *macro) {
|
|
Assert(pp != null && macro != null);
|
|
return scc_hashtable_set(&pp->table, ¯o->name, macro);
|
|
}
|
|
|
|
// 查找宏定义
|
|
scc_pproc_macro_t *scc_pproc_macro_table_get(scc_pproc_macro_table_t *pp,
|
|
const scc_cstring_t *name) {
|
|
return scc_hashtable_get(&pp->table, name);
|
|
}
|
|
|
|
// 从预处理器中删除宏
|
|
cbool scc_pproc_macro_table_remove(scc_pproc_macro_table_t *pp,
|
|
const scc_cstring_t *name) {
|
|
if (!pp || !name)
|
|
return false;
|
|
|
|
scc_pproc_macro_t *macro = scc_hashtable_get(&pp->table, name);
|
|
if (!macro)
|
|
return false;
|
|
|
|
scc_hashtable_del(&pp->table, name);
|
|
scc_pproc_macro_drop(macro);
|
|
return true;
|
|
}
|
|
|
|
static u32 hash_func(const void *key) {
|
|
const scc_cstring_t *string = (const scc_cstring_t *)key;
|
|
return scc_strhash32(scc_cstring_as_cstr(string));
|
|
}
|
|
|
|
static int hash_cmp(const void *key1, const void *key2) {
|
|
const scc_cstring_t *str1 = (const scc_cstring_t *)key1;
|
|
const scc_cstring_t *str2 = (const scc_cstring_t *)key2;
|
|
|
|
if (str1->size != str2->size) {
|
|
return str1->size - str2->size;
|
|
}
|
|
return scc_strcmp(scc_cstring_as_cstr(str1), scc_cstring_as_cstr(str2));
|
|
}
|
|
|
|
void scc_pproc_marco_table_init(scc_pproc_macro_table_t *macros) {
|
|
Assert(macros != null);
|
|
macros->table.hash_func = hash_func;
|
|
macros->table.key_cmp = hash_cmp;
|
|
scc_hashtable_init(¯os->table);
|
|
}
|
|
|
|
static int macro_free(const void *key, void *value, void *context) {
|
|
(void)key;
|
|
(void)context;
|
|
scc_pproc_macro_drop(value);
|
|
return 0;
|
|
}
|
|
|
|
void scc_pproc_macro_table_drop(scc_pproc_macro_table_t *macros) {
|
|
Assert(macros != null);
|
|
scc_hashtable_foreach(¯os->table, macro_free, null);
|
|
scc_hashtable_drop(¯os->table);
|
|
}
|