Procedures & The Stack II CSE 351 Autumn 2016

Slides:



Advertisements
Similar presentations
University of Washington Procedures and Stacks II The Hardware/Software Interface CSE351 Winter 2013.
Advertisements

Copyright 2014 – Noah Mendelsohn UM Macro Assembler Functions Noah Mendelsohn Tufts University Web:
Machine/Assembler Language Putting It All Together Noah Mendelsohn Tufts University Web:
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.
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.
Ithaca College Machine-Level Programming IV: IA32 Procedures Comp 21000: Introduction to Computer Systems & Assembly Lang Spring 2013 * Modified slides.
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: X86-64 Topics Registers Stack Function Calls Local Storage X86-64.ppt CS 105 Tour of Black Holes of Computing.
Machine-Level Programming III: Procedures Topics IA32 stack discipline Register-saving conventions Creating pointers to local variables CS 105 “Tour of.
Machine-level Programming III: Procedures Topics –IA32 stack discipline –Register saving conventions –Creating pointers to local variables.
X86_64 programming Tutorial #1 CPSC 261. X86_64 An extension of the IA32 (often called x86 – originated in the Intel 8086 processor) instruction set to.
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.
1 Assembly Language: Function Calls Jennifer Rexford.
Spring 2016Assembly Review Roadmap 1 car *c = malloc(sizeof(car)); c->miles = 100; c->gals = 17; float mpg = get_mpg(c); free(c); Car c = new Car(); c.setMiles(100);
Reading Condition Codes (Cont.)
Credits and Disclaimers
Credits and Disclaimers
CSCE 212 Computer Architecture
Instructor: Your TA(s)
Recitation: Attack Lab
143A: Principles of Operating Systems Lecture 4: Calling conventions
Procedures & Executables CSE 351 Spring 2017
The Stack & Procedures CSE 351 Spring 2017
Machine-Level Programming III: Procedures
Carnegie Mellon Machine-Level Programming III: Switch Statements and IA32 Procedures / : Introduction to Computer Systems 7th Lecture, Sep.
Procedures & The Stack Announcements Lab 1 graded HW 2 out
Switch & Procedures & The Stack I CSE 351 Winter 2017
x86-64 Programming III & The Stack CSE 351 Winter 2018
x86 Programming II CSE 351 Autumn 2016
Machine-Level Programming III: Switch Statements and IA32 Procedures
Machine-Level Programming 4 Procedures
Assembly Programming IV CSE 351 Spring 2017
x86-64 Programming III & The Stack CSE 351 Winter 2018
Machine-Level Programming III: Procedures /18-213/14-513/15-513: Introduction to Computer Systems 7th Lecture, September 18, 2018.
Procedures & The Stack I CSE 351 Autumn 2016
Arrays CSE 351 Winter
Carnegie Mellon Machine-Level Programming III: Procedures : Introduction to Computer Systems October 22, 2015 Instructor: Rabi Mahapatra Authors:
Procedures & The Stack II CSE 351 Winter 2017
Procedures & Executables CSE 351 Autumn 2017
Recitation: Attack Lab
Roadmap C: Java: Assembly language: OS: Machine code: Computer system:
x86-64 Programming III & The Stack CSE 351 Autumn 2017
Machine-Level Programming III: Procedures Sept 18, 2001
The Stack & Procedures CSE 351 Winter 2018
Machine Level Representation of Programs (IV)
x86 Programming III CSE 351 Autumn 2016
The Stack & Procedures CSE 410 Winter 2017
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.
Oct 15, 2018 Instructor: Your TA(s) 1.
Machine-Level Representation of Programs (x86-64)
Ithaca College Machine-Level Programming VII: Procedures Comp 21000: Introduction to Computer Systems & Assembly Lang Spring 2017.
Where is all the knowledge we lost with information? T. S. Eliot
CS201- Lecture 8 IA32 Flow Control
Loops, Switches, Intro to Stack & Procedures CSE 351 Winter 2019
Ithaca College Machine-Level Programming VII: Procedures Comp 21000: Introduction to Computer Systems & Assembly Lang Spring 2017.
Credits and Disclaimers
Credits and Disclaimers
Credits and Disclaimers
Instructor: Your TA(s)
Presentation transcript:

