Real-Time Rendering Architectures

Similar documents
From Shader Code to a Teraflop: How GPU Shader Cores Work. Jonathan Ragan- Kelley (Slides by Kayvon Fatahalian)

From Shader Code to a Teraflop: How Shader Cores Work

GPU Architecture. Robert Strzodka (MPII), Dominik Göddeke G. TUDo), Dominik Behr (AMD)

Lecture 7: The Programmable GPU Core. Kayvon Fatahalian CMU : Graphics and Imaging Architectures (Fall 2011)

Antonio R. Miele Marco D. Santambrogio

Scientific Computing on GPUs: GPU Architecture Overview

Administrivia. HW0 scores, HW1 peer-review assignments out. If you re having Cython trouble with HW2, let us know.

GPU Programming. Lecture 1: Introduction. Miaoqing Huang University of Arkansas 1 / 27

Lecture 6: Texture. Kayvon Fatahalian CMU : Graphics and Imaging Architectures (Fall 2011)

GPU! Advanced Topics on Heterogeneous System Architectures. Politecnico di Milano! Seminar Room, Bld 20! 11 December, 2017!

CS GPU and GPGPU Programming Lecture 3: GPU Architecture 2. Markus Hadwiger, KAUST

CS GPU and GPGPU Programming Lecture 3: GPU Architecture 2. Markus Hadwiger, KAUST

Numerical Simulation on the GPU

GRAPHICS HARDWARE. Niels Joubert, 4th August 2010, CS147

Intro to Shading Languages

Fundamental CUDA Optimization. NVIDIA Corporation

Hardware/Software Co-Design

CSE 591/392: GPU Programming. Introduction. Klaus Mueller. Computer Science Department Stony Brook University

CSE 591: GPU Programming. Introduction. Entertainment Graphics: Virtual Realism for the Masses. Computer games need to have: Klaus Mueller

Fundamental CUDA Optimization. NVIDIA Corporation

GPGPUs in HPC. VILLE TIMONEN Åbo Akademi University CSC

CS GPU and GPGPU Programming Lecture 8+9: GPU Architecture 7+8. Markus Hadwiger, KAUST

CUDA OPTIMIZATIONS ISC 2011 Tutorial

Modern Processor Architectures. L25: Modern Compiler Design

Portland State University ECE 588/688. Graphics Processors

Finite Element Integration and Assembly on Modern Multi and Many-core Processors

Efficient and Scalable Shading for Many Lights

GPU Architecture & CUDA Programming

EE382N (20): Computer Architecture - Parallelism and Locality Spring 2015 Lecture 09 GPUs (II) Mattan Erez. The University of Texas at Austin

GPU for HPC. October 2010

Kernel optimizations Launch configuration Global memory throughput Shared memory access Instruction throughput / control flow

GRAPHICS PROCESSING UNITS

Accelerator cards are typically PCIx cards that supplement a host processor, which they require to operate Today, the most common accelerators include

Parallel Programming on Larrabee. Tim Foley Intel Corp

Modern Processor Architectures (A compiler writer s perspective) L25: Modern Compiler Design

CS427 Multicore Architecture and Parallel Computing

High Performance Computing on GPUs using NVIDIA CUDA

CSCI 402: Computer Architectures. Parallel Processors (2) Fengguang Song Department of Computer & Information Science IUPUI.

Challenges for GPU Architecture. Michael Doggett Graphics Architecture Group April 2, 2008

Spring 2010 Prof. Hyesoon Kim. AMD presentations from Richard Huddy and Michael Doggett

GPU Computation Strategies & Tricks. Ian Buck NVIDIA

Fundamental Optimizations

Supercomputing, Tutorial S03 New Orleans, Nov 14, 2010

Preparing seismic codes for GPUs and other

TUNING CUDA APPLICATIONS FOR MAXWELL

Lecture 23: Domain-Specific Parallel Programming

