0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

用C语言实现一个全连接层和激活函数ReLU

OpenFPGA 来源:OpenFPGA 2023-03-03 09:43 次阅读

在上一篇文章中,我们用C语言实现了一个卷积层,并查看了结果。在本文中,我们将实现其余未实现的层:全连接层、池化层和激活函数 ReLU。

每一层的实现

全连接层

全连接层是将输入向量X乘以权重矩阵W,然后加上偏置B的过程。下面转载第二篇的图,能按照这个图计算就可以了。

d1d45f6a-b8ec-11ed-bfe3-dac502259ad0.png

全连接层的实现如下。

voidlinear(constfloat*x,constfloat*weight,constfloat*bias,
int64_tin_features,int64_tout_features,float*y){
for(int64_ti=0;i< out_features; ++i) {
    float sum = 0.f;
    for (int64_t j = 0; j < in_features; ++j) {
      sum += x[j] * weight[i * in_features + j];
    }
    y[i] = sum + bias[i];
  }
}

该函数的接口和各个数据的内存布局如下。

考虑稍后设置 PyTorch 参数,内存布局与 PyTorch 对齐。

输入

x: 输入图像。shape=(in_features)

weight: 权重因子。shape=(out_features, in_features)

bias: 偏置值。shape=(out_features)

输出

y: 输出图像。shape=(out_features)

参数

in_features: 输入顺序

out_features: 输出顺序

在全连接层中,内部操作数最多为out_channels * in_channels一个,对于典型参数,操作数远低于卷积层。

另一方面,关注权重因子,卷积层为shape=(out_channels, in_channels, ksize, ksize),而全连接层为shape=(out_features, in_features)。

例如,如果层从卷积层变为全连接层,in_features = channels * width * height则以下关系成立。width, height >> ksize考虑到这一点,在很多情况下,全连接层参数的内存需求大大超过了卷积层。

由于FPGA内部有丰富的SRAM缓冲区,因此擅长处理内存访问量大和内存数据相对于计算总量的大量复用。

单个全连接层不会复用权重数据,但是在视频处理等连续处理中,这是一个优势,因为要进行多次全连接。

另一方面,本文标题中也提到的边缘环境使用小型FPGA,因此可能会出现SRAM容量不足而需要访问外部DRAM的情况。

如果你有足够的内存带宽,你可以按原样访问它,但如果你没有足够的内存带宽,你可以在参数调整和训练后对模型应用称为剪枝和量化的操作。

池化层

池化层是对输入图像进行缩小的过程,这次使用的方法叫做2×2 MaxPooling。在这个过程中,取输入图像2x2区域的最大值作为输出图像一个像素的值。这个看第二张图也很容易理解,所以我再贴一遍。

d1e5c2dc-b8ec-11ed-bfe3-dac502259ad0.png

即使在池化层,输入图像有多个通道,但池化过程本身是针对每个通道独立执行的。因此,输入图像中的通道数和输出图像中的通道数在池化层中始终相等。

池化层的实现如下所示:

voidmaxpool2d(constfloat*x,int32_twidth,int32_theight,int32_tchannels,int32_tstride,float*y){
for(intch=0;ch< channels; ++ch) {
    for (int32_t h = 0; h < height; h += stride) {
      for (int32_t w = 0; w < width; w += stride) {
        float maxval = -FLT_MAX;

        for (int bh = 0; bh < stride; ++bh) {
          for (int bw = 0; bw < stride; ++bw) {
            maxval = std::max(maxval, x[(ch * height + h + bh) * width + w + bw]);
          }
        }

        y[(ch * (height / stride) + (h / stride)) * (width / stride) + w / stride] = maxval;
      }
    }
  }
}

这个函数的接口是:

此实现省略了边缘处理,因此图像的宽度和高度都必须能被stride整除。

输入

x: 输入图像。shape=(channels, height, width)

输出

y: 输出图像。shape=(channels, height/stride, width/stride)