Procedures & The Stack II CSE 351 Autumn 2016 Instructor: Justin Hsia Teaching Assistants: Chris Ma Hunter Zahn John Kaltenbach Kevin Bi Sachin Mehta Suraj Bhat Thomas Neuman Waylon Huang Xi Liu Yufang Sun http://xkcd.com/244/

Administrivia Lab 2 due Friday Midterm on Nov. 2 in lecture CSE351 Lecture 12 Administrivia Lab 2 due Friday Midterm on Nov. 2 in lecture Make a cheat sheet! – two-sided letter page, handwritten Historically my exams have averages of 65-70% Check Piazza this week for announcements & practice problems Midterm review session 5-7pm on Monday, Oct. 31 in EEB 105 Look for additional staff office hours as well

Example: increment long increment(long *p, long val) { long x = *p; CSE351 Lecture 12 Example: increment long increment(long *p, long val) { long x = *p; long y = x + val; *p = y; return x; } (on handout) Register Use(s) %rdi 1st arg (p) %rsi 2nd arg (val), y %rax x, return value increment: movq (%rdi), %rax addq %rax, %rsi movq %rsi, (%rdi) ret

Procedure Call Example (initial state) CSE351 Lecture 12 Procedure Call Example (initial state) Initial Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> ⟵%rsp call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret Return address on stack is the address of instruction immediately following the call to “call_incr” Shown here as main, but could be anything) (on handout) We’re at the beginning of call_incr, let’s say it was called from main, so we already have some stuff up further in the stack, and the last thing, because we just started call_incr, is the return address to jump back into main

Procedure Call Example (step 1) CSE351 Lecture 12 Procedure Call Example (step 1) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> 351 Unused ⟵old %rsp call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret ⟵%rsp+8 Allocate space for local vars ⟵%rsp Setup space for local variables Only v1 needs space on the stack Compiler allocated extra space Often does this for a variety of reasons, including alignment Remember back to when everyone was asking how we know where our variables live… Here, the compiler has created v1 on the stack…

Procedure Call Example (step 2) CSE351 Lecture 12 Procedure Call Example (step 2) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> 351 Unused call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret ⟵%rsp+8 ⟵%rsp Set up parameters for call to increment Now, we set up the arguments for our call to increment First parameter: &v1 8(%rsp) is the address of that value on our stack Second parameter: 100 use movl because 100 is a small positive value that will fit in 32 bits. The high order bits of %rsi get set to zero for us automatically. Next will be the actual call Register Use(s) %rdi &v1 %rsi 100 Aside: movl is used because 100 is a small positive value that fits in 32 bits. High order bits of rsi get set to zero automatically. It takes one less byte to encode a movl than a movq.

Procedure Call Example (step 3) CSE351 Lecture 12 Procedure Call Example (step 3) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> 351 Unused Return addr <call_incr+?> call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret ⟵%rsp State while inside increment Return address on top of stack is address of the addq instruction immediately following call to increment Push address of next instruction onto the stack In this case it’s the addq Then jump to the increment label Like push %rip+? Execute increment Read initial value where %rdi points Save that in %rax, that’s our return value do increment 100 with addq update what’s in memory Register Use(s) %rdi &v1 %rsi 100 %rax increment: movq (%rdi), %rax addq %rax, %rsi movq %rsi, (%rdi) ret

Procedure Call Example (step 4) CSE351 Lecture 12 Procedure Call Example (step 4) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> 451 Unused Return addr <call_incr+?> call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret ⟵%rsp State while inside increment After code in body has been executed Return Pop return address from stack Jump Like pop %rip Register Use(s) %rdi &v1 %rsi 451 %rax 351 increment: movq (%rdi), %rax # x = *p addq %rax, %rsi # y = x+100 movq %rsi, (%rdi) # *p = y ret

Procedure Call Example (step 5) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> 451 Unused call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret ⟵%rsp+8 ⟵%rsp After returning from call to increment Registers and memory have been modified and return address has been popped off stack Register Use(s) %rdi &v1 %rsi 451 %rax 351

Procedure Call Example (step 6) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> 451 Unused call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret ⟵%rsp+8 ⟵%rsp Update %rax to contain v1+v2 Register Use(s) %rdi &v1 %rsi 451 %rax 451+351

