mlir 编译器学习笔记之九 -- 后端生成

方案1、在mlir中使用标准的emitc转换生成

EmitC 是 MLIR(Multi-Level Intermediate Representation)框架中的一个官方方言(Dialect),而不是一个独立的编程语言标准(如 C++11 或 C99 那样的标准),其中总体实现可以参数社区的现有实现方法:

populateSCFToEmitCConversionPatterns(patterns, converter);

populateHLCToEmitCConversionPatterns(patterns, converter);

populateArithToEmitCPatterns(converter, patterns);

populateConvertMathToEmitCPatterns(patterns, emitc::LanguageTarget::c99);

populateFuncToEmitCPatterns(converter, patterns);

对于结构体,比如OpCommHead.opType = TAU_TYPE表达

%424 = literal "TAU_TYPE" : !emitc.opaque<"BauOpType">

%429 = "emitc.variable"() <{value = #emitc.opaque<"{0}">}> : () -> !emitc.lvalue<!emitc.opaque<"OpCommHead">>

%431 = "emitc.member"(%429) <{member = "opType"}> : (!emitc.lvalue<!emitc.opaque<"OpCommHead">>) -> !emitc.lvalue<!emitc.opaque<"BauOpType">>

assign %424 : !emitc.opaque<"BauOpType"> to %431 : <!emitc.opaque<"BauOpType">>

方案2、直接将输出操作拼接,即依次直接导入数据流

1.1 基于方案1:结构体的处理可以转换多元素的依次处理。比如CommHead结构

// Convert CreateOpCommHeadPtrOp to "emitc::VariableOp" and "emitc::ApplyOp".

struct CreateOpCommHeadPtrOpConversion

: public OpConversionPattern<bau::CreateOpCommHeadPtrOp> {

using OpConversionPattern<bau::CreateOpCommHeadPtrOp>::OpConversionPattern;

LogicalResult

matchAndRewrite(bau::CreateOpCommHeadPtrOp configOp, OpAdaptor adaptor,

ConversionPatternRewriter &rewriter) const override {

auto getMemberInfo =

\&\]() -\> SmallVector\\>\> { return SmallVector\\>\>{ {"wordBitmap", SmallVector\{adaptor.getWordBitmap()}}, {"condEn", SmallVector\{adaptor.getCondEn()}}}; }; return createStructOrUnionPtr\( configOp, rewriter, this-\>getTypeConverter(), getMemberInfo); } };

其中上面的文件也可以使用py生成,比如:

def generate_emitc_pattern(head_data):

f = open(EMITC_PATTERN_FILE, 'w', encoding='utf-8')

f.write(CLANG_OFF)

add_pattern_str = 'void ' + ADD_PATTERN_FUNC

add_pattern_str += '(RewritePatternSet &patterns, '

add_pattern_str += 'TypeConverter &typeConverter) {\n'

add_pattern_str += ' MLIRContext *ctx = patterns.getContext();\n'

for name, _ in utils.get_struct_and_union_asembly(head_data).items():

op_name = utils.get_create_op_name(name)

out_str = get_conversion_comment(op_name) + '\n'

out_str += (get_conversion_define(op_name) + '\n\n')

out_str += (get_conversion_main_func_def(op_name) + '\n')

out_str += (get_member_info_head() + '\n')

out_str += (get_member_convert_map(head_data, name, '', True) + '};\n')

out_str += ' };\n'

out_str += ' return createStructOrUnionPtr<bau::' + op_name + '>(\n'

out_str += ' configOp, rewriter, '

out_str += 'this->getTypeConverter(), getMemberInfo);\n'

out_str += ' }\n'

out_str += '};\n'

f.write(out_str)

f.write('\n')

add_pattern_str += (' patterns.add<' + get_conversion_name(op_name))

add_pattern_str += '>(typeConverter, ctx);\n'

add_pattern_str += '}\n'

f.write(add_pattern_str)

f.write(CLANG_ON)

print(f'Success generate {EMITC_PATTERN_FILE}')

1.2 基于方案1:call_opaqueverbatim 的主要区别

call_opaque:专门用来调用 C/C++ 函数,生成格式固定、清晰的函数调用代码。

verbatim:一个"万能"的文本插入工具,可以生成任何 C/C++ 代码

注:call_opaque 的参数只能是 SSA 值 (即其他操作的结果),不能是任意表达式,所以类似的内联emitc.call_opaque "MyFunc"(&%arg0, %arg1) : (!emitc.ptr<i32>, i32) -> ()不支持,需要改为emitc.verbatim "MyFunc(&{}, {});" args %arg0, %arg1

2、@declare_module 和 @define_module 被框架/工具链强制要求放在不同的逻辑模块(module)中,并且生成到不同的输出文件

@declare_module:头文件

@define_module:源文件

相关推荐
叶~小兮11 小时前
Zabbix 7.0 深度原理与生产进阶运维学习笔记
运维·学习·zabbix
我想我不够好。11 小时前
2026.5.21 消防监控学习 1.5hours
学习
燐妤11 小时前
前端HTML编程6:ES6与前后端交互
前端·javascript·学习·html5
什仙11 小时前
Ansys Maxwell 默认求解器选择
人工智能·笔记·算法·基础·ansys·maxwell
清平乐的技术专栏11 小时前
【Kafka笔记】(二)核心架构与专属名词解释
笔记·架构·kafka
清平乐的技术专栏11 小时前
【Kafka笔记】(三)常用命令整理
笔记·分布式·kafka
Hua-Jay11 小时前
OpenCV联合C++/Qt 学习笔记(二十四)----差值法检测移动物体、稠密光流法跟踪移动物体及稀疏光流法跟踪移动物体
c++·笔记·qt·opencv·学习·计算机视觉
清平乐的技术专栏12 小时前
【FlinkSQL笔记】(一)什么是Flink SQL
笔记·sql·flink
坤坤藤椒牛肉面12 小时前
stm32学习1--新建工程
stm32·单片机·学习