ops-softmax:Transformer 推理中的概率归一化引擎

Transformer 里 Attention 的核心是 Softmax------它把注意力分数变成概率分布。没有 Softmax,注意力分数就只是一个数值没有归一化的矩阵,无法作为权重来聚合 Value。

CANN 的 ops-softmax 仓库专门管理 Softmax 及其变体的实现。Softmax 的计算量不大------就是 exp → sum → div 三步------但它的数据访问模式决定了它是 Memory Bound 算子,在昇腾NPU 上需要针对大序列长度做专门的优化。


Softmax 为什么是 Transformer 核心

Attention 的计算公式:Attention(Q,K,V) = Softmax(Q × K^T / √d) × V

Q × K^T 输出的注意力分数矩阵 S[n, n] 的矩阵。矩阵中的每个元素 S_ij 表示第 i 个 Token 对第 j 个 Token 的注意力强度。但这些分数是未归一化的------可能很大也可能很小。Softmax 把它们归一化成概率分布,让 sum(S_ij over j) = 1

Softmax 的步骤:

  1. exp(x_i)------指数化,把分数转为正数
  2. sum(exp(x_i))------求所有指数值的和
  3. exp(x_i) / sum------每个指数值除以总和,归一化为概率

Softmax 为什么会成为性能瓶颈

Softmax 的计算量很小------每个元素一次指数运算、一次除法。但它的数据访问模式很差:

  • 输入读取 [n, n] 矩阵的全部元素(从 DDR 搬到 L1)
  • 对所有元素做指数运算(Vector Unit 执行)
  • 在行方向做 sum(归约操作,需要对整行扫描)
  • 再读取一次,每个元素除以 sum(从 DDR 搬到 L1)

对于 n=4096 的序列,Score 矩阵 32MB。整个流程需要搬运约 64MB------两次读 S、一次写 S_softmax。计算/搬运比很低。

FlashAttention 中的 Softmax 优化

FlashAttention 对 Softmax 的优化是让它原地完成------Score 矩阵不落地 DDR。具体做法:Score 矩阵被切成 block×block 的子块,每次只搬运一个子块到 L1。在 L1 上做完 Softmax 后立即跟 Value 做矩阵乘,Softmax 的结果不需要写回 DDR。

这个过程需要 Online Softmax 算法------在不知道全局最大值的情况下分块计算:

复制代码
初始化:max_val = -inf, sum_val = 0
循环每个 K/V 块:
  当前块的最大值 local_max = max(S_ij)
  更新 max_val = max(max_val, local_max)
  缩放旧的 sum_val:sum_val *= exp(max_val - local_max)
  当前块的 exp 和:local_sum = sum(exp(S_ij - max_val))
  累积:sum_val += local_sum

Online Softmax 的计算精度跟标准 Softmax 完全一致,但避免了 Score 矩阵的整体搬运。在长序列场景中,Softmax 不再是性能瓶颈。

Online Softmax 的数值稳定性

Softmax 的朴素实现:exp(x_i)x_i 很大时(如 Attention Score 的值可能超过 30)会导致 float16 溢出。标准做法是减去最大值:exp(x_i - max(x)) / sum(exp(x_j - max(x)))

FlashAttention 的 Online Softmax 在分块计算时也保持了数值稳定性------每个分块独立减去自己的局部最大值,跨分块时用 running max 修正。这个修正的数值误差在 10^-5 级别------不影响推理精度。

ops-softmax 在 Vector Unit 上的实现

ops-softmax 在 Vector Unit 上的实现不是直接写一条 softmax 指令------Vector Unit 只有基本的数学指令。Softmax 被拆解为:

  1. vec_max(x) --- SIMD 找最大值
  2. vec_sub(x, max) --- 每个元素减最大值
  3. vec_exp(x) --- SIMD 指数运算(使用多项式近似)
  4. vec_sum(exp_x) --- SIMD 求和
  5. vec_div(exp_x, sum) --- 每个元素除以总和

这 5 条 Vector 指令在 L1 上执行,不需要写 DDR。对于 4096 个元素的 Softmax,Vector Unit 的执行时间约 1-2μs。

大序列长度(n > 4096)时,Score 矩阵 [n, n] 超出了一次 Kernel 可以处理的 L1 容量。ops-softmax 把 Score 矩阵按行分成多块------每块在 L1 上做完完整的 Softmax 后再写回 DDR。

参考仓库

ops-softmax 仓库

FlashAttention 融合优化

相关推荐
冬奇Lab15 小时前
Workflow 系列(06):安全——跨步骤注入传播与四层防御
人工智能·工作流引擎
冬奇Lab15 小时前
每日一个开源项目(第149篇):RAG-Anything - 把图片、表格、公式当成一等公民的多模态 RAG 框架
人工智能·开源
米小虾15 小时前
AI Agent 安全实战指南:当智能体开始"不听话",开发者该如何应对?
人工智能·安全·agent
IT_陈寒17 小时前
Vite的热更新突然不香了,排查三小时差点砸键盘
前端·人工智能·后端
阿里云大数据AI技术19 小时前
构建高转化海外电商搜索:阿里云OpenSearch行业算法版的全链路智能优化策略实战
人工智能·搜索引擎
Awu122719 小时前
⚡从零开发 Agent CLI(五)实现一个可治理、可扩展的工具系统
前端·人工智能·claude
字节跳动视频云技术团队19 小时前
让 Agent 成为音视频工作台:AI MediaKit CLI + Skill 发布
人工智能·音视频开发
魏祖潇19 小时前
framework 整合实战——DDD/TDD/SDD 三件套在 framework 仓的真实落地
人工智能·后端
Token炼金师20 小时前
去噪扩散:从随机噪声到高保真图像的数学之路
人工智能·aigc