Procedure Call Example (step 7) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> 451 Unused ⟵%rsp call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret ⟵old %rsp De-allocate space for local vars Register Use(s) %rdi &v1 %rsi 451 %rax 802

Procedure Call Example (step 8) Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • Return addr <main+8> ⟵%rsp call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret State just before returning from call to call_incr Register Use(s) %rdi &v1 %rsi 451 %rax 802

Procedure Call Example (step 9) Final Stack Structure long call_incr() { long v1 = 351; long v2 = increment(&v1, 100); return v1+v2; } • • • ⟵%rsp call_incr: subq $16, %rsp movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq 8(%rsp), %rax addq $16, %rsp ret State immediately after returning from call to call_incr Return addr has been popped off stack Control has returned to the instruction immediately following the call to call_incr (not shown here) Register Use(s) %rdi &v1 %rsi 451 %rax 802

Procedures Stack Structure Calling Conventions CSE351 Lecture 12 Procedures Stack Structure Calling Conventions Passing control Passing data Managing local data Register Saving Conventions Illustration of Recursion

Register Saving Conventions When procedure yoo calls who: yoo is the caller who is the callee Can registers be used for temporary storage? No! Contents of register %rdx overwritten by who! This could be trouble – something should be done. Either: Caller should save %rdx before the call (and restore it after the call) Callee should save %rdx before using it (and restore it before returning) yoo: • • • movq $15213, %rdx call who addq %rdx, %rax ret who: • • • subq $18213, %rdx ret ?

Register Saving Conventions CSE351 Lecture 12 Register Saving Conventions “Caller-saved” registers It is the caller’s responsibility to save any important data in these registers before calling another procedure (i.e. the callee can freely change data in these registers) Caller saves values in its stack frame before calling Callee, then restores values after the call “Callee-saved” registers It is the callee’s responsibility to save any data in these registers before using the registers (i.e. the caller assumes the data will be the same across the callee procedure call) Callee saves values in its stack frame before using, then restores them before returning to caller

Silly Register Convention Analogy CSE351 Lecture 12 Silly Register Convention Analogy Parents (caller) leave for the weekend and give the keys to the house to their child (callee) Being suspicious, they put away/hid the valuables (caller-saved) before leaving Warn child to leave the bedrooms untouched: “These rooms better look the same when we return!” Child decides to throw a wild party (computation), spanning the entire house To avoid being disowned, child moves all of the stuff from the bedrooms to the backyard shed (callee-saved) before the guests trash the house Child cleans up house after the party and moves stuff back to bedrooms Parents return home and are satisfied with the state of the house Move valuables back and continue with their lives

x86-64 Linux Register Usage, part 1 %rax Return value Also caller-saved & restored Can be modified by procedure %rdi, ..., %r9 Arguments %r10, %r11 Caller-saved & restored Return value %rax %rdi %rsi %rdx Arguments %rcx %r8 %r9 %r10 Caller-saved temporaries %r11

x86-64 Linux Register Usage, part 2 %rbx, %r12, %r13, %r14 Callee-saved Callee must save & restore %rbp May be used as frame pointer Can mix & match %rsp Special form of callee save Restored to original value upon exit from procedure %rbx %r12 Callee-saved Temporaries %r13 %r14 %rbp Special %rsp

x86-64 64-bit Registers: Usage Conventions CSE351 Lecture 12 x86-64 64-bit Registers: Usage Conventions %rax %r8 Return value - Caller saved Argument #5 - Caller saved %rbx %r9 Callee saved Argument #6 - Caller saved %rcx %r10 Argument #4 - Caller saved Caller saved %rdx %r11 Argument #3 - Caller saved Caller Saved %rsi %r12 Argument #2 - Caller saved Callee saved %rdi %r13 Argument #1 - Caller saved Callee saved %rsp %r14 Stack pointer Callee saved %rbp %r15 Callee saved Callee saved

