← back to syllabus ← back to notes
Let’s learn how to program MIPS assembly language, and run these programs on a MIPS emulator MARS
. You’ll run it using Java, java -jar Mars4_5.jar
.
These class notes focus on on MIPS assembly programming
Your best friend here, MIPS Green Card reference card.
There are two types of procedures, leaf procedures and nested procedures. Each leaf procedure is simply a code block that can be called with or without parameters. It’s a mechanism for reusability. They do not call other procedures. Each nested procedure is a reusable code block with or without parameters, and it does call other procedures within its code block.
When implementing procedures in MIPS assembly, there’s a general flow you should follow, which varies slightly depending on whether the procedure is a leaf or nested.
For emphasis, a leaf procedure is a procedure that does not call any other procedures. The flow for implementing a leaf procedure is relatively straightforward:
$a0 - $a3
).$v0
.jr $ra
instruction.Write a MIPS assembly procedure named add_two_numbers
that takes two integer arguments, adds them together, and returns the sum.
.text
.globl add_two_numbers
add_two_numbers:
# Procedure Prologue (not needed for a simple leaf procedure)
# Procedure Body
add $v0, $a0, $a1 # $v0 = $a0 + $a1
# Procedure Epilogue
jr $ra # Return to the caller
.text:
This directive indicates that the following code is part of the text segment in memory, which contains executable instructions..globl add_two_numbers:
This directive makes the add_two_numbers
label globally visible, allowing it to be called from other parts of the program.add_two_numbers:
: This label marks the beginning of the add_two_numbers
leaf procedure.add $v0, $a0, $a1
: This instruction performs the addition.
$a0
and $a1
are argument registers, where the two input integers are passed to the procedure.$v0
is the value register, where the result of the addition is stored, by ISA convention.jr $ra
: This instruction jumps to the address stored in the return address register $ra
, effectively returning control, that is the PC
, to the calling procedure. .text
.globl main
main:
# Set up arguments
li $a0, 5 # First argument: 5
li $a1, 10 # Second argument: 10
# Call the add_two_numbers procedure
jal add_two_numbers
# Print the result (optional, requires syscall)
move $a0, $v0 # Move result to $a0
li $v0, 1 # System call code for print integer
syscall
# Exit the program
li $v0, 10 # System call code for exit
syscall
main:
: This label marks the beginning of the main program.li $a0, 5
and li $a1, 10
: These instructions load the immediate values 5 and 10 into the argument registers $a0
and $a1
, respectively.jal add_two_numbers
: This instruction jumps and links to the add_two_numbers
procedure, saving the return address in $ra
.move $a0, $v0
: This instruction moves the result from $v0
(where add_two_numbers
stored it) to $a0
, preparing it for the print system call. Note move
is not a core instruction in the MIPS ISA.li $v0, 1
and syscall
: These instructions perform the system call to print the integer in $a0
.li $v0, 10
and syscall
: These instructions perform the system call to exit the program.Again for emphasis, a nested procedure is a procedure that calls other procedures. The flow for implementing a nested procedure is more complex:
$ra
and any callee-saved registers ($s0 - $s7
) on the stack.jal
instruction.$ra
from the stack.jr $ra
instruction.Write the procedure sum_of_squares
that takes two integer arguments, calls the square
procedure to calculate the square of each argument, and returns the sum of the squares.
.text
.globl sum_of_squares
sum_of_squares:
# Prologue (save $ra)
addi $sp, $sp, -4 # Make space on the stack
sw $ra, 0($sp) # Save return address
# Calculate square of the first argument
move $a0, $a0 # Move first argument to $a0 for square
jal square
move $t0, $v0 # Store the result in $t0
# Calculate square of the second argument
move $a0, $a1 # Move second argument to $a0 for square
jal square
add $t0, $t0, $v0 # Add the squares
# Epilogue (restore $ra)
move $v0, $t0 # Store the sum of squares in $v0
lw $ra, 0($sp) # Restore return address
addi $sp, $sp, 4 # Deallocate stack space
jr $ra # Return
.globl square
square:
# Calculate square
mul $v0, $a0, $a0 # $v0 = $a0 * $a0
jr $ra # Return
main
Procedure:
sum_of_squares
.jal
.sum_of_squares
Procedure:
$ra
on the stack because this procedure calls another procedure.
square
.$t0
.square
.$t0
.square
Procedure:
$v0
. .globl main
main:
# Set up arguments for sum_of_squares
li $a0, 3 # First argument: 3
li $a1, 4 # Second argument: 4
# Call the sum_of_squares procedure
jal sum_of_squares
# Print the result (optional, requires syscall)
move $a0, $v0 # Move result to $a0
li $v0, 1 # System call code for print integer
syscall
# Exit the program
li $v0, 10 # System call code for exit
syscall
.globl sum_of_squares
sum_of_squares:
# Prologue (save $ra)
addi $sp, $sp, -4 # Make space on the stack
sw $ra, 0($sp) # Save return address
# Calculate square of the first argument
move $a0, $a0 # Move first argument to $a0 for square
jal square
move $t0, $v0 # Store the result in $t0
# Calculate square of the second argument
move $a0, $a1 # Move second argument to $a0 for square
jal square
add $t0, $t0, $v0 # Add the squares
# Epilogue (restore $ra)
move $v0, $t0 # Store the sum of squares in $v0
lw $ra, 0($sp) # Restore return address
addi $sp, $sp, 4 # Deallocate stack space
jr $ra # Return
.globl square
square:
# Calculate square
mul $v0, $a0, $a0 # $v0 = $a0 * $a0
jr $ra # Return
sum_of_squares
calls square
, making it a nested procedure.sum_of_squares
to allow for the nested call.$a0
and $a1
.$v0
.$t0
is used to store the sum of the squared values.A recursive procedure is a procedure that calls itself. Therefore, it also is a nested procedure. Designing and implementing a recursive procedure involves the following steps:
factorial:
# Prologue (save $ra, etc.)
# ...
beq $a0, $zero, base_case # if n == 0, go to base case
addi $sp, $sp, -8 # make space on stack
sw $ra, 4($sp) # save return address
sw $a0, 0($sp) # save n
addi $a0, $a0, -1 # n = n - 1
jal factorial # recursive call
lw $a0, 0($sp) # restore n
lw $ra, 4($sp) # restore return address
addi $sp, $sp, 8 # deallocate stack space
mul $v0, $a0, $v0 # v0 = n * factorial(n-1)
jr $ra # return
base_case:
li $v0, 1 # v0 = 1
jr $ra # return
Left as an exercise, write the main
and test the program with input values.
add_two_numbers.c
add_two_numbers.asm
as a leaf procedure
nested_proc.asm
square_root.asm