Design of Digital Circuits Lecture 21: GPUs. Prof. Onur Mutlu ETH Zurich Spring May 2017

Introduction to Multicore architecture. Tao Zhang Oct. 21, 2010

CPU Architecture Overview. Varun Sampath CIS 565 Spring 2012

Graphics Architectures and OpenCL. Michael Doggett Department of Computer Science Lund university

TUNING CUDA APPLICATIONS FOR MAXWELL

Windowing System on a 3D Pipeline. February 2005

CSCI-GA Graphics Processing Units (GPUs): Architecture and Programming Lecture 2: Hardware Perspective of GPUs

GPU Programming. Performance Considerations. Miaoqing Huang University of Arkansas Fall / 60

GPUs have enormous power that is enormously difficult to use

Improving Performance of Machine Learning Workloads

From Brook to CUDA. GPU Technology Conference

ECE 574 Cluster Computing Lecture 16

Introduction to CUDA Algoritmi e Calcolo Parallelo. Daniele Loiacono

Graphics Hardware. Graphics Processing Unit (GPU) is a Subsidiary hardware. With massively multi-threaded many-core. Dedicated to 2D and 3D graphics

Lecture 13: Memory Consistency. + a Course-So-Far Review. Parallel Computer Architecture and Programming CMU , Spring 2013

Introduction: Modern computer architecture. The stored program computer and its inherent bottlenecks Multi- and manycore chips and nodes

NVIDIA GTX200: TeraFLOPS Visual Computing. August 26, 2008 John Tynefield

ENGN1640: Design of Computing Systems Topic 06: Advanced Processor Design

The NVIDIA GeForce 8800 GPU

Threading Hardware in G80

Introduction. L25: Modern Compiler Design

CS8803SC Software and Hardware Cooperative Computing GPGPU. Prof. Hyesoon Kim School of Computer Science Georgia Institute of Technology

GPU Basics. Introduction to GPU. S. Sundar and M. Panchatcharam. GPU Basics. S. Sundar & M. Panchatcharam. Super Computing GPU.

Addressing the Memory Wall

A Code Merging Optimization Technique for GPU. Ryan Taylor Xiaoming Li University of Delaware

Graphics Processing Unit Architecture (GPU Arch)

Computer Architecture

ECE 571 Advanced Microprocessor-Based Design Lecture 20

GPUs and GPGPUs. Greg Blanton John T. Lubia

Lecture 5. Performance programming for stencil methods Vectorization Computing with GPUs

COSC 6385 Computer Architecture - Data Level Parallelism (III) The Intel Larrabee, Intel Xeon Phi and IBM Cell processors

Mattan Erez. The University of Texas at Austin


Anatomy of AMD s TeraScale Graphics Engine

CUDA and GPU Performance Tuning Fundamentals: A hands-on introduction. Francesco Rossi University of Bologna and INFN

CS 426 Parallel Computing. Parallel Computing Platforms

General Purpose GPU Programming. Advanced Operating Systems Tutorial 9

GPU Architecture. Michael Doggett Department of Computer Science Lund university

CME 213 S PRING Eric Darve

CS 179 Lecture 4. GPU Compute Architecture

Optimizing DirectX Graphics. Richard Huddy European Developer Relations Manager

Parallel programming: Introduction to GPU architecture. Sylvain Collange Inria Rennes Bretagne Atlantique

Martin Kruliš, v

COSC 6339 Accelerators in Big Data

What s New with GPGPU?

NVIDIA s Compute Unified Device Architecture (CUDA)

NVIDIA s Compute Unified Device Architecture (CUDA)

Mathematical computations with GPUs

Technology for a better society. hetcomp.com

Advanced CUDA Programming. Dr. Timo Stich

Introduction to CUDA Algoritmi e Calcolo Parallelo. Daniele Loiacono

Graphics Hardware, Graphics APIs, and Computation on GPUs. Mark Segal

XT Node Architecture

Parallel Computing: Parallel Architectures Jin, Hai

Transcription:

Real-Time Rendering Architectures Mike Houston, AMD

Part 1: throughput processing Three key concepts behind how modern GPU processing cores run code Knowing these concepts will help you: 1. Understand space of GPU core (and throughput CPU core) designs 2. Optimize shaders/compute kernels 3. Establish intuition: what workloads might benefit from the design of these architectures?

What s in a GPU? A GPU is a heterogeneous chip multi-processor (highly tuned for graphics) Shader Core Shader Core Tex Input Assembly Rasterizer Shader Core Shader Core Tex Output Blend Shader Core Shader Core Tex Video Shader Core Shader Core Tex Work Distributor HW or SW?

A diffuse reflectance shader sampler mysamp; Texture2D<float3> mytex; float3 lightdir; Shader programming model: float4 diffuseshader(float3 norm, float2 uv) { float3 kd; kd = mytex.sample(mysamp, uv); kd *= clamp( dot(lightdir, norm), 0.0, 1.0); Fragments are processed independently, but there is no explicit parallel programming return float4(kd, 1.0); }

Compile shader 1 unshaded fragment input record sampler mysamp; Texture2D<float3> mytex; float3 lightdir; float4 diffuseshader(float3 norm, float2 uv) { float3 kd; kd = mytex.sample(mysamp, uv); kd *= clamp( dot(lightdir, norm), 0.0, 1.0); return float4(kd, 1.0); } <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0) 1 shaded fragment output record

Execute shader ALU (Execute) Execution Context <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Execute shader ALU (Execute) Execution Context <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Execute shader ALU (Execute) Execution Context <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Execute shader ALU (Execute) Execution Context <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Execute shader ALU (Execute) Execution Context <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Execute shader ALU (Execute) Execution Context <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Execute shader ALU (Execute) Execution Context <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

CPU-style cores ALU (Execute) Data cache (a big one) Execution Context Out-of-order control logic Fancy branch predictor Memory pre-fetcher

Slimming down ALU (Execute) Execution Context Idea #1: Remove components that help a single instruction stream run fast

Two cores (two fragments in parallel) fragment 1 fragment 2 <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 ALU (Execute) ALU (Execute) <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0) Execution Context Execution Context mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Four cores (four fragments in parallel) ALU (Execute) Execution Context ALU (Execute) Execution Context ALU (Execute) Execution Context ALU (Execute) Execution Context

Sixteen cores (sixteen fragments in parallel) 16 cores = 16 simultaneous instruction streams

Instruction stream sharing But... many fragments should be able to share an instruction stream! <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0)

Recall: simple processing core ALU (Execute) Execution Context

Add ALUs ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 Idea #2: Amortize cost/complexity of managing an instruction stream across many ALUs Ctx Ctx Ctx Ctx Ctx Ctx Ctx Ctx SIMD processing Shared Ctx Data

Modifying the shader ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 Ctx Ctx Ctx Ctx <diffuseshader>: sample r0, v4, t0, s0 mul r3, v0, cb0[0] madd r3, v1, cb0[1], r3 madd r3, v2, cb0[2], r3 clmp r3, r3, l(0.0), l(1.0) mul o0, r0, r3 mul o1, r1, r3 mul o2, r2, r3 mov o3, l(1.0) Ctx Ctx Ctx Ctx Original compiled shader: Shared Ctx Data Processes one fragment using scalar ops on scalar registers

Modifying the shader ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 Ctx Ctx Ctx Ctx <VEC8_diffuseShader>: VEC8_sample vec_r0, vec_v4, t0, vec_s0 VEC8_mul vec_r3, vec_v0, cb0[0] VEC8_madd vec_r3, vec_v1, cb0[1], vec_r3 VEC8_madd vec_r3, vec_v2, cb0[2], vec_r3 VEC8_clmp vec_r3, vec_r3, l(0.0), l(1.0) VEC8_mul vec_o0, vec_r0, vec_r3 VEC8_mul vec_o1, vec_r1, vec_r3 VEC8_mul vec_o2, vec_r2, vec_r3 VEC8_mov o3, l(1.0) Ctx Ctx Ctx Ctx New compiled shader: Shared Ctx Data Processes eight fragments using vector ops on vector registers