参数

width: 图像宽度

height: 图像高度

stride:减速比

ReLU

ReLU 非常简单,因为它只是将负值设置为 0。

voidrelu(constfloat*x,int64_tsize,float*y){
for(int64_ti=0;i< size; ++i) {
    y[i] = std::max(x[i], .0f);
  }
}

由于每个元素的处理是完全独立的,x, y因此未指定内存布局。

硬件生成

到这里为止的内容,各层的功能都已经完成了。按照上一篇文章中的步骤,可以确认这次创建的函数也产生了与 libtorch 相同的输出。

此外,Vivado HLS 生成了一个通过 RTL 仿真的电路。从这里开始,我将简要说明实际生成了什么样的电路。

如果将上述linear函数原样输入到 Vivado HLS,则会发生错误。这里,将输入输出设为指针->数组是为了决定在电路制作时用于访问数组的地址的位宽。

另外,in_features的值为778=392,out_将features的值固定为32。这是为了避免Vivado HLS 在循环次数可变时输出性能不佳。

staticconststd::size_tkMaxSize=65536;

voidlinear_hls(constfloatx[kMaxSize],constfloatweight[kMaxSize],
constfloatbias[kMaxSize],floaty[kMaxSize]){
dnnk::linear(x,weight,bias,7*7*8,32,y);
}

linear_hls函数的综合报告中的“性能估计”如下所示:

d1fd7c74-b8ec-11ed-bfe3-dac502259ad0.png

在Timing -> Summary中写入了综合时指定的工作频率,此时的工作频率为5.00 ns = 200MHz。

重要的是 Latency -> Summary 部分,它描述了执行此函数时的周期延迟(Latency(cycles))和实时延迟(Latency(absolute))。看看这个,我们可以看到这个全连接层在 0.566 ms内完成。

在 Latency -> Detail -> Loop 列中,描述了每个循环的一次迭代所需的循环次数(Iteration Latency)和该循环的迭代次数(Trip Count)。

延迟(周期)包含Iteration Latency * Trip Count +循环初始化成本的值。Loop 1 是out_features循环到loop 1.1 in_features。

在Loop1.1中进行sum += x[j] * weight[i * in_features + j]; 简单计算会发现需要 9 个周期才能完成 Loop 1.1 所做的工作。

使用HLS中的“Schedule Viewer”功能,可以更详细地了解哪些操作需要花费更多长时间。

下图横轴的2~10表示Loop1.1的处理内容,大致分为x,weights等的加载2个循环,乘法(fmul)3个循环,加法(fadd)4个循环共计9个循环。

d21899a0-b8ec-11ed-bfe3-dac502259ad0.png

在使用 HLS 进行开发期间通过添加#pragma HLS pipeline指令,向此代码添加优化指令以指示它创建高效的硬件。

与普通的 FPGA 开发类似,运算单元的流水线化和并行化经常用于优化。通过这些优化,HLS 报告证实了加速:

流水线:减少迭代延迟(min=1)

并行化:减少行程次数,删除循环

正如之前也说过几次的那样,这次的课程首先是以FPGA推理为目的,所以不会进行上述的优化。

最后,该函数的接口如下所示。

d245d35c-b8ec-11ed-bfe3-dac502259ad0.png

由于本次没有指定接口,所以数组接口如x_ 等ap_memory对应FPGA上可以1个周期读写的存储器(BRAM/Distributed RAM)。

在下一篇文章中,我们将连接每一层的输入和输出,但在这种情况下,我们计划连接 FPGA 内部的存储器作为每一层之间的接口,如本例所示。

总结

在本文中,我们实现了全连接层、池化层和 ReLU。现在我们已经实现了所有层,我们将在下一篇文章中组合它们。之后我们会实际给MNIST数据,确认我们可以做出正确的推论。






