色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何對GPU中的矩陣乘法(GEMM)進行優(yōu)化

perfxlab ? 來源:澎峰科技PerfXLab ? 2023-05-25 09:03 ? 次閱讀

本篇文章主要是介紹如何對GPU中的矩陣乘法(GEMM)進行優(yōu)化。目前針對GEMM的優(yōu)化,網(wǎng)絡(luò)上已經(jīng)有非常多的教程和示例了。大部分的重要資料我都看了看。但總的來說,還是不夠接地氣,然后理解起來還是會比較費解。所以希望寫這么一篇文章,盡可能地去把GPU的GEMM優(yōu)化說清楚,說明白。然后讓小白讀者也能通過這么一兩篇文章去更好地了解GEMM優(yōu)化的相關(guān)技術(shù)。

不像上次的reduce優(yōu)化一樣,能一篇文章說完。這次的GEMM優(yōu)化會分為三個部分。第一個部分只說優(yōu)化思路和分析,沒有任何代碼,這么做考慮也是為了減輕讀者的負擔(dān),看代碼太累,盡可能地讓讀者先明白原理,為什么要這么做。第二個部分是對代碼的詳細解析,這個里面就是一行一行地去分析代碼。因為之前的很多博客進行了分析,但是代碼本身并沒有開源,或者說開源了代碼,但沒有解析,看起來太累了。我希望提供一個盡可能詳細的代碼解析,讀者看完之后能明白相關(guān)優(yōu)化技巧,并且可以直接把代碼拿去驗證使用。第三個部分主要涉及到匯編,最重要的是說明在NV的卡上,怎么去解決寄存器的bank沖突來獲取極致的性能。

本篇文章是GEMM優(yōu)化的第一個部分,在這篇文章中,只說優(yōu)化思路和分析

前言

在高性能領(lǐng)域,對于矩陣乘(GEMM)的優(yōu)化是一個非常重要的課題。GEMM可以非常廣泛地應(yīng)用于航空航天、流體力學(xué)等科學(xué)計算領(lǐng)域,這也是之前HPC的主要應(yīng)用場景。后來深度學(xué)習(xí)開展地如火如荼,由于對高算力的需要,也成為HPC的主要應(yīng)用場景之一。這些年涌現(xiàn)了一系列的深度學(xué)習(xí)模型。模型里面最耗時的東西,包括卷積、全連接層、attention,都可以轉(zhuǎn)換成GEMM操作。所以說,GEMM優(yōu)化的重要性,怎么突出都不過分。

目前網(wǎng)上能找到的針對GEMM優(yōu)化的資料主要有這么幾個方面:(1)論文,目前針對GPU進行GEMM優(yōu)化的論文非常多,這里主要推薦Understanding the GPU Microarchitecture和Fast implementation of dgemm on fermi gpu以及Dissecting the NVIDIA Volta GPU Architecture via Microbenchmarking。這幾篇論文在業(yè)界都比較有影響力,就是代碼開源方面做的不算太好。(2)官方博客,主要是CUTLASS和NervanaSystems-SGEMM優(yōu)化。還有前段時間曠視發(fā)的文章CUDA矩陣乘法優(yōu)化,寫的都很詳細。(3)github的一些demo,代碼量不大,看起來比較舒服。我是看了這兩個:

demo1 :

https://github.com/Cjkkkk/CUDA_gemm

demo2 :

https://github.com/yzhaiustc/Optimizing-SGEMM-on-NVIDIA-Turing-GPUs