Modifying the shader 1 2 3 4 5 6 7 8 ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 Ctx Ctx Ctx Ctx <VEC8_diffuseShader>: VEC8_sample vec_r0, vec_v4, t0, vec_s0 VEC8_mul vec_r3, vec_v0, cb0[0] VEC8_madd vec_r3, vec_v1, cb0[1], vec_r3 VEC8_madd vec_r3, vec_v2, cb0[2], vec_r3 VEC8_clmp vec_r3, vec_r3, l(0.0), l(1.0) VEC8_mul vec_o0, vec_r0, vec_r3 VEC8_mul vec_o1, vec_r1, vec_r3 VEC8_mul vec_o2, vec_r2, vec_r3 VEC8_mov o3, l(1.0) Ctx Ctx Ctx Ctx Shared Ctx Data

128 fragments in parallel 16 cores = 128 ALUs, 16 simultaneous instruction streams

128 [ vertices/fragments ] in parallel primitives OpenCL work items vertices primitives fragments

But what about branches? Time (clocks) 1 2...... 8 ALU 1 ALU 2...... ALU 8 <unconditional shader code> if (x > 0) { refl = y + Ka; } else { x = 0; } y = pow(x, exp); y *= Ks; refl = Ka; <resume unconditional shader code>

But what about branches? Time (clocks) 1 2...... 8 ALU 1 ALU 2...... ALU 8 <unconditional shader code> T T F T F F F F if (x > 0) { refl = y + Ka; } else { x = 0; } y = pow(x, exp); y *= Ks; refl = Ka; <resume unconditional shader code>

But what about branches? Time (clocks) 1 2...... 8 ALU 1 ALU 2...... ALU 8 <unconditional shader code> T T F T F F F F Not all ALUs do useful work! Worst case: 1/8 peak if (x > 0) { refl = y + Ka; } else { x = 0; } y = pow(x, exp); y *= Ks; refl = Ka; <resume unconditional shader code> performance

But what about branches? Time (clocks) 1 2...... 8 ALU 1 ALU 2...... ALU 8 <unconditional shader code> T T F T F F F F if (x > 0) { refl = y + Ka; } else { x = 0; } y = pow(x, exp); y *= Ks; refl = Ka; <resume unconditional shader code>

Clarification SIMD processing does not imply SIMD instructions Option 1: explicit vector instructions x86 SSE, AVX, Intel Larrabee Option 2: scalar instructions, implicit HW vectorization HW determines instruction stream sharing across ALUs (amount of sharing hidden from software) NVIDIA GeForce ( SIMT warps), ATI Radeon architectures ( wavefronts ) In practice: 16 to 64 fragments share an instruction stream.

Stalls! Stalls occur when a core cannot run the next instruction because of a dependency on a previous operation. Texture access latency = 100 s to 1000 s of cycles We ve removed the fancy caches and logic that helps avoid stalls.

But we have LOTS of independent fragments. Idea #3: Interleave processing of many fragments on a single core to avoid stalls caused by high latency operations.

Hiding shader stalls Time (clocks) Frag 1 8 ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 Ctx Ctx Ctx Ctx Ctx Ctx Ctx Ctx Shared Ctx Data

Hiding shader stalls Time (clocks) Frag 1 8 Frag 9 16 Frag 17 24 Frag 25 32 1 2 3 4 ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 1 2 3 4

Hiding shader stalls Time (clocks) Frag 1 8 Frag 9 16 Frag 17 24 Frag 25 32 1 2 3 4 Stall Runnable

Hiding shader stalls Time (clocks) Frag 1 8 Frag 9 16 Frag 17 24 Frag 25 32 1 2 3 4 Stall Stall Runnable Stall Stall