Callee-Saved Example (step 1) CSE351 Lecture 12 Callee-Saved Example (step 1) Initial Stack Structure long call_incr2(long x) { long v1 = 351; long v2 = increment(&v1, 100); return x+v2; } %rsp . . . Rtn address call_incr2: pushq %rbx subq $16, %rsp movq %rdi, %rbx movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq %rbx, %rax addq $16, %rsp popq %rbx ret Resulting Stack Structure 351 Unused %rsp . . . Rtn address %rsp+8 Saved %rbx Modified example from earlier Now call_incr takes an argument, long x, which is used in computing its result x is in %rdi Need to save it somewhere: use a callee-saved register But call_incr is a callee, too! Have to save %rbx for whoever called call_incr2

Callee-Saved Example (step 2) CSE351 Lecture 12 Callee-Saved Example (step 2) Stack Structure long call_incr2(long x) { long v1 = 351; long v2 = increment(&v1, 100); return x+v2; } 351 Unused %rsp . . . Rtn address %rsp+8 Saved %rbx call_incr2: pushq %rbx subq $16, %rsp movq %rdi, %rbx movq $351, 8(%rsp) movl $100, %esi leaq 8(%rsp), %rdi call increment addq %rbx, %rax addq $16, %rsp popq %rbx ret Can you think of another way we could have done this? Push %rdi onto the stack Why did the compiler choose this way? In this case it’s a wash, but if we ever use %rbx again, then it was worth it Pre-return Stack Structure %rsp . . . Rtn address

Why Caller and Callee Saved? We want one calling convention to simply separate implementation details between caller and callee In general, neither caller-save nor callee-save is “best”: If caller isn’t using a register, caller-save is better If callee doesn’t need a register, callee-save is better If “do need to save”, callee-save generally makes smaller programs Functions are called from multiple places So… “some of each” and compiler tries to “pick registers” that minimize amount of saving/restoring

Register Conventions Summary Caller-saved register values need to be pushed onto the stack before making a procedure call only if the Caller needs that value later Callee may change those register values Callee-saved register values need to be pushed onto the stack only if the Callee intends to use those registers Caller expects unchanged values in those registers Don’t forget to restore/pop the values later!

Procedures Stack Structure Calling Conventions CSE351 Lecture 12 Procedures Stack Structure Calling Conventions Passing control Passing data Managing local data Register Saving Conventions Illustration of Recursion

Recursive Function Compiler Explorer: https://godbolt.org/g/4ZJbz1 /* Recursive popcount */ long pcount_r(unsigned long x) { if (x == 0) return 0; else return (x&1)+pcount_r(x >> 1); } pcount_r: movl $0, %eax testq %rdi, %rdi je .L6 pushq %rbx movq %rdi, %rbx shrq %rdi call pcount_r andl $1, %ebx addq %rbx, %rax popq %rbx .L6: rep ret Compiler Explorer: https://godbolt.org/g/4ZJbz1 Compiled with -O1 for brevity instead of -Og Try -O2 instead!

Recursive Function: Base Case CSE351 Lecture 12 Recursive Function: Base Case /* Recursive popcount */ long pcount_r(unsigned long x) { if (x == 0) return 0; else return (x&1)+pcount_r(x >> 1); } Register Use(s) Type %rdi x Argument %rax Return value pcount_r: movl $0, %eax testq %rdi, %rdi je .L6 pushq %rbx movq %rdi, %rbx shrq %rdi call pcount_r andl $1, %ebx addq %rbx, %rax popq %rbx .L6: rep ret First: look at the test: test if %rdi == 0 If it does, then jump to the ret Note: share ret between both cases, so set rax=0 before jump Rep; ret is just because some HW doesn‘t like jumping to ret Trick because some AMD hardware doesn’t like jumping to ret

Recursive Function: Callee Register Save CSE351 Lecture 12 Recursive Function: Callee Register Save /* Recursive popcount */ long pcount_r(unsigned long x) { if (x == 0) return 0; else return (x&1)+pcount_r(x >> 1); } Register Use(s) Type %rdi x Argument pcount_r: movl $0, %eax testq %rdi, %rdi je .L6 pushq %rbx movq %rdi, %rbx shrq %rdi call pcount_r andl $1, %ebx addq %rbx, %rax popq %rbx .L6: rep ret The Stack Need original value of x after recursive call to pcount_r. “Save” by putting in %rbx (callee saved), but need to save old value of %rbx before you change it. . . . rtn <main+?> saved %rbx %rdi (x) is argument for both pcount_r instances (recursive caller and callee). Choosing “to save” %rsp →

