Lecture 16

  1. Runtime Stack Organisation

1.0 - Runtime Stack Organisation

1.1 - Stack Frame Organisation

Figure 1: Procedure call stack frame organisation

1.2 - Contents of Activation Record

1.3 - Calling a Procedure

  1. The parameters to the procedure are pushed onto the stack by the calling procedure.
    • Must push the parameters onto the stack in the reverse order, to preserve the convention that the first parameter is stored at an offset of -1 from the frame pointer.
  2. The static link (of the procedure being called) is pushed onto the stack
  3. The current frame pointer is pushed onto the stack to create the dynamic link
  4. Set the frame pointer of the stack machine is set so that it contains the address of the start of the new stack frames (that is, we want the new frame pointer to point to the static link).
  5. The current value of the program counter (which is the address of the instruction after the call) is pushed onto the stack to form the return address.
  6. The program counter is set to the address of the procedure
    • From this point onward, when we execute code (stack machine instructions), we ??
  7. Space is allocated on the stack for any local variables, using the ALLOC_STACK instruction
    • The procedure (that is being called) sets up the space on the stack for the local variables.

1.4 - Returning from a Procedure

  1. The program counter is set to the return address in the current activation record
    • Restore the program counter to the return address of the current activation record.
  2. The frame pointer is set to the dynamic link
    • The position on the stack of the activation record of the calling procedure
  3. The stack pointer is set so that all of the space used by the stack frame (but not the parameters) is popped from the stack (essentially, deallocate the space from the stack).
  4. Execution continues at the instruction addressed by the restored program counter
  5. After return, the calling procedure handles deallocating any parameters (using the instruction DEALLOC_STACK).

1.5 - Procedure Call Example

1.5.1 - Begin Executing the Main Method

begin // Main
	n := 2;
	call fact();
	write f
end
  1. Set up an activation record for the main method.
    • The static link, dynamic link and return address don’t have a value for the main method - the main method is not called by a procedure.
    • We allocate space on the stack for the local variables (n and f)
      • The variable nn is stored at an offset of +3+3 (address fp+3\text{fp}+3)
      • The variable ff is stored at an offset of +4+4 (address of fp+4\text{fp}+4)
    • We then execute the code n:=2n := 2 which assigns the variable n the value of 2 - this is trivial as it is defined as a local variable, and therefore, is stored at some offset from the current frame pointer.
  1. Procedure call to fact()

1.5.2 - Set Up Procedure Call to Fact

  1. To set up the activation record for the procedure fact(), we need to do a few things:
    • Static Link Address on the stack of the activation record for the most recent invocation of the enclosing procedure (in this case, Main). Therefore, in this case, we point to the start of the Main procedure’s activation record. Used to access variables in the parent scope(s).
    • Dynamic Link Address of activation record of the calling procedure.
    • Return Address Should be in between the call fact(); and write f statements. We refer to this position as “Main”

1.5.3 - Executing Procedure Fact

Else Body Execution

1.5.4 - Second Call to Fact()

1.5.5 - Executing Fact Procedure for the Second Time

1.6 - Generating Code for Non-Local Variable Access

// Push the static link for the current frame 
// (i.e. the address of frame for procedure C)
ZERO
LOAD_FRAME
// Replace with static link of next outer frame
// i.e. the address of the frame for the main program
LOAD_ABS
// Add offset of x
LOAD_CON 3
ADD
// Convert from an offset from fp
TO_LOCAL
// Load the value of x
LOAD_FRAME
var x: int;  // first argument to C
    y: int;  // second argument to C
    res: int;  // result from C
procedure C() =
  var n: int;  // argument to fact
      f: int;  // result from fact
  procedure fact() =
    begin
      // assume 0 <= n
      if n = 0 then
        f := 1    
        // f = n!
      else // 0 < n
       begin
        n := n - 1;
        call fact();  
        // f = n!
        n:= n + 1;
        f := f * n   
        // f = n!
       end
    end; // fact
  begin // C
    n := x;
    call fact();   
    // f = x!
    res := f;
    n := y;
    call fact();   
    // f = y!
    res := res / f; 
    // res = x! / y!
    n := x - y;
    call fact();   
    // f = (x-y)!
    res := res / f   
    // res = x! / (y! * (x-y!))
  end; // C
begin // main program
  x := 6;
  y := 2;
  call C();
  write res
end

1.6.1 - visitVariableNode method