Skip to content

Latest commit

 

History

History
52 lines (36 loc) · 3.99 KB

using-block-prefetch-for-optimized-memory-performance.org

File metadata and controls

52 lines (36 loc) · 3.99 KB

Using Block Prefetch for Optimized Memory Performance

AMD_block_prefetch_paper.pdf 这篇文章比较早,2001年AMD公司写的。里面使用了两个例子来说明如何有效地进行block prefetch来优化内存性能。一个例子是memcpy, 另外一个则是将两个浮点数组相加,这里面就只说第一个例子。

memcpy是个比较经典的例子,文章中也列举出来了各种方法来进行加速,大部分方法也是广为人知。这里我想摘抄一下里面是如何做block prefetch的。最开始他们使用的指令是 “prefetchnta”,这个指令对于CPU来说只是一个hint, 在执行的时候其实完全可以忽略的。为了”真实”地进行block prefetch, 我们可以使用mov指令。

Significantly, the MOV instruction is used, rather than the software prefetch instruction. Unlike a prefetch instruction, which is officially only a “hint” to the processor, a MOV instruction cannot be ignored and must be executed in-order. The result is that the memory system reads sequential, back-to-back address blocks, which yields the fastest memory read bandwidth.

../images/block-prefetch-memcpy.png

这里一个CACHEBLOCK是L2 Cache大小。大致意思就是,在真正地去读src内容之前,先对一个cache block的内容进行“真正”地prefetch, 我们只取每个cache line上的头4字节,这样CPU会将对应的这条cache line也放入进来。注意这里要求src地址必须是和64字节对齐的,否则就没有这个效果了。另外注意在写入到dst的时候,使用的是movntq是不会涉及到cache line的,最后需要做一下sfence(因为我们使用了ntq这样的指令)。

还有一个小点需要注意,这里prefetch的顺序是逆序的。如果按照顺序进行加载的话,CPU可能会触发不必要的读请求。

One additional trick is to read the cache lines in descending address order, rather than ascending order. This can improve performance a bit, by keeping the processor’s hardware prefetcher from issuing any redundant read requests.


文章最后给了一些老生常谈的优化建议:

Block prefetch and three phase processing are general techniques for improving the performance of memory-intensive applications on PCs. In a nutshell, the key points are:

#1 To get the maximum memory read bandwidth, read data into the cache in large blocks (e.g. 1K to 8K bytes), using block prefetch. When creating a block prefetch loop:

  • unroll the loop by at least 2X
  • use the MOV instruction (not the Prefetch instruction)
  • read only one address per cache line
  • read data into an ALU scratch register, like EAX
  • read only one linear stream per loop
  • to prefetch several streams, use a separate loop for each – read cache lines in descending address order
  • make sure all data is aligned

#2 To get maximum memory write bandwidth, write data from the cache to main memory in large blocks, using streaming store instructions. When creating a memory write loop:

  • use the MMX registers to pass the data – read from cache
  • use MOVNTQ for writing to memory
  • make sure the data is aligned
  • write every address, in ascending order, without any gaps – end with an SFENCE to flush the write buffer

#3 Whenever possible, code that actually “does the real work” should be reading its data from cache, and writing its output to an in-cache buffer. To enable this to happen, use #1 and #2 above.

还有就是hot-path循环指令最好是和64字节对齐(文章里面写的是16字节,但是我觉得应该是64字节)

Aligning “hot” branch targets to 16 byte boundaries can improve speed, by maximizing the number of instruction fills into the instruction-byte queue. This is especially important for short loops, like a block prefetch loop. This wasn’t shown in the code examples, for the sake of readability. It can be done with the ALIGN pragma, like this:

align 16
prefetchloop:
  mov ebx, [esi+ecx*8-64]
  mov ebx, [esi+ecx*8-128]
  sub ecx, 16
  dec eax
  jnz  prefetchloop