翻开 CANN 任何一个算子仓库------ops-math、ops-nn、ops-blas、ops-fft、ops-rand------它们都依赖同一个基础库:opbase。
opbase 不提供面向用户的算子能力。它提供的是所有算子共享的基础设施:Tensor 的内存管理和形状描述、Kernel 的注册和加载机制、Buffer 的分配和生命周期管理。
opbase 在 CANN 中的位置
上层算子库:ops-math、ops-nn、ops-blas、ops-fft、ops-rand ......(各自管一类算子)
↓ 全部依赖
opbase:
┌──────────────────────────────┐
│ Tensor 数据结构 │
│ Buffer 内存管理 │
│ Kernel 注册框架 │
│ 通用类型定义 │
└──────────────────────────────┘
↓
CANN 底层:Runtime、driver
每个新算子仓库启动时不需要从零写 Tensor 管理、Buffer 分配、Kernel 注册------这些在 opbase 中已就绪,新仓库直接引用。
为什么所有算子仓库都依赖 opbase
一个算子从写到 Runtime 执行,需要的公共能力包括:
Tensor 描述。 每个算子需要知道输入 Tensor 的形状、数据类型、内存地址。opbase 定义了统一的 TensorDesc 结构------所有算子仓库都使用这个描述。ops-math 和 ops-nn 的算子之间传递 Tensor 信息时不需要做格式转换。
Buffer 分配。 算子执行过程中的临时 Tensor 需要分配显存。opbase 提供了 opbase::AllocBuffer------封装了 Runtime 的 aclrtMalloc,但增加了算子级别的生命周期追踪。Buffer 在算子执行完后自动回收。
Kernel 注册。 写好的算子 Kernel 需要注册到 CANN 的算子表中才能被 GE 识别和调度。opbase 提供了 REGISTER_KERNEL 宏------开发者只需要在 Kernel 实现文件中加一行注册语句。
cpp
#include "opbase/opbase.h"
// 自定义 Add Kernel
__aicore__ void AddKernel(...) { ... }
// 注册到算子表------一行搞定
REGISTER_KERNEL("CustomAdd", AddKernel);
// GE 在编译时就能在算子表中找到 "CustomAdd"
Tensor 基础能力如何复用
所有算子仓库中 Tensor 的创建和销毁走同一条路径:
cpp
// opbase 的 Tensor 创建------所有算子仓库共用
opbase::Tensor tensor = opbase::AllocTensor(
{batch, seq_len, hidden_dim}, // Shape
opbase::FLOAT16, // 数据类型
opbase::ND // 格式
);
// tensor 的内存由 opbase 管理
// 算子执行完自动回收
每个算子仓库不需要重新实现 Tensor 分配逻辑。
Buffer 的生命周期管理也是复用的:
cpp
// ops-math 的 ReduceSum 算子内部
void ReduceSumKernel(...) {
// opbase 管理的临时 Buffer
auto temp = opbase::AllocBuffer(partial_sum_size);
// 计算...
// temp 在函数结束时自动回收
}
算子开发中的基础设施
写一个新的 CANN 算子时,opbase 提供的工作流:
- 继承 OpBase。 新算子继承
opbase::Operator基类,获得输入输出解析、参数校验、生命周期管理等默认实现 - 实现 Compute。 只需要实现
Compute()虚函数------算子的核心计算逻辑 - 注册。 使用
REGISTER_KERNEL宏将算子注册到算子表
cpp
class MyNewOp : public opbase::Operator {
public:
OpBase::Status Init(const OpDesc& desc) override {
// opbase 自动解析输入输出描述
return opbase::SUCCESS;
}
OpBase::Status Compute(const std::vector<opbase::Tensor>& inputs,
std::vector<opbase::Tensor>& outputs) override {
// 只有这段代码需要开发者写
auto input = inputs[0].data<float16>();
auto output = outputs[0].data<float16>();
// 计算逻辑...
return opbase::SUCCESS;
}
};
REGISTER_KERNEL("MyNewOp", MyNewOp);
opbase 的类型系统
opbase 定义了一套跨仓库统一的类型系统:
- DataType:FLOAT32、FLOAT16、INT8、INT32、INT64、BOOL
- Format:ND、NZ、NHWC、NCHW
- Shape:动态 Shape 用 -1 表示,支持 ShapeRange
所有算子仓库都使用同一套类型。ops-math 的 ReduceSum 输出的 DataType 跟 ops-nn 的 Conv 输入的 DataType 是同一个枚举值------不需要类型转换。
opbase 的 Kernel 注册表
Kernel 注册表是 opbase 的另一个核心功能。每个算子库在加载时调用 REGISTER_KERNEL 把 Kernel 注册到全局表中。GE 在编译时查表找到对应算子的 Kernel。
注册表是进程级的------跨模型共享。模型 A 加载了 ops-nn 的 Conv Kernel,模型 B 加载同一个 Operator 时不需要重复注册。这个机制在多模型服务中节省了重复加载 Kernel 的时间。