opbase:CANN 所有算子的公共地基

翻开 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 提供的工作流:

  1. 继承 OpBase。 新算子继承 opbase::Operator 基类,获得输入输出解析、参数校验、生命周期管理等默认实现
  2. 实现 Compute。 只需要实现 Compute() 虚函数------算子的核心计算逻辑
  3. 注册。 使用 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 的时间。

参考仓库

opbase 算子基础组件库

Ascend C 算子编程指南

相关推荐
火山引擎开发者社区12 小时前
没有长期记忆,Agent 谈何持续进化?一图看懂火山 Mem0:解锁 Agent 持续学习与进化之路
人工智能
冬奇Lab15 小时前
Workflow 系列(06):安全——跨步骤注入传播与四层防御
人工智能·工作流引擎
冬奇Lab15 小时前
每日一个开源项目(第149篇):RAG-Anything - 把图片、表格、公式当成一等公民的多模态 RAG 框架
人工智能·开源
米小虾16 小时前
AI Agent 安全实战指南:当智能体开始"不听话",开发者该如何应对?
人工智能·安全·agent
IT_陈寒17 小时前
Vite的热更新突然不香了,排查三小时差点砸键盘
前端·人工智能·后端
这个DBA有点耶18 小时前
NULL不是空——数据库里最反直觉的设计,90%新人踩过的坑
数据库·mysql·代码规范
阿里云大数据AI技术19 小时前
构建高转化海外电商搜索:阿里云OpenSearch行业算法版的全链路智能优化策略实战
人工智能·搜索引擎
Awu122719 小时前
⚡从零开发 Agent CLI(五)实现一个可治理、可扩展的工具系统
前端·人工智能·claude
字节跳动视频云技术团队19 小时前
让 Agent 成为音视频工作台:AI MediaKit CLI + Skill 发布
人工智能·音视频开发