Skip to content

Latest commit

 

History

History
47 lines (47 loc) · 5.7 KB

File metadata and controls

47 lines (47 loc) · 5.7 KB

Programming Massively Parallel Processors

介绍

此项目为学习《Programming Massively Parallel Processors》时的代码记录,主要是第7~12章的并行计算模式代码。 主要包含Convolution, Prefix Sum, Histogram Computation, Sparse Matrix Computation, Merge Sort, Graph Search 6大计算模式。 由于书中没有完整代码,所以此项目中代码并不一定是最优解,只是按照书中讲解自我推理得来。

简单笔记

Convolution

  • 卷积核体积小,初始化后不会变,适合使用__constant__保存
  • 图像体积大,需要使用tile策略,同一个像素点可能被block中多个thread使用,适合读入__shared__中
  • 图像数据读入__shared__中时,尽量保证coalescing的方式读入
  • 对于halo cells可以不用特殊处理,因为其他block在将其载入__shared__中时L2 Cache会将其缓存起来,

Prefix Sum

  • 浮点数计算存在大数吃小数的情况,可以使用Kahan算法进行一定程度的补偿
  • 按照Brent Kung和Kogge Stone累加算法,读取数据效率不高,故采用corner tuning的方式,将数据先以coalescing的方式载入__shared__
  • 通常情况下Kogge Stone算法直观性更好,而Brent Kung则是在速度和前者差不多的情况下功耗更低
  • 对于任意长度的累加序列,可以采用三段法。但是由于要多次读写全局内存,效率十分底下,可以采用Single-Pass的方式。 起难点在于需要实现block之间的通信和同步,采用全局内存+原子操作解决。

Histogram

  • 直方图统计需要用到atomicAdd原子操作,而原子操作是否受限于内存读取速率,因为同一个内存地址同时只能进行一个原子操作,效率十分低下。而Cache也能一定程度的缓解这个问题
  • 对于超大序列的统计需要用到多个block并行,而太多的请求即使在Cache中也难以解决,故考虑私有化,每个block中先用__shared__进行统计,最后再汇总
  • 对于某些统计,例如图像直方图统计中,存在多个连续相同地址的atomicAdd操作请求,故考虑聚合,将多次请求合并为一次,提升效率
  • 书中关于Aggregation的代码Figure.9.11有错误,每次累加的位置应当是prev_index坐标的地址。

Sparse Matrix

  • 稀疏矩阵如果按照密集矩阵存储计算,则会浪费许多存储空间以及计算资源。CSR,ELL,COO为三种常用的存储方式。
  • 通常来说CSR存储空间最少,但是对于老式CUDA显卡,存在divergence和non-coalescing。
  • ELL采用Padding的方式,以保证每次数据读取的一致性,但是当某一行的非零元素特别多时,整体存储空间就会变得特别大,效率低。
  • COO类似于CSR,采用data,col_idx,row_idx来存储,具有非常好的灵活性,但是只有当稀疏性<1/3才能有压缩的性能。
  • Hybrid(ELL + COO)是一种不错的选择,将某行特别长的元素用COO存储,剩下的用ELL存储。
  • JDS是一种按row划分,并按照非零元素长度进行排序的算法。排序好后适合CUDA按照block划分处理,每一个block中的threads处理的数据长度比较类似。
  • 老式CUDA设备上JDS-ELL效果更好,新式CUDA设备上对于对齐的要求比较低,JDS-CSR效果更好
  • 通常来说,完全随机=>ELL,完全随机但行方差大=>ELL+COO,近似对角=>JDS,极度稀疏=>COO

Merge Sort

  • 这一部分主要是讲解合并两个有序序列,也就是归并排序的核心部分。我的代码只实现了non-coalescing方式,使用shared memory的部分实在太复杂了,以后有空再更
  • 假设输入A,B,合并之后为C,长度分别为m,n,m + n,那么对于C中任意位置k,必定是从A[i]]或者B[j]得来,并且k = i + j。并且可知A[i-1] <= B[j] && A[i] > B[j-1]
  • 由上可知,对于C的任意位置k都可以确定对应的i和j,于是可以将合并算法并行化。
  • 由k确定i和j位置的算法称为co_rank,采用二分查找。书中采用控制i和j的low-bound,但是实际上k = i + j我们只需要控制i就可以了,所以我的代码里面采用的是标准的二分查找
  • 此时数据读取方式是stride的方式而不是coalescing,故采用shared memory的方式。每一个block采用tiled进行循环解决任意长度的序列。每次先将AB tiled长度数据读入共享内存中,再合并,再读取
  • 上述的方法读取数据方式coalescing,但是每次只使用了tiled个数据,还有tiled个数据未被使用又重新读入,整体上来说浪费了一半的带宽。
  • 解决此问题的方式是引入一个变量记录每次消耗共享内存之后的Index,下一次迭代时候只读取被消耗长度的数据并填入到已经消耗的部分,此时共享内存类似一个循环队列。
  • 上述方法可以完美节省带宽,提升效率,但是代码复杂度会大大提高,并且会大幅增加寄存器使用量

Graph Search

  • 图搜索中的广度优先遍历BFS可以改写为并行编程模式。其核心部分在于遍历当前队列中所有的节点以便于寻找下一级别节点的部分,可以用多线程完成。
  • 由于整体需要用到大量的原子操作,会拖慢程序速度,故可以考虑私有化的方式,而同一warp中仍然存在竞争关系,可以考虑采用3级缓存的方式,即使用sub-queue
  • 对压缩的稀疏矩阵CSR遍历时,row_ptr访问不规则,可以用纹理内存解决no-coalescing的问题
  • 当cur_frontier队列中的元素少时,启动核函数开销过大,可以采用设置阈值的方式自动选择序列化求解或者并行求解(我代码里面没有加,因为和普通的重复了,而且很简单)
  • data load imbalance的问题可以用动态编程的方式解决