Carnegie Mellon Machine-Level Programming III: Switch Statements and IA32 Procedures 15-213 / 18-213: Introduction to Computer Systems 7th Lecture, Sep.

Slides:



Advertisements
Similar presentations
Machine-Level Programming III: Procedures Feb. 8, 2000 Topics IA32 stack Stack-based languages Stack frames Register saving conventions Creating pointers.
Advertisements

Fabián E. Bustamante, Spring 2007 Machine-Level Programming II: Control Flow Today Condition codes Control flow structures Next time Procedures.
University of Washington Procedures and Stacks II The Hardware/Software Interface CSE351 Winter 2013.
Machine Programming – Procedures and IA32 Stack CENG334: Introduction to Operating Systems Instructor: Erol Sahin Acknowledgement: Most of the slides are.
Carnegie Mellon Instructors: Randy Bryant and Dave O’Hallaron Machine-Level Programming III: Switch Statements and IA32 Procedures : Introduction.
University of Washington Last Time For loops  for loop → while loop → do-while loop → goto version  for loop → while loop → goto “jump to middle” version.
Machine-Level Programming III: Procedures Apr. 17, 2006 Topics IA32 stack discipline Register saving conventions Creating pointers to local variables CS213.
1 Function Calls Professor Jennifer Rexford COS 217 Reading: Chapter 4 of “Programming From the Ground Up” (available online from the course Web site)
Machine-Level Programming III: Procedures Sept. 17, 2007 IA32 stack discipline Register saving conventions Creating pointers to local variablesx86-64 Argument.
– 1 – , F’02 ICS05 Instructor: Peter A. Dinda TA: Bin Lin Recitation 4.
Machine-Level Programming III: Procedures Jan 30, 2003
Assembly תרגול 8 פונקציות והתקפת buffer.. Procedures (Functions) A procedure call involves passing both data and control from one part of the code to.
Machine-Level Programming III: Procedures Sept. 15, 2006 IA32 stack discipline Register saving conventions Creating pointers to local variablesx86-64 Argument.
Stack Activation Records Topics IA32 stack discipline Register saving conventions Creating pointers to local variables February 6, 2003 CSCE 212H Computer.
Introduction to x86 Assembly or “How does someone hack my laptop?”
Machine-Level Programming 3 Control Flow Topics Control Flow Switch Statements Jump Tables.
Carnegie Mellon Introduction to Computer Systems /18-243, spring 2009 Recitation, Jan. 14 th.
1 Carnegie Mellon Stacks : Introduction to Computer Systems Recitation 5: September 24, 2012 Joon-Sup Han Section F.
Lee CSCE 312 TAMU 1 Based on slides provided by Randy Bryant and Dave O’Hallaron Machine-Level Programming III: Switch Statements and IA32 Procedures Instructor:
Ithaca College Machine-Level Programming IV: IA32 Procedures Comp 21000: Introduction to Computer Systems & Assembly Lang Spring 2013 * Modified slides.
University of Washington x86 Programming III The Hardware/Software Interface CSE351 Winter 2013.
Machine-Level Programming III: Switch Statements and IA32 Procedures Seoul National University.
Machine-Level Programming V: Switch Statements Comp 21000: Introduction to Computer Systems & Assembly Lang Systems book chapter 3* * Modified slides from.
University of Washington Today More on procedures, stack etc. Lab 2 due today!  We hope it was fun! What is a stack?  And how about a stack frame? 1.
Fabián E. Bustamante, Spring 2007 Machine-Level Programming III - Procedures Today IA32 stack discipline Register saving conventions Creating pointers.
Machine-Level Programming III: Procedures Topics IA32 stack discipline Register-saving conventions Creating pointers to local variables CS 105 “Tour of.
Machine-Level Programming 3 Control Flow Topics Control Flow Switch Statements Jump Tables.
Carnegie Mellon Machine-Level Programming III: Switch Statements, 1-D Array and IA32 Procedures Lecture, Mar. 7, 2013 These slides are from
Machine-level Programming III: Procedures Topics –IA32 stack discipline –Register saving conventions –Creating pointers to local variables.
F’08 Stack Discipline Aug. 27, 2008 Nathaniel Wesley Filardo Slides stolen from CMU , whence they were stolen from CMU CS 438.
University of Amsterdam Computer Systems – the instruction set architecture Arnoud Visser 1 Computer Systems The instruction set architecture.
Bryant and O’Hallaron, Computer Systems: A Programmer’s Perspective, Third Edition Carnegie Mellon Instructor: San Skulrattanakulchai Machine-Level Programming.
MACHINE-LEVEL PROGRAMMING III: SWITCH STATEMENTS AND IA32 PROCEDURES.
1 Assembly Language: Function Calls Jennifer Rexford.
IA32 Stack –Region of memory managed with stack discipline –Grows toward lower addresses –Register %esp indicates lowest stack address address of top element.
Reading Condition Codes (Cont.)
C function call conventions and the stack
CSCE 212 Computer Architecture
Machine-Level Programming III: Procedures
Chapter 3 Machine-Level Representation of Programs
Switch & Procedures & The Stack I CSE 351 Winter 2017
Machine-Level Programming 1 Introduction
Machine-Level Programming II: Control Flow
Machine-Level Programming III: Switch Statements and IA32 Procedures
Y86 Processor State Program Registers
Instructors: Majd Sakr and Khaled Harras
Machine-Level Programming 4 Procedures
Machine-Level Programming III: Procedures /18-213/14-513/15-513: Introduction to Computer Systems 7th Lecture, September 18, 2018.
Carnegie Mellon Machine-Level Programming III: Procedures : Introduction to Computer Systems October 22, 2015 Instructor: Rabi Mahapatra Authors:
Condition Codes Single Bit Registers
Roadmap C: Java: Assembly language: OS: Machine code: Computer system:
Machine-Level Programming 2 Control Flow
Assembly Language Programming II: C Compiler Calling Sequences
Machine-Level Programming 2 Control Flow
Machine-Level Programming III: Procedures Sept 18, 2001
Today Control flow if/while/do while/for/switch
Machine-Level Programming: Introduction
Roadmap C: Java: Assembly language: OS: Machine code: Computer system:
Ithaca College Machine-Level Programming VII: Procedures Comp 21000: Introduction to Computer Systems & Assembly Lang Spring 2017.
Chapter 3 Machine-Level Representation of Programs
Machine-Level Representation of Programs (x86-64)
X86 Assembly - Control.
Machine-Level Programming II: Control Flow Sept. 12, 2007
“Way easier than when we were students”
CS201- Lecture 8 IA32 Flow Control
Ithaca College Machine-Level Programming VII: Procedures Comp 21000: Introduction to Computer Systems & Assembly Lang Spring 2017.
Presentation transcript:

