550 likes | 687 Views
Yaser Zhian Fanafzar Game Studio IGDI, Workshop 07, January 2 nd , 2013. Modern CPUs and Caches A Starting Point for Programmers. Agenda. Some notes about the subject CPUs and their gimmicks Caches and their importance How CPU and OS handle memory logically. A Word of Caution.
E N D
Yaser Zhian Fanafzar Game Studio IGDI, Workshop 07,January 2nd, 2013 Modern CPUs and CachesA Starting Point for Programmers
Agenda • Some notes about the subject • CPUs and their gimmicks • Caches and their importance • How CPU and OS handle memory logically http://yaserzt.com/blog/
A Word of Caution • These are very complex subjects • Expect very few details and much simplification • These are verycomplicated subjects • Expect much generalization and omission • No time • Even a full course would be hilariously insufficient • Not an expert • Sorry! Can’t help much. • Just a pile of loosely related stuff http://yaserzt.com/blog/
A Real Mess • Pressure for performance • Backwards compatibility • Cost/power/etc. • The ridiculous “numbers game” • Law of diminishing returns • Latency vs. Throughput http://yaserzt.com/blog/
Latency vs. Throughput • You can always solve your bandwidth (throughput) problems with money, but it is rarely so for lag (latency.) • Relative rate of improvements (from David Patterson’s keynote, HPEC 2004) • CPU, 80286 till Pentium 4: 21x vs. 2250x • Ethernet, 10Mb till 10Gb: 16x vs. 1000x • Disk, 3600 till 15000rpm: 8x vs. 143x • DRAM, plain till DDR: 4x vs. 120x http://yaserzt.com/blog/
Not the von Neumann Model • At the simplest level, the von Neumann model stipulates: • Program is data and is stored in memory along with data (departing from Turing’s model) • Program is executed sequentially • Not the way computers function anymore… • Abstraction still used for thinking about programs • But it’s leaky as heck! • “Not Your Fathers’ von Neumann Machine!” http://yaserzt.com/blog/
“Hit the Wall” • Speed of Light: can’t send and receive signals to and from all parts of the die in a cycle anymore • Power: more transistors leads to more power, which leads to much more heat • Memory: the CPU isn’t even close to the bottleneck anymore. “All your base are belong to” memory • Complexity: adding more transistors for more sophisticated operation won’t give much of a speedup (e.g. doubling transistors might give 2%.) http://yaserzt.com/blog/
x86 • Family introduced with 8086 in 1978 • Today, new members are still fully binary backward-compatible with that puny machine (5MHz clock, 20-bit addressing, 16-bit regs.) • It had very few registers • It had segmented memory addressing (joy!) • It had many complex instructions and several addressing modes http://yaserzt.com/blog/
Newer x86 (1/2) • 1982 (80286): Protected mode, MMU • 1985 (80386): 32-bit ISA, Paging • 1989 (80486): Pipelining, Cache, Intgrtd. FPU • 1993 (Pentium): Superscalar, 64-bit bus, MMX • 1995 (P-Pro): μ-ops, OoOExec., Register Renaming, Speculative Exec. • 1997 (K6-2, PIII): 3DNow!/SSE • 2003 (Opteron): 64-bit ISA • 2006 (Core 2): Multi-core http://yaserzt.com/blog/
Newer x86 (2/2) • Registers got expanded from (all 16 bit, non really general purpose) • AX, BX, CX, DX • SI, DI, BP, SP • CS, DS, ES, SS, Flags, IP • To • 16 x 64-bit GPRs (RAX, RBX, RCX, RDX, RBP, RSP, RSI, RDI, R8-R15) plus RIP and Flags and others • 16 x 128-bit XMM regs. (XMM0-...) • Or 16 x 256-bit YMM regs. (YMM0-...) • More than a thousand logically different instructions (the usual, plus string processing, cryptography, CRC, complex numbers, etc.) http://yaserzt.com/blog/
The Central Processing Unit • The Fetch-Decode-Execute-Retire Cycle • Strategies for more performance: • More complex instructions, doing more in hardware (CISCing things up) • Faster CPU clock rates (the free lunch) • Instruction-Level Parallelism (SIMD + gimmicks) • Adding cores (free lunch is over!) • And then, there are gimmicks… http://yaserzt.com/blog/
“Performance Enhancement” • Pipelining • µ-ops • Superscalar Pipelines • Out-of-order Execution • Speculative Execution • Register Renaming • Branch Prediction • Prefetching • Store Buffer • Trace Cache • … http://yaserzt.com/blog/
Pipelining (1/4) Classic sequential execution: • Length of instruction executions vary a lot (5-10 times usual, several orders of magnitude also happen.) Instruction 1 Instruction 2 Instruction 3 Instruction 4 http://yaserzt.com/blog/
Pipelining (2/4) It’s really more like this for the CPU: • Instructions may have many sub-parts, and they engage different parts of the CPU F1 D1 E1 R1 F2 D2 E2 R2 F3 D3 E3 R3 F4 D4 E4 R4 http://yaserzt.com/blog/
Pipelining (3/4) So why not do this: • This is called “pipelining” • It increases throughput (significantly) • Doesn’t decrease latency for single instructions F1 D1 E1 R1 F2 D2 E2 R2 F3 D3 E3 R3 R4 D4 E4 F4 http://yaserzt.com/blog/
Pipelining (4/4) But it has its own share of problems • Hazards, stalls, flushing, etc. • Execution of i2 depends on the result of i1 • After i2, we jump and the i3, i4,… are flushed out F1 D1 E1 R1 add EAX,120 F2 D2 E2 R2 jmp [EAX] F3 D3 E3 R3 mov [4*EBX+42],EDX F4 D4 R4 E4 add ECX,[EAX] http://yaserzt.com/blog/
Micro Operations (µ-ops) (1/2) • Instructions are broken up into simple, orthogonal µ-ops • movEAX,EDX might generate only one µ-op • mov EAX,[EDX] might generate two: • µld tmp0,[EDX] • µmov EAX,tmp0 • add [EAX],EDX probably generates three: • µld tmp0,[EAX] • µadd tmp0,EDX • µst [EAX],tmp0 http://yaserzt.com/blog/
Micro Operations (µ-ops) (2/2) • The CPU then, gets two layers: • The one that breaks up operations into µ-ops • The one that executes µ-ops • The part that executes µ-ops can be simpler (more RISCy) and therefore faster. • More complex instructions can be supported without (much) complicating the CPU • The pipelining (and other gimmicks) can happen at the µ-op level http://yaserzt.com/blog/
Superscalar Execution • CPUs that issue (or retire) more than one instruction per cycle are called Superscalar • Can be thought of as a pipeline with more than one line • Simplest form: integer pipe plus floating-point pipe • These days, CPUs do 4 or more • Obviously requires more of each type of operational unit in the CPU http://yaserzt.com/blog/
Out-of-Order Execution (1/2) • To prevent your pipeline from stalling as much as possible, issue the next instructions even if you can’t start the current one. • But of course, only if there are no hazards (dependencies) and there are operational units available. • add RAX,RAXadd RAX,RBXadd RCX,RDX This can be and is started before the previous instruction. http://yaserzt.com/blog/
Out-of-Order Execution (2/2) • This obviously also applies at the µ-op level: • mov RAX,[mem0]mul RAX,42add RAX,[mem1] • push RAXcall Func Fetching mem1 is started long before the result of the multiply becomes available. Pushing RAX is sub RSP,8 and then mov [RSP],RAX. Since call instruction needs RSP too, it will only wait for the subtraction and not the store to finish to start. http://yaserzt.com/blog/
Register Renaming (1/3) • Consider this:mov RAX,[mem0]mul RAX,42mov [mem1],RAXmov RAX,[mem2]add RAX,7mov[mem3],RAX • Logically, the two parts are totally separate. • However, the use of RAX will stall the pipeline. http://yaserzt.com/blog/
Register Renaming (2/3) • Modern CPUs have a lot of temporary, unnamed registers at their disposal. • They will detect the logical independence, and will use one of those in the second block instead on RAX. • And they will track which reg. is which, where. • In effect, they are renaming another register to RAX. • There might not even be a real RAX! http://yaserzt.com/blog/
Register Renaming (3/3) • This is, for once, simpler than it might seem! • Every time a register is assigned to, a new temporary register is used in its stead. • Consider this:mov RAX,[cached]movRBX,[uncached]add RBX,RAXmulRAX,42mov [mem0],RAXmov[mem1],RBX Rename happens Renaming on mul means that it won’t clobber RAX (which we need for the add, that is waiting on the load of [uncached]) and we can do the multiply and reach the first store much sooner. http://yaserzt.com/blog/
Branch Prediction (1/9) • The CPU always depends on knowing where the next instruction is, so it can go ahead and work on it. • That’s why branches in code are anathema to modern, deep pipelines and all the gimmicks they pull. • Only if the CPU could somehow guess where the target of each branch is going to be… • That’s where branch prediction comes in. http://yaserzt.com/blog/
Branch Prediction (2/9) • So the CPU guesses the target of a jump (if it doesn’t know for sure,) and continues to speculatively execute instructions from there. • For a conditional jump, the CPU must also predict whether the branch is taken or not. • If the CPU is right, the pipeline flows smoothly. If not, the pipeline must be flushed and much time and resource is wasted on a misprediction. http://yaserzt.com/blog/
Branch Prediction (3/9) • In this code:cmp RAX,0jne [RBX]both the target and whether the jump happens or not must be predicted. • The above can effectively jump anywhere! • But usually branches are closer to this:cmp RAX,0jnesomewhere_specific • Which can only have two possible targets. http://yaserzt.com/blog/
Branch Prediction (4/9) • In a simple form, when a branch is executed, its target is stored in a table called the BTB (or Branch Target Buffer.) When that branch is encountered again, the target address is predicted to be the value read from the BTB. • As you might guess, this doesn’t work for many situations (e.g. alternating branch.) • Also, the size of the BTB is limited, so CPU will forget about the last target of some jumps. http://yaserzt.com/blog/
Branch Prediction (5/9) • A simple expansion on the previous idea is to use a saturating counter along with each entry of the BTB. • For example, with a 2-bit counter, • Branch is predicted not to be taken if the counter is 0 or 1. • The branch is predicted to be taken if the counter is 2 or 3. • Each time it is taken, counter is incremented, and vice versa. T T T Strongly Not Taken Weakly Not Taken Weakly Taken Strongly Taken NT T NT NT NT http://yaserzt.com/blog/
Branch Prediction (6/9) • But this behaves very badly in common situations. • For an alternating branch, • If the counter starts in 00 or 11, it will mispredict 50%. • If the counter starts in 01, and the first time the branch is taken, it will mispredict100%! • As an improvement, we can store the history of the last N occurrences of the branch in the BTB, and use 2N counters for each of the possible history patterns. http://yaserzt.com/blog/
Branch Prediction (7/9) • For N=4 and 2-bit counters, we’ll have: • This is an extremely cool method of doing branch prediction! Prediction (0 or 1) Branch History . . . 0010 http://yaserzt.com/blog/
Branch Prediction (8/9) • Some predictions are simpler: • For each ret instruction, the target is somewhere on the stack (pushed before.) Modern CPUs keep track of return addresses in an internal return stack buffer. Each time acall is executed, an entry is added and is used for the return address. • On a cold encounter (a.k.a. static prediction) a branch is sometimes predicted to • fall through if it goes forward. • be taken if it goes backward. http://yaserzt.com/blog/
Branch Prediction (9/9) • Best general advice is to arrange your code so that the most common path for branches is “not taken”. This improves the effectiveness of code prefetching and the trace cache. • Branch prediction, register renaming and speculative execution work extremely well together. http://yaserzt.com/blog/
An Example (1/15) mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] http://yaserzt.com/blog/
An Example (2/15) Clock 0 – Instruction 0 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Load RAX from memory Assume cache miss – 300 cycles to load Instruction starts and dispatch continues... http://yaserzt.com/blog/
An Example (3/15) Clock 0 – Instruction 1 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] This instruction writes RBX, which conflicts with the read in instruction 0. Rename this instance of RBX and continue… http://yaserzt.com/blog/
An Example (4/15) Clock 0 – Instruction 2 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Value of RAX not available yet; cannot calculate value of Flags reg. Queue up behind instruction 0… http://yaserzt.com/blog/
An Example (5/15) Clock 0 – Instruction 3 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Flags reg. still not available. Predict that this branch is not taken. Assuming 4-wide dispatch, instruction issue limit is reached. http://yaserzt.com/blog/
An Example (6/15) Clock 1 – Instruction 4 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Store is speculative. Result kept in Store Buffer. Also, RBX might not be available yet (from instruction 1.) Load/Store Unit is tied up from now on; can’t issue any more memory ops in this cycle. http://yaserzt.com/blog/
An Example (7/15) Clock 2 – Instruction 5 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Had to wait for L/S Unit. Assume this is another (and unrelated) cache miss. We have 2 overlapping cache misses now. L/S Unit is busy again. http://yaserzt.com/blog/
An Example (8/15) Clock 3 – Instruction 6 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] RAX is not ready yet (300-cycle latency, remember?!) This load cannot even start until instruction 0 is done. http://yaserzt.com/blog/
An Example (9/15) Clock 301 – Instruction 2 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] At clock 300 (or 301,) RAX is finally ready. Do the comparison and update Flags register. http://yaserzt.com/blog/
An Example (10/15) Clock 301 – Instruction 6 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Issue this load too. Assume a cache hit (finally!) Result will be available in clock 304. http://yaserzt.com/blog/
An Example (11/15) Clock 302 – Instruction 3 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Now the Flags reg. is ready. Check the prediction. Assume prediction was correct. http://yaserzt.com/blog/
An Example (12/15) Clock 302 – Instruction 4 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] This speculative store can actually be committed to memory (or cache, actually.) http://yaserzt.com/blog/
An Example (13/15) Clock 302 – Instruction 5 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] At clock 302, the result of this load arrives. http://yaserzt.com/blog/
An Example (14/15) Clock 305 – Instruction 6 mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] Result arrived at clock 304; instruction retired at 305. http://yaserzt.com/blog/
An Example (15/15) mov RAX,[RBX+16] add RBX,16 cmp RAX,0 je IsNull mov [RBX-16],RCX mov RCX,[RDX+0]mov RAX,[RAX+8] • To summarize, • In 4 clocks, started 7 ops and 2 cache misses • Retired 7 ops in 306 cycles. • Cache misses totally dominate performance. • The only real benefit came from being able to have 2 overlapping cache misses! http://yaserzt.com/blog/
A New Performance Goal To get to the next cache miss as early as possible. http://yaserzt.com/blog/
What Be This Cache?! • Main memory is slow; S.L.O.W. • Very slow • Painfully slow • And it specially has very bad (high) latency • But all is not lost! Many (most) references to memory have high temporal and address locality. • So we use a small amount of very fast memory to keep recently-accessed or likely-to-be-accessed chunks of main memory close to CPU. http://yaserzt.com/blog/