demo1代碼寫的好理解一些,但是優(yōu)化工作沒做完全,沒有做到prefetch。demo2是效果很好,11個優(yōu)化技巧,不斷逼近cublas。但是代碼真的看起來比較難受,最重要的很多參數(shù)寫死了,不好去調(diào)。 總而言之,目前列舉的上述資料存在著這么兩個問題:(1)文檔方面,讀起來還是比較費勁,對于小白來說,還是不夠簡單不夠傻,看起來太累了;(2)代碼方面,要么是沒公開代碼,要么是代碼太多了,看不下去;還有的就是代碼可讀性很強,但是優(yōu)化工作還不是特別深,或者就是代碼優(yōu)化做的很好,但是可讀性差了。方方面面總是有點欠缺,所以希望能夠?qū)懸黄M可能地在文檔上簡單明了,在代碼上詳細且可讀性好的文章。當(dāng)然,這是一個逐步迭代的過程,所以這篇文章也會持續(xù)進行更新哈。 本篇文章主要是采納了cutlass的行文思路,主要介紹GEMM中的數(shù)據(jù)分塊和如何在多級存儲進行數(shù)據(jù)搬運。這也是HPC優(yōu)化的核心思想,怎么樣讓數(shù)據(jù)放在更近的存儲上來掩蓋計算的延時,從而減少存儲墻的影響。文章分為四個方面進行敘述,首先介紹在global memory層面如何進行分塊以及數(shù)據(jù)搬運,隨后介紹在shared memory層面如何進行分塊以及數(shù)據(jù)搬運,而后介紹在register層面如何進行分塊以及避免bank沖突,最后介紹如何進行prefetch以更好地掩蓋訪存時延。

一、從global memory到shared memory

假設(shè)有矩陣A、B,需要計算矩陣A和B的乘,即矩陣C。A、B、C三個矩陣的維度分別為,,,且三個矩陣中的數(shù)據(jù)都是單精度浮點數(shù)。對于C中每一個元素,C[i][j],可以看作是A的一行和B的一列進行一次歸約操作。采用最naive的GEMM算法,在GPU中,一共開啟個線程,每個線程需要讀取矩陣A的一行與矩陣B的一列,而后將計算結(jié)果寫回至矩陣C中。因而,完成計算一共需要從global memory中進行次讀操作和次寫操作。大量的訪存操作使得GEMM效率難以提高,因而考慮global memory中進行分塊,并將矩陣塊放置到shared memory中。其示意圖如下: 5fe8a226-fa8c-11ed-90ce-dac502259ad0.jpg 對global memory進行分塊的GEMM算法示意圖見上圖右側(cè)。首先將A、B、C三個矩陣劃分為多個維度為?,,?的小矩陣塊。三個矩陣形成?,,?的小矩陣網(wǎng)格。其中,,,。隨后在GPU中開啟??個block,每個block負責(zé)C中一個維度為??的小矩陣塊的計算。計算中一共有K次迭代,每一次迭代都需要讀取A中一個維度為??的小矩陣塊和B中一個維度為??的小矩陣塊,并將其放置在shared memory中。因而,完成C中所有元素的計算一共需要從global memory中讀取?,即??個單精度浮點數(shù)。相比于naive的GEMM算法,訪存量減少為原來的?。通過global memory中分塊算法極大地減少了對global memory的訪存量。并且,相比于naive算法,對global進行分塊可以更充分地利用數(shù)據(jù)局部性。在naive算法中,每一個線程都需要直接從global memory中取數(shù),其時延非常長,計算性能非常差。而進行分塊后,將維度為?,?的小矩陣塊先存儲到shared memory之中。而后計算單元進行計算時可以直接從shared memory中取數(shù),大大減少了訪存所需要的時延。

二、從shared memory到register

隨后,我們進一步考慮從shared memory到register的過程。在這里,只分析一個block中的計算。當(dāng)進行K輪迭代中某一輪迭代時,GPU將維度為,的小矩陣塊存儲到shared memory中,而后各個線程將shared memory中的數(shù)據(jù)存入register中進行計算。 601b8984-fa8c-11ed-90ce-dac502259ad0.jpg不對shared memory分塊時,一個block中含有個線程,每一個線程負責(zé)C中一個元素的計算。則一個block一共需要對shared memory進行次讀操作。而后考慮對shared memory進行分塊,對的小矩陣進行再一次劃分,將其劃分為多個維度為的子矩陣。則一個block需要負責(zé)個子矩陣。其中,,。隨后,在一個block中開啟個線程,每個線程負責(zé)一個維度為的子矩陣的計算。在計算中,一個block一共需要從shared memory讀取,即個單精度浮點數(shù)。相比于未分塊的算法,對于shared memory中的訪存量減少為原來的。并且,由于將數(shù)據(jù)放入register中,可以直接對數(shù)據(jù)進行運算,減少了從shared memory中取數(shù)的時延。