Recursive Function: Call Setup /* Recursive popcount */ long pcount_r(unsigned long x) { if (x == 0) return 0; else return (x&1)+pcount_r(x >> 1); } Register Use(s) Type %rdi x (new) Argument %rbx x (old) Callee saved pcount_r: movl $0, %eax testq %rdi, %rdi je .L6 pushq %rbx movq %rdi, %rbx shrq %rdi call pcount_r andl $1, %ebx addq %rbx, %rax popq %rbx .L6: rep ret The Stack . . . rtn <main+?> saved %rbx %rsp →

Recursive Function: Call /* Recursive popcount */ long pcount_r(unsigned long x) { if (x == 0) return 0; else return (x&1)+pcount_r(x >> 1); } Register Use(s) Type %rax Recursive call return value Return value %rbx x (old) Callee saved pcount_r: movl $0, %eax testq %rdi, %rdi je .L6 pushq %rbx movq %rdi, %rbx shrq %rdi call pcount_r andl $1, %ebx addq %rbx, %rax popq %rbx .L6: rep ret The Stack . . . rtn <main+?> saved %rbx rtn <pcount_r+22> %rsp →

Recursive Function: Result /* Recursive popcount */ long pcount_r(unsigned long x) { if (x == 0) return 0; else return (x&1)+pcount_r(x >> 1); } Register Use(s) Type %rax Return value %rbx x&1 Callee saved pcount_r: movl $0, %eax testq %rdi, %rdi je .L6 pushq %rbx movq %rdi, %rbx shrq %rdi call pcount_r andl $1, %ebx addq %rbx, %rax popq %rbx .L6: rep ret The Stack . . . rtn <main+?> saved %rbx %rsp →

Recursive Function: Completion /* Recursive popcount */ long pcount_r(unsigned long x) { if (x == 0) return 0; else return (x&1)+pcount_r(x >> 1); } Register Use(s) Type %rax Return value %rbx Previous %rbx value Callee restored pcount_r: movl $0, %eax testq %rdi, %rdi je .L6 pushq %rbx movq %rdi, %rbx shrq %rdi call pcount_r andl $1, %ebx addq %rbx, %rax popq %rbx .L6: rep ret The Stack . . . rtn <main+?> saved %rbx %rsp →

Observations About Recursion CSE351 Lecture 12 Observations About Recursion Works without any 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 code explicitly does so (e.g., buffer overflow) Stack discipline follows call / return pattern If P calls Q, then Q returns before P Last-In, First-Out (LIFO) Also works for mutual recursion (P calls Q; Q calls P)

x86-64 Stack Frames Many x86-64 procedures need no stack frame at all Only return address is pushed onto the stack when calling another procedure A procedure does need a stack frame when it: Has too many local variables to hold in caller-saved registers Has local variables that are arrays or structs Uses & to compute the address of a local variable Calls another function that takes more than six arguments Is using caller-saved registers and then calls a procedure Modifies/uses callee-saved registers

x86-64 Procedure Summary Important Points Heavy use of registers Return Addr Saved Registers + Local Variables Argument Build Old %rbp Arguments 7+ Caller Frame %rbp (Optional) %rsp Important Points Procedures are a combination of instructions and conventions Conventions prevent functions from disrupting each other Stack is the right data structure for procedure call/return If P calls Q, then Q returns before P Recursion handled by normal calling conventions Heavy use of registers Faster than using memory Use limited by data size and conventions Minimize use of the Stack

CSE351 Lecture 12 BONUS SLIDES One more x86-64 example that shows passing of more than 6 arguments and passing addresses of local variables. The following example, along with a brief re-cap of x86-64 calling conventions is in the following video: https://courses.cs.washington.edu/courses/cse351/videos/05/056.mp4 Alternate (but similar) version: https://godbolt.org/g/E7UFJ7

Return address to caller of call_proc CSE351 Lecture 12 x86-64 Example (1) long int call_proc() { long x1 = 1; int x2 = 2; short x3 = 3; char x4 = 4; proc(x1, &x1, x2, &x2, x3, &x3, x4, &x4); return (x1+x2)*(x3-x4); } call_proc: subq $32,%rsp movq $1,16(%rsp) # x1 movl $2,24(%rsp) # x2 movw $3,28(%rsp) # x3 movb $4,31(%rsp) # x4 • • • Return address to caller of call_proc ← %rsp Note: Details may vary depending on compiler!

