ops-elementwise:小算子的融合艺术

Add、Mul、Sub、Div------这些逐元素运算的计算量几乎为零,但在推理中出现的频率最高。一个 Transformer Block 里几十次 Add(残差连接、偏置加),几十次 Mul(Attention 的 scale、Dropout 的 mask 乘)。

每个小算子独立 Launch 一次就有 5-15μs 的 Runtime 调度开销。把这些小算子融合到一个 Kernel 里------省掉的 Launch 时间比计算时间还多。


为什么简单算子也会拖慢推理

A + B 的 Vector Unit 执行时间约 0.5μs(4096 个元素的加法)。但独立 Launch 这个算子需要:

  • Runtime Task 创建:3-5μs
  • Driver 提交:2-5μs
  • Vector Unit 执行:0.5μs

调度开销是计算时间的 10-20 倍。


ElementWise 为什么适合融合

ElementWise 算子之间没有数据依赖关系------连续的 Add、Mul、Scale 可以合并成一个复合算子。融合后的 Kernel 只做一次 Launch,调度开销分摊到多个操作上。

cpp 复制代码
// 不融合:3 次 Launch
z = x + y;    // Launch Add
w = z * scale;  // Launch Mul  
u = w + bias;   // Launch Add

// 融合:1 次 Launch
u = (x + y) * scale + bias;  // Launch Fusion

融合后的 Launch 开销减少了 66%。Vector Unit 在 L1 上流水线执行加法和乘法------x + y 的结果在 Vector 寄存器中直接传给 * scale,不需要写 DDR 再读。


昇腾NPU如何减少 Kernel 启动

ops-elementwise 的融合策略:

  1. 连续 ElementWise 融合。 检测连续出现的 Add、Mul、Scale、Bias 等操作,合并为一个 Composite Kernel
  2. 激活函数融合。 GELU 这种需要多个 Vector 指令的激活函数也可以跟前面的 ElementWise 合并------x * scale + bias → GELU
  3. 与 GEMM 的 Epilogue 融合。 ElementWise 算子作为 GEMM 的 Epilogue------GEMM 算完立即做 Add/Mul/Scale,中间结果不落 DDR

Transformer 中的典型融合场景

残差连接的 Add 融合。 Attention 子层的输出 + 输入残差------这个 Add 不独立 Launch,而是作为 LayerNorm 的输入的一部分,GE 把 Add 融合到 LayerNorm 的 Kernel 中。

Scale + Add(偏置)。 GEMM 输出的 Scale 和 Bias Add 也作为 GEMM 的 Epilogue------GEMM 计算最后一个 Tile 后立即做 Scale 和 Bias,在 L1 上完成。

Dropout 的 Mul 融合。 Dropout 的 mask(ops-rand 生成)跟激活输出的 Mul 融合到上一个算子的 Epilogue 中------激活函数算完后立即跟 mask 相乘,不写 DDR。

更多典型融合案例

Softmax + Mul(Dropout) 的融合:Softmax 的输出是概率分布,Dropout 随机丢弃一些概率值。如果不融合,Softmax 的 [n,n] 输出写入 DDR,Dropout 从 DDR 读入再写出------32MB 的额外搬运。融合后 Dropout 的 mask 在 Softmax 的 exp 结果上直接相乘------Softmax 的分母求和、mask 的逐元素乘在同一个 Kernel 内完成。

Add + Mul + Add 的残差融合:Transformer Block 的 残差输出 = attention_output + input 加完后的结果经过乘法和下一次加法。这三个 ElementWise 向量计算可以合并为一个 Vector Kernel------一次 Launch、一次数据搬运。

实际推理中的收益

Ascend 910 上 LLaMA-7B 的 ElementWise 融合前后的对比:

配置 单 Block 的 Kernel 数 Launch 开销 Block 延迟
不融合 24 个独立 Kernel 240μs 2.85ms
ElementWise 融合 12 个融合 Kernel 120μs 2.65ms

Launch 开销减半是直接收益。另外的间接收益是中间 Tensor 搬运减少------融合后只要原来一半的 ElementWise 中间结果写 DDR。

参考仓库

ops-elementwise 逐元素算子库

graph-autofusion 自动融合框架

相关推荐
czhm571 小时前
深度解析CANN架构下昇腾NPU Vector算子开发新范式:ATVOSS模板库设计理念与工程实践
cann
czhm571 天前
昇腾CANN计算机视觉专用算子库ops-cv快速上手实战教程:从环境配置到image/objdetect类接口调用的全步骤可复现操作指南
cann
czhm572 天前
CANN进阶指南|hccl集合通信库算法实现与大规模集群优化:从Ring到Tree的通信路径选择与拓扑感知调优实践
cann
czhm572 天前
CANN架构解析|graph-autofusion算子自动融合框架的设计原理与工程实现全链路深度解读
cann
czhm572 天前
CANN技术解读|hcomm通信库主机侧网络优化与零拷贝技术:深入剖析分布式训练通信瓶颈的高效解决方案
cann
xiaoqi9222 天前
Python 高手编程系列四百九十三:何时应该使用多线程
cann
czhm572 天前
CANN技术解读|metadef元数据结构与模型定义规范——深度解析昇腾CANN计算架构中基础数据层的核心设计
cann
czhm572 天前
CANN架构解析|GE图编译引擎核心原理与优化策略:深度剖析图编译技术在异构计算中的应用与实践
cann
zjun30212 天前
【昇腾950 cv融合算子体验】L0C Buffer到UB的单向数据通路
cann·融合算子·昇腾950
luozhen1103 天前
ops-math昇腾数学算子库深度解析:高性能数值计算与算法实现完全指南
cann