Carnegie Mellon Machine-Level Programming III: Switch Statements and IA32 Procedures 15-213 / 18-213: Introduction to Computer Systems 7th Lecture, Sep 16, 2014 Instructors: Greg Ganger, Greg Kesden, and Dave O’Hallaron

Today Switch statements IA 32 Procedures Stack Structure Carnegie Mellon Today Switch statements IA 32 Procedures Stack Structure Calling Conventions Illustrations of Recursion & Pointers

Switch Statement Example Carnegie Mellon long switch_eg (long x, long y, long z) { long w = 1; switch(x) { case 1: w = y*z; break; case 2: w = y/z; /* Fall Through */ case 3: w += z; case 5: case 6: w -= z; default: w = 2; } return w; Switch Statement Example Multiple case labels Here: 5 & 6 Fall through cases Here: 2 Missing cases Here: 4

Jump Table Structure Jump Targets Switch Form Jump Table Targ0: jtab: Carnegie Mellon Jump Table Structure Jump Targets Switch Form Jump Table Targ0: Code Block switch(x) { case val_0: Block 0 case val_1: Block 1 • • • case val_n-1: Block n–1 } jtab: Targ0 Targ1 Targ2 Targ1: Code Block 1 • Targ2: Code Block 2 Targn-1 • Approximate Translation target = JTab[x]; goto *target; Targn-1: Code Block n–1

Switch Statement Example (IA32) Carnegie Mellon Switch Statement Example (IA32) long switch_eg(long x, long y, long z) { long w = 1; switch(x) { . . . } return w; What range of values takes default? Setup: switch_eg: pushl %ebp # Setup movl %esp, %ebp # Setup movl 8(%ebp), %eax # %eax = x cmpl $6, %eax # Compare x:6 ja .L8 # If unsigned > goto default jmp *.L4(,%eax,4) # Goto *JTab[x] Note that w not initialized here

Switch Statement Example (IA32) Carnegie Mellon Switch Statement Example (IA32) long switch_eg(long x, long y, long z) { long w = 1; switch(x) { . . . } return w; Jump table .section .rodata .align 4 .L4: .long .L8 # x = 0 .long .L3 # x = 1 .long .L5 # x = 2 .long .L9 # x = 3 .long .L8 # x = 4 .long .L7 # x = 5 .long .L7 # x = 6 Setup: switch_eg: pushl %ebp # Setup movl %esp, %ebp # Setup movl 8(%ebp), %eax # eax = x cmpl $6, %eax # Compare x:6 ja .L8 # If unsigned > goto default jmp *.L4(,%eax,4) # Goto *JTab[x] Indirect jump

Assembly Setup Explanation Carnegie Mellon Assembly Setup Explanation Table Structure Each target requires 4 bytes Base address at .L4 Jumping Direct: jmp .L2 Jump target is denoted by label .L2 Indirect: jmp *.L4(,%eax,4) Start of jump table: .L4 Must scale by factor of 4 (labels have 32-bits = 4 Bytes on IA32) Fetch target from effective Address .L4 + eax*4 Only for 0 ≤ x ≤ 6 Jump table .section .rodata .align 4 .L4: .long .L8 # x = 0 .long .L3 # x = 1 .long .L5 # x = 2 .long .L9 # x = 3 .long .L8 # x = 4 .long .L7 # x = 5 .long .L7 # x = 6

Jump Table Jump table switch(x) { case 1: // .L3 w = y*z; break; Carnegie Mellon Jump Table Jump table switch(x) { case 1: // .L3 w = y*z; break; case 2: // .L5 w = y/z; /* Fall Through */ case 3: // .L9 w += z; case 5: case 6: // .L7 w -= z; default: // .L8 w = 2; } .section .rodata .align 4 .L4: .long .L8 # x = 0 .long .L3 # x = 1 .long .L5 # x = 2 .long .L9 # x = 3 .long .L8 # x = 4 .long .L7 # x = 5 .long .L7 # x = 6

Code Blocks (x == 1) switch(x) { case 1: // .L3 w = y*z; break; . . . Carnegie Mellon Code Blocks (x == 1) switch(x) { case 1: // .L3 w = y*z; break; . . . } .L3: # x == 1 movl 12(%ebp), %eax # y imull 16(%ebp), %eax # w = y*z jmp .L2 # Goto done

Handling Fall-Through Carnegie Mellon Handling Fall-Through long w = 1; . . . switch(x) { case 2: w = y/z; /* Fall Through */ case 3: w += z; break; } case 2: w = y/z; goto merge; case 3: w = 1; merge: w += z;

Code Blocks (x == 2, x == 3) .L5: # x == 2 movl 12(%ebp), %eax # y Carnegie Mellon Code Blocks (x == 2, x == 3) .L5: # x == 2 movl 12(%ebp), %eax # y cltd idivl 16(%ebp) # y/z jmp .L6 # goto merge .L9: # x == 3 movl $1, %eax # w = 1 .L6: # merge: addl 16(%ebp), %eax # += z jmp .L2 # goto done long w = 1; . . . switch(x) { case 2: w = y/z; /* Fall Through */ case 3: w += z; break; }

Code Blocks (x == 5, x == 6, default) Carnegie Mellon Code Blocks (x == 5, x == 6, default) switch(x) { . . . case 5: // .L7 case 6: // .L7 w -= z; break; default: // .L8 w = 2; } .L7: # x == 5, 6 movl $1, %eax # w = 1 subl 16(%ebp), %eax # w -= z jmp .L2 # goto done .L8: # default movl $2, %eax # w = 2 .L2: # done:

Switch Code (Finish) Noteworthy Features Carnegie Mellon Switch Code (Finish) .L2: # done: popl %ebp ret return w; Noteworthy Features Jump table avoids sequencing through cases Constant time, rather than linear Use jump table to handle holes and duplicate tags Use program sequencing and/or jumps to handle fall-through Don’t initialize w = 1 unless really need it

x86-64 Switch Implementation Carnegie Mellon x86-64 Switch Implementation Same general idea, adapted to 64-bit code Table entries 64 bits (pointers) Cases use revised code Jump Table switch(x) { case 1: // .L3 w = y*z; break; . . . } .section .rodata .align 8 .L7: .quad .L8 # x = 0 .quad .L3 # x = 1 .quad .L5 # x = 2 .quad .L9 # x = 3 .quad .L8 # x = 4 .quad .L7 # X = 5 .quad .L7 # x = 6 .L3: movq %rdx, %rax imulq %rsi, %rax ret

Summarizing C Control Assembler Control Standard Techniques Carnegie Mellon Summarizing C Control if-then-else do-while while, for switch Assembler Control Conditional jump Conditional move Indirect jump (via jump tables) Compiler generates code sequence to implement more complex control Standard Techniques Loops converted to do-while form Large switch statements use jump tables Sparse switch statements may use decision trees (if-elseif-elseif-else)

Today Switch statements IA 32 Procedures Stack Structure Carnegie Mellon Today Switch statements IA 32 Procedures Stack Structure Calling Conventions Illustrations of Recursion & Pointers

IA32 Stack Stack “Bottom” Carnegie Mellon IA32 Stack Stack Pointer: %esp Stack Grows Down Increasing Addresses Stack “Top” Stack “Bottom” Region of memory managed with stack discipline Grows toward lower addresses Register %esp contains lowest stack address address of “top” element

IA32 Stack: Push Stack “Bottom” pushl Src Stack Pointer: %esp Carnegie Mellon IA32 Stack: Push Stack “Bottom” pushl Src Fetch operand at Src Decrement %esp by 4 Write operand at address given by %esp Increasing Addresses Stack Grows Down Stack Pointer: %esp Stack “Top” -4

IA32 Stack: Pop Stack “Bottom” popl Dest Stack Pointer: %esp Carnegie Mellon IA32 Stack: Pop Stack “Bottom” popl Dest Read value at address given by %esp Increment %esp by 4 Store value at Dest (must be register) Increasing Addresses Stack Grows Down +4 Stack Pointer: %esp Stack “Top”

Procedure Control Flow Carnegie Mellon Procedure Control Flow Use stack to support procedure call and return Procedure call: call label Push return address on stack Jump to label Return address: Address of the next instruction right after call Example from disassembly 804854e: e8 3d 06 00 00 call 8048b90 <main> 8048553: 50 pushl %eax Return address = 0x8048553 Procedure return: ret Pop address from stack Jump to address

Procedure Call Example Carnegie Mellon Procedure Call Example 804854e: e8 3d 06 00 00 call 8048b90 <main> 8048553: 50 pushl %eax call 8048b90 0x110 0x110 0x10c 0x10c 0x108 123 0x108 123 0x104 0x8048553 %esp 0x108 %esp 0x104 %eip 0x804854e %eip 0x8048b90 %eip: program counter

Procedure Return Example Carnegie Mellon Procedure Return Example 8048591: c3 ret ret 0x110 0x110 0x10c 0x10c 0x108 123 0x108 123 0x104 0x8048553 0x8048553 %esp 0x104 %esp 0x108 %eip 0x8048591 %eip 0x8048553 %eip: program counter

Stack-Based Languages Carnegie Mellon Stack-Based Languages Languages that support recursion e.g., C, Pascal, Java Code must be “Reentrant” Multiple simultaneous instantiations of single procedure Need some place to store state of each instantiation Arguments Local variables Return pointer Stack discipline State for given procedure needed for limited time From when called to when return Callee returns before caller does Stack allocated in Frames state for single procedure instantiation

Call Chain Example Example Call Chain yoo(…) { • who(); } yoo who(…) { Carnegie Mellon Call Chain Example Example Call Chain yoo(…) { • who(); } yoo who(…) { • • • amI(); } who amI(…) { • amI(); } amI amI amI amI Procedure amI() is recursive

Stack Frames Contents Management Stack “Top” Previous Frame Carnegie Mellon Stack Frames Previous Frame Frame for proc Contents Local variables Return information Temporary space Management Space allocated when enter procedure “Set-up” code Deallocated when return “Finish” code Frame Pointer: %ebp Stack Pointer: %esp Stack “Top”

Example Stack yoo yoo yoo(…) %ebp { • yoo who(); who %esp } amI amI Carnegie Mellon Example Stack yoo yoo yoo(…) { • who(); } %ebp %esp yoo who amI amI amI amI

Example Stack yoo who yoo(…) { • who(); } yoo who(…) { • • • amI(); } Carnegie Mellon Example Stack yoo who yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo who %ebp %esp amI amI amI amI

Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • Carnegie Mellon Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo amI(…) { • amI(); } who amI amI %ebp %esp amI amI

Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • Carnegie Mellon Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo amI(…) { • amI(); } who amI(…) { • amI(); } amI amI amI %ebp %esp amI

Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • Carnegie Mellon Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo amI(…) { • amI(); } who amI(…) { • amI(); } amI amI amI(…) { • amI(); } amI amI %ebp %esp

Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • Carnegie Mellon Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo amI(…) { • amI(); } who amI(…) { • amI(); } amI amI amI %ebp %esp amI

Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • Carnegie Mellon Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo amI(…) { • amI(); } who amI amI %ebp %esp amI amI

Example Stack yoo who yoo(…) { • who(); } yoo who(…) { • • • amI(); } Carnegie Mellon Example Stack yoo who yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo who %ebp %esp amI amI amI amI

Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • Carnegie Mellon Example Stack yoo who amI yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo amI(…) { • amI(); } who amI amI %ebp %esp amI amI

Example Stack yoo who yoo(…) { • who(); } yoo who(…) { • • • amI(); } Carnegie Mellon Example Stack yoo who yoo(…) { • who(); } yoo who(…) { • • • amI(); } yoo who %ebp %esp amI amI amI amI

Example Stack yoo yoo %ebp yoo(…) yoo { • who %esp who(); } amI amI Carnegie Mellon Example Stack yoo yoo %ebp %esp yoo(…) { • who(); } yoo who amI amI amI amI

IA32/Linux Stack Frame Current Stack Frame (“Top” to Bottom) Carnegie Mellon IA32/Linux Stack Frame Current Stack Frame (“Top” to Bottom) “Argument build:” Parameters for function about to call Local variables If can’t keep in registers Saved register context Old frame pointer Caller Stack Frame Return address Pushed by call instruction Arguments for this call Caller Frame Arguments Frame pointer %ebp Return Addr Old %ebp Saved Registers + Local Variables Argument Build Stack pointer %esp

Calling swap from call_swap Carnegie Mellon Revisiting swap Calling swap from call_swap int course1 = 15213; int course2 = 18213; void call_swap() { swap(&course1, &course2); } call_swap: • • • subl $24, %esp movl $course2, 4(%esp) movl $course1, (%esp) call swap Resulting Stack . . . void swap(int *xp, int *yp) { int t0 = *xp; int t1 = *yp; *xp = t1; *yp = t0; } &course2 &course1 Rtn adr %esp

Revisiting swap swap: pushl %ebp movl %esp, %ebp pushl %ebx Set Carnegie Mellon Revisiting swap swap: pushl %ebp movl %esp, %ebp pushl %ebx movl 8(%ebp), %edx movl 12(%ebp), %eax movl (%edx), %ecx movl (%eax), %ebx movl %ebx, (%edx) movl %ecx, (%eax) popl %ebx popl %ebp ret Set Up void swap(int *xp, int *yp) { int t0 = *xp; int t1 = *yp; *xp = t1; *yp = t0; } Body Finish

swap Setup #1 Entering Stack Resulting Stack • %ebp • %ebp &course2 yp Carnegie Mellon swap Setup #1 Entering Stack Resulting Stack • %ebp • %ebp &course2 yp &course1 xp Rtn adr %esp Rtn adr Old %ebp %esp swap: pushl %ebp movl %esp,%ebp pushl %ebx

swap Setup #2 Entering Stack Resulting Stack • %ebp • &course2 yp Carnegie Mellon swap Setup #2 Entering Stack Resulting Stack • %ebp • &course2 yp &course1 xp Rtn adr %esp Rtn adr %ebp Old %ebp %esp swap: pushl %ebp movl %esp,%ebp pushl %ebx

swap Setup #3 Entering Stack Resulting Stack • %ebp • &course2 yp Carnegie Mellon swap Setup #3 Entering Stack Resulting Stack • %ebp • &course2 yp &course1 xp Rtn adr %esp Rtn adr Old %ebp %ebp Old %ebx %esp swap: pushl %ebp movl %esp,%ebp pushl %ebx

Offset relative to %ebp Carnegie Mellon swap Body Entering Stack Resulting Stack • %ebp • Offset relative to %ebp &course2 12 yp &course1 8 xp Rtn adr %esp 4 Rtn adr Old %ebp %ebp Old %ebx %esp movl 8(%ebp),%edx # get xp movl 12(%ebp),%eax # get yp . . .

swap Finish Observation Stack Before Finish Resulting Stack Carnegie Mellon swap Finish Stack Before Finish Resulting Stack yp xp Rtn adr Old %ebp %ebp • %esp Old %ebx • %ebp popl %ebx popl %ebp yp xp Rtn adr %esp Observation Saved and restored register %ebx Not so for %eax, %ecx, %edx

Disassembled swap Calling Code 08048490 <swap>: Carnegie Mellon Disassembled swap 08048490 <swap>: 8048490: 55 push %ebp 8048491: 89 e5 mov %esp,%ebp 8048493: 53 push %ebx 8048494: 8b 55 08 mov 0x8(%ebp),%edx 8048497: 8b 45 0c mov 0xc(%ebp),%eax 804849a: 8b 0a mov (%edx),%ecx 804849c: 8b 18 mov (%eax),%ebx 804849e: 89 1a mov %ebx,(%edx) 80484a0: 89 08 mov %ecx,(%eax) 80484a2: 5b pop %ebx 80484a3: 5d pop %ebp 80484a4: c3 ret Leave: Movl %ebp, %esp # set stack pointer to beginning of frame Popl %ebp # restore beginning of previous frame Calling Code 8048426: c7 44 24 04 18 98 04 movl $0x8049818,0x4(%esp) 804842d: 08 804842e: c7 04 24 1c 98 04 08 movl $0x804981c,(%esp) 8048435: e8 56 00 00 00 call 8048490 <swap> 804843a: c9 leave 804843b: c3 ret

Today Switch statements IA 32 Procedures Stack Structure Carnegie Mellon Today Switch statements IA 32 Procedures Stack Structure Calling Conventions Illustrations of Recursion & Pointers

Register Saving Conventions Carnegie Mellon Register Saving Conventions When procedure yoo calls who: yoo is the caller who is the callee Can register be used for temporary storage? Contents of register %edx overwritten by who This could be trouble ➙ something should be done! Need some coordination yoo: • • • movl $15213, %edx call who addl %edx, %eax ret who: • • • movl 8(%ebp), %edx addl $18213, %edx ret

Register Saving Conventions Carnegie Mellon Register Saving Conventions When procedure yoo calls who: yoo is the caller who is the callee Can register be used for temporary storage? Conventions “Caller Save” Caller saves temporary values in its frame before the call “Callee Save” Callee saves temporary values in its frame before using Callee restores them before returning to caller

IA32/Linux+Windows Register Usage Carnegie Mellon IA32/Linux+Windows Register Usage %eax, %edx, %ecx Caller saves prior to call if values are used after call returns %eax also used to return integer value %ebx, %esi, %edi Callee saves if wants to use them %esp, %ebp special form of callee save Restored to original values upon exit from procedure %eax Caller-Save Temporaries %edx %ecx %ebx Callee-Save Temporaries %esi %edi %esp Special %ebp

Today Switch statements IA 32 Procedures Stack Structure Carnegie Mellon Today Switch statements IA 32 Procedures Stack Structure Calling Conventions Illustrations of Recursion & Pointers

Recursive Function Registers %eax, %edx used without first saving Carnegie Mellon Recursive Function pcount_r: pushl %ebp movl %esp, %ebp pushl %ebx subl $20, %esp movl 8(%ebp), %ebx movl $0, %eax testl %ebx, %ebx je .L3 movl %ebx, %eax shrl %eax movl %eax, (%esp) call pcount_r movl %ebx, %edx andl $1, %edx addl %edx, %eax .L3: addl $20, %esp popl %ebx popl %ebp ret /* Recursive popcount */ int pcount_r(unsigned x) { if (x == 0) return 0; else return (x & 1) + pcount_r(x >> 1); } Registers %eax, %edx used without first saving %ebx used, but saved at beginning & restored at end

Recursive Call #1 Actions Save old value of %ebx on stack Carnegie Mellon Recursive Call #1 pcount_r: pushl %ebp movl %esp, %ebp pushl %ebx subl $20, %esp movl 8(%ebp), %ebx • • • /* Recursive popcount */ int pcount_r(unsigned x) { if (x == 0) return 0; else return (x & 1) + pcount_r(x >> 1); } • x Actions Save old value of %ebx on stack Allocate space for argument to recursive call Store x in %ebx Rtn adr Old %ebp %ebp Old %ebx Unused %ebx x %esp

Recursive Call #2 Actions If x == 0, return with %eax set to 0 • • • Carnegie Mellon Recursive Call #2 • • • movl $0, %eax testl %ebx, %ebx je .L3 .L3: ret /* Recursive popcount */ int pcount_r(unsigned x) { if (x == 0) return 0; else return (x & 1) + pcount_r(x >> 1); } Actions If x == 0, return with %eax set to 0 %ebx x

Recursive Call #3 Actions Effect Store x >> 1 on stack Carnegie Mellon Recursive Call #3 • • • movl %ebx, %eax shrl %eax movl %eax, (%esp) call pcount_r /* Recursive popcount */ int pcount_r(unsigned x) { if (x == 0) return 0; else return (x & 1) + pcount_r(x >> 1); } • Actions Store x >> 1 on stack Make recursive call Effect %eax set to function result %ebx still has value of x Rtn adr Old %ebp %ebp Old %ebx Unused x >> 1 %esp %ebx x

Recursive Call #4 Assume Actions Effect Carnegie Mellon Recursive Call #4 /* Recursive popcount */ int pcount_r(unsigned x) { if (x == 0) return 0; else return (x & 1) + pcount_r(x >> 1); } • • • movl %ebx, %edx andl $1, %edx addl %edx, %eax Assume %eax holds value from recursive call %ebx holds x Actions Compute (x & 1) + returned value Effect %eax set to function result %ebx x

Recursive Call #5 Actions Restore values of %ebx and %ebp Restore %esp Carnegie Mellon Recursive Call #5 /* Recursive popcount */ int pcount_r(unsigned x) { if (x == 0) return 0; else return (x & 1) + pcount_r(x >> 1); } • • • L3: addl $20, %esp popl %ebx popl %ebp ret • Actions Restore values of %ebx and %ebp Restore %esp • %ebp Rtn adr Old %ebp %ebp %esp Old %ebx Unused %ebx Old %ebx x >> 1 %esp

Observations About Recursion Carnegie Mellon Observations About Recursion Handled Without Special Consideration Stack frames mean that each function call has private storage Saved registers & local variables Saved return pointer Register saving conventions prevent one function call from corrupting another’s data Unless the C code explicitly does so (e.g., buffer overflow in Lecture 9) Stack discipline follows call / return pattern If P calls Q, then Q returns before P Last-In, First-Out Also works for mutual recursion P calls Q; Q calls P

Pointer Code Generating Pointer Referencing Pointer Carnegie Mellon Pointer Code Generating Pointer /* Compute x + 3 */ int add3(int x) { int localx = x; incrk(&localx, 3); return localx; } Referencing Pointer /* Increment value by k */ void incrk(int *ip, int k) { *ip += k; } add3 creates pointer and passes it to incrk

Creating and Initializing Local Variable Carnegie Mellon Creating and Initializing Local Variable Variable localx must be stored on stack Because: Need to create pointer to it Compute pointer as -4(%ebp) int add3(int x) { int localx = x; incrk(&localx, 3); return localx; } 8 x 4 Rtn adr Old %ebp %ebp First part of add3 -4 localx = x add3: pushl %ebp movl %esp, %ebp subl $24, %esp # Alloc. 24 bytes movl 8(%ebp), %eax movl %eax, -4(%ebp) # Set localx to x -8 Unused -12 -16 -20 -24 %esp

Creating Pointer as Argument Carnegie Mellon Creating Pointer as Argument Use leal instruction to compute address of localx int add3(int x) { int localx = x; incrk(&localx, 3); return localx; } 8 x 4 Rtn adr Old %ebp %ebp Middle part of add3 -4 localx movl $3, 4(%esp) # 2nd arg = 3 leal -4(%ebp), %eax # &localx movl %eax, (%esp) # 1st arg = &localx call incrk -8 Unused -12 -16 -20 3 %esp+4 -24 %esp

Retrieving local variable Carnegie Mellon Retrieving local variable Retrieve localx from stack as return value int add3(int x) { int localx = x; incrk(&localx, 3); return localx; } 8 x 4 Rtn adr Old %ebp %ebp Final part of add3 -4 localx movl -4(%ebp), %eax # Return val= localx leave ret -8 Unused -12 -16 -20 -24 %esp

IA32 Optimization: Inlining Carnegie Mellon IA32 Optimization: Inlining When both functions in same file: Generating Pointer /* Compute x + 3 */ int add3(int x) { int localx = x; incrk(&localx, 3); return localx; } add3: pushl %ebp movl %esp, %ebp movl 8(%ebp), %eax addl $3, %eax popl %ebp ret Referencing Pointer /* Increment value by k */ void incrk(int *ip, int k) { *ip += k; }

Expand callee into caller Carnegie Mellon How Inlining Works Generating Pointer Expand callee into caller /* Compute x + 3 */ int add3(int x) { int localx = x; incrk(&localx, 3); return localx; } /* Compute x + 3 */ int add3_inline(int x) { int localx = x; *(&localx) += 3; return localx; } Referencing Pointer /* Increment value by k */ void incrk(int *ip, int k) { *ip += k; }

IA 32 Procedure Summary Important Points Carnegie Mellon IA 32 Procedure Summary Important Points Stack is the right data structure for procedure call / return If P calls Q, then Q returns before P Recursion (& mutual recursion) handled by normal calling conventions Can safely store values in local stack frame and in callee-saved registers Put function arguments at top of stack Result return in %eax Pointers are addresses of values On stack or global Caller Frame Arguments Return Addr %ebp Old %ebp Saved Registers + Local Variables Argument Build %esp

IA32 Object Code Setup Assembly Code Disassembled Object Code Carnegie Mellon IA32 Object Code Setup Label .L8 becomes address 0x80484b8 Label .L4 becomes address 0x8048680 Assembly Code switch_eg: . . . ja .L8 # If unsigned > goto default jmp *.L4(,%eax,4) # Goto *JTab[x] Disassembled Object Code 08048480 <switch_eg>: . . . 8048489: 77 2d ja 80484b8 <switch_eg+0x38> 804848b: ff 24 85 80 86 04 08 jmp *0x8048680(,%eax,4)

IA32 Object Code (cont.) Jump Table Carnegie Mellon IA32 Object Code (cont.) Jump Table Doesn’t show up in disassembled code Can inspect using GDB gdb switch (gdb) x/7xw 0x8048680 Examine 7 hexadecimal format “words” (4-bytes each) Use command “help x” to get format documentation 0x8048680: 0x080484b8 0x08048492 0x0804849b 0x080484a4 0x8048690: 0x080484b8 0x080484ae 0x080484ae

IA32 Object Code (cont.) Deciphering Jump Table Carnegie Mellon IA32 Object Code (cont.) Deciphering Jump Table 0x8048680: 0x080484b8 0x08048492 0x0804849b 0x080484a4 0x8048690: 0x080484b8 0x080484ae 0x080484ae Address Value x 0x8048680 0x80484b8 0x8048684 0x8048492 1 0x8048688 0x804849b 2 0x804868c 0x80484a4 3 0x8048690 4 0x8048694 0x80484ae 5 0x8048698 6

Disassembled Targets 8048492: 8b 45 0c mov 0xc(%ebp),%eax Carnegie Mellon Disassembled Targets 8048492: 8b 45 0c mov 0xc(%ebp),%eax 8048495: 0f af 45 10 imul 0x10(%ebp),%eax 8048499: eb 22 jmp 80484bd <switch_eg+0x3d> 804849b: 8b 45 0c mov 0xc(%ebp),%eax 804849e: 99 cltd 804849f: f7 7d 10 idivl 0x10(%ebp) 80484a2: eb 05 jmp 80484a9 <switch_eg+0x29> 80484a4: b8 01 00 00 00 mov $0x1,%eax 80484a9: 03 45 10 add 0x10(%ebp),%eax 80484ac: eb 0f jmp 80484bd <switch_eg+0x3d> 80484ae: b8 01 00 00 00 mov $0x1,%eax 80484b3: 2b 45 10 sub 0x10(%ebp),%eax 80484b6: eb 05 jmp 80484bd <switch_eg+0x3d> 80484b8: b8 02 00 00 00 mov $0x2,%eax

Matching Disassembled Targets Carnegie Mellon Matching Disassembled Targets 8048492: 8b 45 0c mov 0xc(%ebp),%eax 8048495: 0f af 45 10 imul 0x10(%ebp),%eax 8048499: eb 22 jmp 80484bd <switch_eg+0x3d> 804849b: 8b 45 0c mov 0xc(%ebp),%eax 804849e: 99 cltd 804849f: f7 7d 10 idivl 0x10(%ebp) 80484a2: eb 05 jmp 80484a9 <switch_eg+0x29> 80484a4: b8 01 00 00 00 mov $0x1,%eax 80484a9: 03 45 10 add 0x10(%ebp),%eax 80484ac: eb 0f jmp 80484bd <switch_eg+0x3d> 80484ae: b8 01 00 00 00 mov $0x1,%eax 80484b3: 2b 45 10 sub 0x10(%ebp),%eax 80484b6: eb 05 jmp 80484bd <switch_eg+0x3d> 80484b8: b8 02 00 00 00 mov $0x2,%eax Value 0x80484b8 0x8048492 0x804849b 0x80484a4 0x80484ae