How could it be possible to use a single n box as my tutor taught? Won't a new n box be created whenever factorial(n - 1) is called?
Yes, exactly. Each time you call a function, additional memory is set aside for the function arguments and local variables.
Let's drop a couple of levels below C and talk about how a compiled program is laid out in memory1. We're going to assume a 32-bit, x86-like system for this discussion.
While the details vary from platform to platform, the general concepts are the same for most systems you're likely to work on. When a compiled program is loaded into memory, it's usually divided up into several logical segments; one segment contains the machine code, another contains constant (read-only) data items, another contains space for non-constant, global data items, another contains space for the stack, another contains the space for the heap, etc. The general layout looks something like the following (taken from this page):
+------------------------+
high address | Command line arguments |
| and environment vars |
+------------------------+
| stack |
| - - - - - - - - - - - |
| | |
| V |
| |
| ^ |
| | |
| - - - - - - - - - - - |
| heap |
+------------------------+
| global and read- |
| only data |
+------------------------+
low address | program text |
| (machine code) |
+------------------------+
As your program runs, it uses the stack to store function arguments and local variables, along with some program state (previous frame pointer, address to the next instruction). On x86-like systems, the stack grows "downwards", towards decreasing addresses. Among the registers2 provided, there are usually two to keep track of what's on the stack. There's the stack pointer, which points to the most recent item on the stack, and the frame pointer3, which points to the current stack frame. A stack frame is a region of the stack that contains the arguments and local variables for the current function, along with any saved program state:
+ - - - - - - - - - +
high address | previous stack |
| frame |
+ - - - - - - - - - +
| previous stack |
| frame |
+--------------------+ ----------------------+
| function arguments | |
| ... | |
+--------------------+ |
| return address | |
+--------------------+ +-- current stack frame
| old frame pointer | <-- frame pointer |
+--------------------+ |
| local variables | |
| ... | <-- stack pointer |
+--------------------+ -----------------------+
low address | available memory |
+ - - - - - - - - - +
On the systems I'm familiar with, the function arguments and program state are stored "above" the location stored in the frame pointer, and local variables are stored "below" it. Within the generated assembly code, arguments and variables are referred to by their offset from the address stored in the frame pointer. On x86. it would look something like
cmpl $1, 8(%ebp) ;; compare the literal value 1 to the first function argument
movl $1, -4(%ebp) ;; write the literal value 1 to the first local variable
The first function argument is stored 8 bytes "above" the address value stored in the frame pointer %ebp
, while the first local variable would be 4 bytes "below" it (note that this assumes a 32-bit system).
The exact size of the frame depends on how many arguments and local variables need to be stored, although the platform may also require it to be aligned to certain boundary (IOW, the size of the frame must be a multiple of some number of bytes, usually 4 or 8, sometimes 16; for the purposes of this discussion, we're not going to worry about alignment).
Suppose your factorial
function is called with an argument value of 3:
x = factorial( 3 );
A new stack frame will be set up that looks something like this:
+--------------------+
| 3 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer | <-- frame pointer, stack pointer
+--------------------+
where the "return address" is the address of the instruction immediately following the function call in the caller (in this case, it would be the instruction that assigns the result of factorial(3)
to x
).
Since there are no local variables, no space is allocated for them in the stack frame. Now, since n
is not less than 2, you execute the line
n = n * factorial( n - 1 );
This means factorial
is called again, this time with an argument value of 2. A new stack frame is created
+--------------------+
| 3 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer |
+--------------------+
| 2 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer | <-- frame pointer, stack pointer
+--------------------+
n
is still not less than 2, so we call factorial
again with an argument of 1:
+--------------------+
| 3 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer |
+--------------------+
| 2 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer |
+--------------------+
| 1 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer | <-- frame pointer, stack pointer
+--------------------+
1
is less than 2, so we execute the line
return n;
The common convention for returning values from functions is to write the value to a particular register. On x86, integer values are returned through register %eax
, so the compiled code will do something like
movl 8(%ebp), %eax
Now register %eax
will contain the value 1
.
When the function returns, the stack and frame pointers are restored to their previous values (the stack contents are left as-is):
+--------------------+
| 3 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer |
+--------------------+
| 2 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer | <-- frame pointer, stack pointer
+--------------------+
| 1 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer |
+--------------------+
So now we need to execute the remainder of
n = n * factorial( n - 1 );
which will look something like4
movl %eax, %edx ;; save the value returned from factorial(1) to register %edx
movl 8(%ebp), %eax ;; move the value of n (2) to %eax
imull %eax, %edx ;; multiply the values in %eax and %edx, save the result in
;; %eax
Once again, we return n
and exit the function, bringing us back to the previous call:
+--------------------+
| 3 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer | <-- frame pointer, stack pointer
+--------------------+
| 2 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer |
+--------------------+
| 1 | <-- n
+--------------------+
| return address |
+--------------------+
| old frame pointer |
+--------------------+
Lather, rinse, repeat. The value returned in %eax
(2
) is multiplied by the value in the argument 8(%ebp)
(3
), and we write 6
back to %eax
.
Now, the C language definition says nothing about stacks, or stack frames, or registers, or anything like that; it only specifies the behavior of variables within a program, not how to implement that behavior. However, much of C's design is based on what most computer systems already provide (this is what people mean when they say that C is "close to the hardware"), so this is a very natural way of implementing that behavior. If you were working on a machine that had no stack, C programs would be somewhat more difficult to implement.
1. By which I mean a virtual address space, which is an idealized model of memory that your program sees; the OS and memory controllers will map this idealized model to physical RAM.
2. A register is a memory cell on the CPU itself. There are usually several special-purpose registers (program counter, stack pointer, frame pointer) and some number of general-purpose registers. Reading and writing to registers is faster than reading and writing to RAM.
3. On 32-bit x86, the stack pointer register is esp
and the frame pointer register is ebp
(a.k.a. the base pointer). On 64-bit x86, they're named rsp
and rbp
.
4. This is how the compiled code on my system does it, anyway.
Best Answer
It's going to allocate the space for the int on the stack, copy the value of 'num1' onto the stack, call the method and then return the value, at which point the stack is popped. Essentially you'll have a copy of 'num1' on the stack for your function to use.
If you passed a pointer or a reference to 'num1' instead, then the address would be put on the stack, and your function would amend the variable via that address. So you wouldn't have a value copy but rather a reference to the original variable.
Here's a useful description of what's going on (from StackOverflow)