x86-64 Example (2) – Allocate local vars long int call_proc() { long x1 = 1; int x2 = 2; short x3 = 3; char x4 = 4; proc(x1, &x1, x2, &x2, x3, &x3, x4, &x4); return (x1+x2)*(x3-x4); } call_proc: subq $32,%rsp movq $1,16(%rsp) # x1 movl $2,24(%rsp) # x2 movw $3,28(%rsp) # x3 movb $4,31(%rsp) # x4 • • • Return address to caller of call_proc x4 x3 x2 x1 8 16 24 ← %rsp

x86-64 Example (3) – setup params to proc CSE351 Lecture 12 x86-64 Example (3) – setup params to proc long int call_proc() { long x1 = 1; int x2 = 2; short x3 = 3; char x4 = 4; proc(x1, &x1, x2, &x2, x3, &x3, x4, &x4); return (x1+x2)*(x3-x4); } call_proc: • • • leaq 24(%rsp),%rcx # %rcx=&x2 leaq 16(%rsp),%rsi # %rsi=&x1 leaq 31(%rsp),%rax # %rax=&x4 movq %rax,8(%rsp) # arg8=&4 movl $4,(%rsp) # arg7=4 leaq 28(%rsp),%r9 # %r9=&x3 movl $3,%r8d # %r8 = 3 movl $2,%edx # %rdx = 2 movq $1,%rdi # %rdi = 1 call proc Return address to caller of call_proc x4 x3 x2 x1 Arg 8 Arg 7 Same instructions as in video, just a different order. 8 16 24 Argument order: Diane’s Silk Dress Cost $8 9 %rdi, %rsi, %rdx, %rcx, %r8, %r9 ← %rsp

x86-64 Example (4) – call proc long int call_proc() { long x1 = 1; int x2 = 2; short x3 = 3; char x4 = 4; proc(x1, &x1, x2, &x2, x3, &x3, x4, &x4); return (x1+x2)*(x3-x4); } call_proc: • • • leaq 24(%rsp),%rcx leaq 16(%rsp),%rsi leaq 31(%rsp),%rax movq %rax,8(%rsp) movl $4,(%rsp) leaq 28(%rsp),%r9 movl $3,%r8d movl $2,%edx movq $1,%rdi call proc Return address to caller of call_proc x4 x3 x2 x1 Arg 8 Arg 7 Return address to line after call to proc ← %rsp

x86-64 Example (5) – after call to proc CSE351 Lecture 12 x86-64 Example (5) – after call to proc long int call_proc() { long x1 = 1; int x2 = 2; short x3 = 3; char x4 = 4; proc(x1, &x1, x2, &x2, x3, &x3, x4, &x4); return (x1+x2)*(x3-x4); } call_proc: • • • movswl 28(%rsp),%eax # %eax=x3 movsbl 31(%rsp),%edx # %edx=x4 subl %edx,%eax # %eax=x3-x4 cltq movslq 24(%rsp),%rdx # %rdx=x2 addq 16(%rsp),%rdx #%rdx=x1+x2 imulq %rdx,%rax # %rax=rax*rdx addq $32,%rsp ret Return address to caller of call_proc x4 x3 x2 x1 Arg 8 Arg 7 cltq: convert %eax to quad-word: R[%rax] <- SignExtend(R[%eax]). Used to help support multiplication/division operations… movs_ _: Move and sign extend cltq: Sign extend %eax into %rax (special-case to save space) 8 16 24 ← %rsp

x86-64 Example (6) – de-allocate local vars long int call_proc() { long x1 = 1; int x2 = 2; short x3 = 3; char x4 = 4; proc(x1, &x1, x2, &x2, x3, &x3, x4, &x4); return (x1+x2)*(x3-x4); } call_proc: • • • movswl 28(%rsp),%eax movsbl 31(%rsp),%edx subl %edx,%eax cltq movslq 24(%rsp),%rdx addq 16(%rsp),%rdx imulq %rdx,%rax addq $32,%rsp ret Return address to caller of call_proc ← %rsp