Throughput! Time (clocks) Frag 1 8 Frag 9 16 Frag 17 24 Frag 25 32 1 2 3 4 Stall Start Stall Start Runnable Stall Start Runnable Stall Done! Runnable Done! Increase run time of one group Runnable to increase throughput of many groups Done!

Storing contexts ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 Pool of context storage 128 KB

Eighteen small contexts (maximal latency hiding) ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8

Twelve medium contexts ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8

Four large contexts (low latency hiding ability) ALU 1 ALU 2 ALU 3 ALU 4 ALU 5 ALU 6 ALU 7 ALU 8 1 2 3 4

Clarification Interleaving between contexts can be managed by hardware or software (or both!) NVIDIA / ATI Radeon GPUs HW schedules / manages all contexts (lots of them) Special on-chip storage holds fragment state Intel Larrabee HW manages four x86 (big) contexts at fine granularity SW scheduling interleaves many groups of fragments on each HW context L1-L2 cache holds fragment state (as determined by SW)

Example chip 16 cores 8 mul-add ALUs per core (128 total) 16 simultaneous instruction streams 64 concurrent (but interleaved) instruction streams 512 concurrent fragments = 256 GFLOPs (@ 1GHz)

Summary: three key ideas 1. Use many slimmed down cores to run in parallel 2. Pack cores full of ALUs (by sharing instruction stream across groups of fragments) Option 1: Explicit SIMD vector instructions Option 2: Implicit sharing managed by hardware 3. Avoid latency stalls by interleaving execution of many groups of fragments When one group stalls, work on another group

Part 2: Putting the three ideas into practice: A closer look at real GPUs NVIDIA GeForce GTX 580 ATI Radeon HD 6970

Disclaimer The following slides describe a reasonable way to think about the architecture of commercial GPUs Many factors play a role in actual chip performance

NVIDIA GeForce GTX 580 (Fermi) NVIDIA-speak: 512 stream processors ( CUDA cores ) SIMT execution Generic speak: 16 cores 2 groups of 16 SIMD functional units per core

NVIDIA GeForce GTX 580 core = SIMD function unit, control shared across 16 units (1 MUL-ADD per clock) Groups of 32 [fragments/vertices/cuda threads] share an instruction stream Execution contexts (128 KB) Up to 48 groups are simultaneously interleaved Shared memory (16+48 KB) Up to 1536 individual contexts can be stored Source: Fermi Compute Architecture Whitepaper CUDA Programming Guide 3.1, Appendix G

NVIDIA GeForce GTX 580 core = SIMD function unit, control shared across 16 units (1 MUL-ADD per clock) The core contains 32 functional units Execution contexts (128 KB) Two groups are selected each clock (decode, fetch, and execute two instruction streams in parallel) Shared memory (16+48 KB) Source: Fermi Compute Architecture Whitepaper CUDA Programming Guide 3.1, Appendix G

NVIDIA GeForce GTX 580 SM = CUDA core (1 MUL-ADD per clock) The SM contains 32 CUDA cores Execution contexts (128 KB) Two warps are selected each clock (decode, fetch, and execute two warps in parallel) Shared memory (16+48 KB) Up to 48 warps are interleaved, totaling 1536 CUDA threads Source: Fermi Compute Architecture Whitepaper CUDA Programming Guide 3.1, Appendix G

NVIDIA GeForce GTX 580 There are 16 of these things on the GTX 580: That s 24,500 fragments! Or 24,500 OpenCL work-items! 51

ATI Radeon HD 6970 (Cayman) AMD-speak: 1536 stream processors Generic speak: 24 cores 16 beefy SIMD functional units per core 4 multiply-adds per functional unit (VLIW processing)

ATI Radeon HD 6970 core Execution contexts (256 KB) Shared memory (32 KB) Groups of 64 [fragments/vertices/etc.] share instruction stream = SIMD function unit, control shared across 16 units (Up to 4 MUL-ADDs per clock) Four clocks to execute an instruction for all fragments in a group