审核编辑:刘清

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • FPGA
    +关注

    关注

    1625

    文章

    21639

    浏览量

    601355
  • DRAM
    +关注

    关注

    40

    文章

    2301

    浏览量

    183229
  • sram
    +关注

    关注

    6

    文章

    762

    浏览量

    114600
  • C语言
    +关注

    关注

    180

    文章

    7596

    浏览量

    135944
收藏 人收藏

    评论

    相关推荐

    C语言实现:见缝插针游戏!代码思路+源码分享

    见缝插圆我们昨天已经C语言实现了,今天将实现见缝插针的游戏。
    发表于 12-05 11:02 695次阅读

    介绍C语言实现的http下载器

    做OTA升级功能时,能直接拿到的往往只是升级包的链接,需要我们自己去下载,这时候就需要用到http下载器,下文介绍C语言实现的http下载器。
    发表于 02-22 10:25 370次阅读

    如何用C语言实现面向对象编程

    1 C语言实现面向对象编程GOF的《设计模式》书的副标题叫做“可复用面向对象软件的基础”,从标题就能看出面向对象是设计模式基本思想。由于C
    发表于 07-12 07:24

    C语言实现的泛型函数swap()

    C语言实现的泛型函数swap():交换两变量中的数据.
    发表于 01-20 07:10

    正余弦函数曲线的C语言绘制方法

    今天来学习小小的实例,使用C语言实现在屏幕上“*”显示0-360度的余弦函数cos(x)曲
    发表于 11-15 18:25 36次下载

    正余弦函数曲线的C语言绘制方法

    今天来学习小小的实例,使用C语言实现在屏幕上“*”显示0-360度的余弦函数cos(x)曲
    发表于 11-15 18:25 16次下载

    C语言实现DES算法

    C语言实现DES算法 本DES算法,使用了效率很高的C完成。目前,国内知名企业的POS终端中,单DES算法,均是采用这个函数完成。
    发表于 01-16 10:09 3258次阅读

    C语言实现FFT算法

    C语言实现FFT算法 /*****************fft programe*********************/#include "typedef.h" #include "math.h" struct
    发表于 10-30 13:39 6324次阅读

    C语言实现数字滤波

    C语言实现数字滤波 。对数值进行精度调整,下面给出几种常用的数字滤波方法的C语言函数,这些
    发表于 07-26 15:44 10次下载

    4重要算法C语言实现源代码

    4重要算法C语言实现源代码
    发表于 06-10 08:00 12次下载

    激活函数中sigmoid、ReLU函数些性质

    非线性:当激活函数是线性的时候,的神经网络就可以基本逼近所有的函数,但是,如果
    的头像 发表于 08-02 14:52 1.1w次阅读
    <b class='flag-5'>激活</b><b class='flag-5'>函数</b>中sigmoid、<b class='flag-5'>ReLU</b>等<b class='flag-5'>函数</b>的<b class='flag-5'>一</b>些性质

    使用C语言实现图书馆管理系统的源代码和函数及软件等资料合集

    本文档的主要内容详细介绍的是使用C语言实现图书馆管理系统的源代码和函数及软件等资料合集
    发表于 03-03 08:00 9次下载
    使用<b class='flag-5'>C</b><b class='flag-5'>语言实现</b>图书馆管理系统的源代码和<b class='flag-5'>函数</b>及软件等资料合集

    在PyTorch中使用ReLU激活函数的例子

    PyTorch已为我们实现了大多数常用的非线性激活函数,我们可以像使用任何其他的那样使用它们。让我们快速看
    的头像 发表于 07-06 15:27 2473次阅读

    怎么C语言实现多态

    这里我想主要介绍下在C语言中是如何实现的面向对象。知道了C语言实现面向对象的方式,我们再联想下,C
    的头像 发表于 10-12 09:12 1988次阅读

    使用C语言实现函数模板

      C语言能不能实现通用的函数,既能完成整数的
    的头像 发表于 11-09 11:38 218次阅读