三、register分塊

在這里,我們考慮最后一層,即register中的計算,并且只分析一個thread。在完成以上的過程后,對于一個線程而言,它現(xiàn)在擁有:個A矩陣的寄存器值,個B矩陣的寄存器值,以及個C矩陣的寄存器值。通過這些寄存器的值,需要計算個數(shù)。這需要條FFMA指令。 這個時候會涉及到寄存器的bank conflict。在NV的GPU中,每個SM不僅會產(chǎn)生shared memroy之間的bank 沖突,也會產(chǎn)生寄存器之間的bank沖突。這一點對于計算密集型的算子十分重要。像shared memory一樣,寄存器的Register File也會被分為幾個bank,如果一條指令的的源寄存器有2個以上來自同一bank,就會產(chǎn)生沖突。指令會重發(fā)射,浪費一個cycle。PS:這個地方是從曠視的博客中看的。然后對于maxwell架構(gòu)的GPU而言,bank數(shù)為4,寄存器id%4即所屬bank。 我們假設(shè)對這個thread來說,、。并且計算C的寄存器以一種非常naive的情況分配,如下圖左側(cè)所示。則需要產(chǎn)生16條FFMA指令,列舉如下:

FFMA R0, R16, R20, R0 FFMA R1, R16, R21, R1 ……

604de352-fa8c-11ed-90ce-dac502259ad0.jpg

可以從中看出,這會產(chǎn)生大量的register bank沖突,所以需要對參與計算的寄存器重新進行分配和排布,如上圖右側(cè)所示。在有些地方,這種方式也可以叫做register分塊。

四、數(shù)據(jù)的prefetch

最后,我們來講講如何通過對數(shù)據(jù)進行prefetch來減少訪存的latency。我們再來回顧GEMM的過程,并且仔細地看看這個訪存的latency到底是怎么導(dǎo)致的。對于一個block而言,需要計算一個的矩陣塊,這個時候需要進行K次迭代,每次迭代都需要先將來自A和B的兩個小塊送到shared memory中再進行計算。而從global中訪存實際上是非常慢的,所以導(dǎo)致了latency。雖然GPU中可以通過block的切換來掩蓋這種latency,但是由于分配的shared memory比較多,活躍的block并不太多,這種延時很難被掩蓋。對于一個thread,需要計算一個的小矩陣,但是必須先將數(shù)據(jù)從shared memory傳到寄存器上,才能開始進行計算。所以導(dǎo)致了每進行一次迭代,計算單元就需要停下來等待,計算單元不能被喂飽。

為此,需要進行數(shù)據(jù)的Prefetch來盡可能地掩蓋這種latency。思想也比較簡單,需要多開一個buffer,進行讀寫分離。示意圖如下。當(dāng)block進行第2輪迭代時,需要對A2和B2進行計算,在計算單元進行計算的同時,我們將A3和B3提前放置到shared memory。而后,在進行第3輪迭代時,就可以直接對shared memory中的A3和B3進行計算,而不需要等待從global memory搬運到shared memory的時間。寄存器上的Prefetch也是同理。

607caea8-fa8c-11ed-90ce-dac502259ad0.jpg

總結(jié)

GEMM的優(yōu)化思想,基本上就是這么幾方面的內(nèi)容。希望大家通過介紹能夠?qū)EMM的優(yōu)化有一個比較直觀且具體的理解。

審核編輯:彭靜
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 寄存器
    +關(guān)注

    關(guān)注

    31

    文章

    5336

    瀏覽量

    120230
  • gpu
    gpu
    +關(guān)注

    關(guān)注

    28

    文章

    4729

    瀏覽量

    128890
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4779

    瀏覽量

    68521
  • 澎峰科技
    +關(guān)注

    關(guān)注

    0

    文章

    55

    瀏覽量

    3168