ATI Radeon HD 6970 SIMD-engine Execution contexts (256 KB) Shared memory (32 KB) Groups of 64 [fragments/vertices/opencl work items] are in a wavefront. = stream processor, control shared across 16 units (Up to 4 MUL-ADDs per clock) Four clocks to execute an instruction for an entire wavefront

ATI Radeon HD 6970 There are 24 of these cores on the 6970: that s about 32,000 fragments!

The talk thus far: processing data Part 3: moving data to processors

Recall: CPU-style core OOO exec logic Branch predictor ALU Data cache (a big one) Execution Context

CPU-style memory hierarchy OOO exec logic Branch predictor ALU L1 cache (32 KB) L3 cache (8 MB) 25 GB/sec to memory Execution contexts L2 cache (256 KB) shared across cores CPU cores run efficiently when data is resident in cache (caches reduce latency, provide high bandwidth)

Throughput core (GPU-style) ALU 1 ALU 2 ALU 3 ALU 4 150 GB/sec ALU 5 ALU 6 ALU 7 ALU 8 Memory Execution contexts (128 KB) More ALUs, no large traditional cache hierarchy: Need high-bandwidth connection to memory

Bandwidth is a critical resource A high-end GPU (e.g. Radeon HD 6970) has... Over twenty times (2.7 TFLOPS) the compute performance of quad-core CPU No large cache hierarchy to absorb memory requests GPU memory system is designed for throughput Wide bus (150 GB/sec) Repack/reorder/interleave memory requests to maximize use of memory bus Still, this is only six times the bandwidth available to CPU

Bandwidth thought experiment Task: element-wise multiply two long vectors A and B 1.Load input A[i] 2.Load input B[i] 3.Load input C[i] A B + C = D 4.Compute A[i] B[i] + C[i] 5.Store result into D[i] Four memory operations (16 bytes) for every MUL-ADD Radeon HD 6970 can do 1536 MUL-ADDS per clock Need ~20 TB/sec of bandwidth to keep functional units busy Less than 1% efficiency but 6x faster than CPU!

Bandwidth limited! If processors request data at too high a rate, the memory system cannot keep up. No amount of latency hiding helps this. Overcoming bandwidth limits are a common challenge for GPU-compute application developers.

Reducing bandwidth requirements Request data less often (instead, do more math) arithmetic intensity Fetch data from memory less often (share/reuse data across fragments on-chip communication or storage

Reducing bandwidth requirements Two examples of on-chip storage Texture caches OpenCL local memory (CUDA shared memory) 1 2 3 4 Texture caches: Capture reuse across Texture data fragments, not temporal reuse within a single shader program

Modern GPU memory hierarchy ALU 1 ALU 2 ALU 3 ALU 4 Texture cache (read-only) ALU 5 ALU 6 ALU 7 ALU 8 L2 cache (~1 MB) Memory Execution contexts (128 KB) Shared local storage or L1 cache (64 KB) On-chip storage takes load off memory system. Many developers calling for more cache-like storage (particularly GPU-compute applications)

Don t forget about offload cost PCIe bandwidth/latency 8GB/s each direction in practice Attempt to pipeline/multi-buffer uploads and downloads Dispatch latency O(10) usec to dispatch from CPU to the GPU This means offload cost is O(10M) instructions 66

Heterogeneous cores to the rescue? Tighter integration of CPU and GPU style cores Reduce offload cost Reduce memory copies/transfers Power management Industry shifting rapidly in this direction AMD Fusion APUs Intel SandyBridge Nvidia Tegra 2 Apple A4 and A5 Qualcomm Snapdragon TI OMAP 67

AMD A-Series APU ( Llano ) L1 cache L1 cache L2 cache L2 cache L1 cache L1 cache L2 cache L2 cache 68

Others GPUs not compute capable, yet Intel Sandy Bridge Nvidia Tegra 2 69