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<std::pair<StringRef, SmallVector<Value>>> {

return SmallVector<std::pair<StringRef, SmallVector<Value>>>{

{"wordBitmap", SmallVector<Value>{adaptor.getWordBitmap()}},

{"condEn", SmallVector<Value>{adaptor.getCondEn()}}};

};

return createStructOrUnionPtr<bau::CreateOpCommHeadPtrOp>(

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:源文件

相关推荐
RainCity4 天前
Java Swing 自定义组件库分享(十二)
java·笔记·后端
LinXunFeng12 天前
Obsidian - 使用 Share Note 分享笔记并自部署
前端·笔记·github
通信小呆呆16 天前
当算法有了“五感”:多模态数据融合如何向人体感官协同学习?
人工智能·学习·算法·机器学习·机器人
H__Rick16 天前
自动对焦学习-3
人工智能·学习·计算机视觉
Daisy Lee16 天前
量化学习-第1章-什么是量化金融
学习·金融·datawhale
Alsn8616 天前
等待学习-学习目录:Docker 容器安全攻防
学习·安全·docker
YM52e16 天前
买菜计算器小应用 - HarmonyOS ArkUI 开发实战-PC版本
学习·华为·harmonyos·鸿蒙·鸿蒙系统
小雨下雨的雨16 天前
HarmonyOS ArkUI训练营入门-组件掌握系列-Animation 动画效果实现-PC版本
学习·华为·harmonyos·鸿蒙
闪闪发亮的小星星16 天前
高斯光以及高斯光公式解释
笔记
cqbzcsq16 天前
CellFlow虚拟细胞论文阅读
论文阅读·人工智能·笔记·学习·生物信息