原文標(biāo)題:深入淺出GPU優(yōu)化系列:GEMM優(yōu)化(一)

文章出處:【微信號:perfxlab,微信公眾號:perfxlab】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    Mali GPU編程特性及二維浮點矩陣運算并行優(yōu)化詳解

    本文針對Mali-T604 GPU論述了基于OpenCL的Linux平臺上進行通用計算并行優(yōu)化的方法,論述了Mali-T604 GPU的硬件特點,并基于OpenCL設(shè)計了二維
    發(fā)表于 08-07 17:12 ?2560次閱讀
    Mali <b class='flag-5'>GPU</b>編程特性及二維浮點<b class='flag-5'>矩陣</b>運算并行<b class='flag-5'>優(yōu)化</b>詳解

    使用CUDA并行化矩陣乘法加速Blender Python

      這篇文章描述了兩種不同的加速矩陣乘法的方法。第一種方法使用 Numba 編譯器來減少 Python 代碼與循環(huán)相關(guān)的開銷。第二種方法使用 CUDA 并行化矩陣
    的頭像 發(fā)表于 04-24 17:04 ?5544次閱讀
    使用CUDA并行化<b class='flag-5'>矩陣</b><b class='flag-5'>乘法</b>加速Blender Python

    FPGA 超越 GPU,問鼎下一代深度學(xué)習(xí)主引擎

    四個int6打包到一個DSP模塊,研究了FPGA的Int6 GEMM。對于本來不支持Int6 的GPU,他們使用了Int8 GPU 的峰值性能進行
    發(fā)表于 04-27 14:10

    講解矩陣運算的放縮,乘法和轉(zhuǎn)置

    第22章 DSP矩陣運算-放縮,乘法和轉(zhuǎn)置矩陣本期教程主要講解矩陣運算的放縮,乘法和轉(zhuǎn)置。目錄
    發(fā)表于 08-11 06:05

    主要講解矩陣運算的放縮,乘法和轉(zhuǎn)置

    第22章 DSP矩陣運算-放縮,乘法和轉(zhuǎn)置矩陣本期教程主要講解矩陣運算的放縮,乘法和轉(zhuǎn)置。目錄
    發(fā)表于 08-11 08:41

    解讀最佳實踐:倚天 710 ARM 芯片的 Python+AI 算力優(yōu)化

    更好的性能,或者更好的性價比。所以說如何整合 Python+AI 的相關(guān)軟件使其發(fā)揮最好的性能成為了我們關(guān)注的重點。下文的分享整體分為兩部分,一部分是介紹我們進行優(yōu)化工作,主要是跟矩陣乘法
    發(fā)表于 12-23 16:02

    Adreno GPU 矩陣乘法——第1講:OpenCL優(yōu)化

    SGEMM采用的相同原理進行有效實現(xiàn)。 一般來說,我們對Adreno GPU優(yōu)化的MM實現(xiàn)比簡單實現(xiàn)至少快兩個數(shù)量級。 接下來? 在下一篇文章,我將給出這些概念背后的OpenCL代碼
    發(fā)表于 09-18 19:15 ?1812次閱讀

    使用英特爾數(shù)學(xué)核心函數(shù)庫優(yōu)化三重嵌套循環(huán)矩陣乘法

    我們使用英特爾?數(shù)學(xué)核心函數(shù)庫(MKL)在Linux *上優(yōu)化了三重嵌套循環(huán)矩陣乘法的版本。
    的頭像 發(fā)表于 11-07 06:04 ?3596次閱讀

    基于GPU的稀疏矩陣存儲格式優(yōu)化綜述

    基于GPU的稀疏矩陣存儲格式優(yōu)化綜述
    發(fā)表于 06-11 11:45 ?18次下載

    深度學(xué)習(xí)矩陣乘法計算速度再次突破

    n階矩陣乘法最優(yōu)解的時間復(fù)雜度再次被突破,達到了 。 按定義直接算的話,時間復(fù)雜度是O(n3)。 光這么說可能不太直觀,從圖上可以看出,n足夠大時優(yōu)化后的算法就開始表現(xiàn)出明顯優(yōu)勢。 矩陣
    的頭像 發(fā)表于 06-24 17:36 ?2631次閱讀
    深度學(xué)習(xí)<b class='flag-5'>中</b><b class='flag-5'>矩陣</b><b class='flag-5'>乘法</b>計算速度再次突破

    使用CUTLASS實現(xiàn)高性能矩陣乘法

      CUTLASS 實現(xiàn)了高性能卷積(隱式 GEMM )。隱式 GEMM 是作為 GEMM 的卷積運算的公式。這允許 Cutslass 通過重用高度優(yōu)化的 warp-wide
    的頭像 發(fā)表于 04-15 10:03 ?2904次閱讀

    頂級FPGA和GPU的PK

    首先,文章使用GPU最擅長處理的工作負載:通用矩陣乘(GEMM)來跑GPU的benchmark(什么是GEMM請移步https://spat
    發(fā)表于 08-16 09:22 ?2589次閱讀

    CUDA矩陣乘法優(yōu)化手段詳解

    單精度矩陣乘法(SGEMM)幾乎是每一位學(xué)習(xí) CUDA 的同學(xué)繞不開的案例,這個經(jīng)典的計算密集型案例可以很好地展示 GPU 編程中常用的優(yōu)化技巧。本文將詳細介紹 CUDA SGEMM
    的頭像 發(fā)表于 09-28 09:46 ?1917次閱讀

    在TensorFlow對Tensor進行拆和裝

    TensorCore改進的方向就是針對矩陣乘法GEMM,General Matrix Mulitiplicaiton)運算進行優(yōu)化
    的頭像 發(fā)表于 12-29 09:24 ?731次閱讀

    NVIDIA Hopper GPU上的新cuBLAS12.0功能和矩陣乘法性能

    NVIDIA Hopper GPU 上的新 cuBLAS 12.0 功能和矩陣乘法性能
    的頭像 發(fā)表于 07-05 16:30 ?2408次閱讀
    NVIDIA Hopper <b class='flag-5'>GPU</b>上的新cuBLAS12.0功能和<b class='flag-5'>矩陣</b><b class='flag-5'>乘法</b>性能
    主站蜘蛛池模板: 胸大的姑娘中文字幕视频| 麻豆传煤网站网址入口在线下载 | 啊好大好厉害好爽真骚| 久久在精品线影院| 亚洲综合色婷婷在线影院| 国产一区免费在线观看| 色偷偷网址| 妇少水多18P蜜泬17P亚洲乱| 欧美xxxx性喷潮| ewp系列虐杀在线视频| 欧美18在线| ZZoo兽2皇| 色噜噜2017最新综合| 国产 欧美 亚洲 日韩视频| 色AV色婷婷66人妻久久久| 村妇偷人内射高潮迭起| 日韩无码在线| 国产精品高清免费网站| 午夜理论片YY4399影院| 国产人成无码视频在线观看| 性夜a爽黄爽| 好姑娘BD高清在线观看免费| 亚洲三级精品| 妹妹成人网| 大桥未久在线看| 午夜伦伦电影理论片大片| 国内高清在线观看视频| 怡春院国产精品视频| 男生互捏jiji的故事| 超碰在线97av视频免费| 午夜福利理论片在线播放| 狠狠色狠狠色狠狠五月ady | 久久高清内射无套| 2019在秋霞理论| 日本工口生肉全彩大全| 国产蜜臀AV在线一区视频| 一边吃奶一边啪啪真舒服| 内射白浆一区二区在线观看| 东日韩二三区| 一个人的视频全免费在线观看www 一个人的免费完整在线观看HD | 国产性色AV内射白浆肛交后入|