TableGen 系统可以帮助记录领域特定的信息。它也可以认为是一种小型的编译系统。
TableGen 责负分析文件, 分析结果交给领域特定的后端进行处理。
一个 record 有一个独立的名称,一系列值和一系列父类。 它保存了特定领域的知识。
可以认为 tablegen 前端生成 recode 流, 对于其中 record 的解释由具体的后端执行。
record的具体形式, 按数据结构看是 key-value 字典, 它的具体含义与作用由特定的后端后决定。可以有名,也可以无名。
例子:
def HelloWorld {
string msg = "Hello world!";
}
表示抽象的 record, 用于构造和描述其它的 record。类方便于定义目标领域的抽象。
class ProcNoItinFeatures> : Processor ;
表示一组抽象的 record, 一次实例化可以产生多个 definations。
可以看到在 multiclasses 内部 def 了多个 defination。 使用 defm 调用 muliclasses 的构造。
multiclass ro_signed_pats{ def : Pat<(i32 (!cast ("sextload" # sty) address)), (!cast ("LDRS" # T # "w_" # Rm # "_RegOffset") Base, Offset, Extend)>; def : Pat<(i64 (!cast ("sextload" # sty) address)), (!cast ("LDRS" # T # "x_" # Rm # "_RegOffset") Base, Offset, Extend)>; } defm : ro_signed_pats<"B", Rm, Base, Offset, Extend, !foreach(decls.pattern, address, !subst(SHIFT, imm_eq0, decls.pattern)), i8>;
clang-tblgen [options] [filename]
lldb-tblgen [options] [filename]
llvm-tblgen [options] [filename]
mlir-tblgen [options] [filename]
大体上是这几个, 但是每个命令指令不同的选项生成的效果也不同。
比如 clang 有这几个和 riscv 相关的指令:
-gen-riscv-vector-header
Generate
riscv_vector.hfor Clang.-gen-riscv-vector-builtins
Generate
riscv_vector_builtins.incfor Clang.-gen-riscv-vector-builtin-codegen
Generate
riscv_vector_builtin_cg.incfor Clang.-gen-riscv-sifive-vector-builtins¶
Generate
riscv_sifive_vector_builtins.incfor Clang.-gen-riscv-sifive-vector-builtin-codegen
Generate
riscv_sifive_vector_builtin_cg.incfor Clang.
list:
表现在打印输出上是一个 ?, 它可以是任意类型
类似于 S-expression, S表达式。 典型的应用场景是 ISel 即指令选择。
(op arg0:$name0, arg1:$name1, ...)
我们可以使用 llvm-tblgen 或在线网站 Compiler Explorer 来查看一个tablegen 文件对应的 class 和 defination 流,这有利于我们了解现有的 tablegen 文件描述了什么东西。
def HelloWorld {
string msg = "Hello world!";
}
def { // 匿名
string msg = "Hello world!";
}
class C {
int c = 7;
int d = 7;
}
class D
int e=a;
}
def X: C {}
def Y: D<1>{} // 模板传参
class C {
int a;
int b;
}
let a=5, b=6 in {
def X: C {}
}
class C {
int c = 7;
int d = 7;
}
class D
int e=a;
}
multiclass E
let c=num in def _c:C;// 在 muliclass 中使用 let 的语法
def _d : D
}
defm INST: E<9>; // 一次定义2个 defination
class C {
int c = 7;
int d = 7;
}
foreach i = [0, 1, 2, 3] in {
def R#i : C; // #i 会替换成对应的值
def F#i : C;
}
根据名称(字符串)获取类
class B{
string b="isb";
}
class C
B b_of_c = !cast(name);
}
def firstB: B;
def c:C<"firstB">;
拼接字符串
class Hello
string msg = "Hello " # _msg;
}
def HelloWorld: Hello<"world!"> {}
1 TableGen Programmer’s Reference — LLVM 19.0.0git documentation