Preparing for Test 1
Information
The first test will take place on November 6th. It serves to examine your understanding of all the material we have seen up until this point. This includes:
Session 1 | Introduction to C |
Session 2 | Assembly basics |
Session 3 | Advanced C & assembly |
Session 4 | Functions and the stack |
Session 5 | Dynamic memory |
The first and second test contribute equally to your final grade for the exercises portion of SOCS. Refer to the ECTS course description for more information about how this course is graded.
The difficulty of the test will not be greater than the more difficult exercises from the exercise sessions. The test will take place on a computer, with a duration of 2 hours. You can access RARS, but you will not have internet access.
Structure of this test
Test 1 will consist of two questions:
Question 1: A question where you have to write RISC-V assembly. You will receive a description of a program, which you will have to implement. We will check that your implementation gives the correct results and that you have respected the RISC-V calling conventions (see session 4). Try to create your own unit tests to check your implementation. This question should take at most 40 minutes.
Question 2: In this question you will be given a program description and an implementation in C. You have to translate this C code into RISC-V assembly. This question is usually a little harder than the first one. It should take at most 1 hour.
Practice exam
To help you prepare for the first test, we have provided a practice exam. The format, difficulty and duration of the practice test are very similar to those of the real test, but the questions will obviously be different.
Question 1: Given a sorted array, the length of the array and a target value. If the target value is in the array, return its index. If it is not in the array, return the index where it should be inserted.
Respect the RISC-V calling conventions. Do not make use of a custom exception handler (it will interfere with the test-suite). Try to write unit tests to test the validity of your implementation. Make sure to think about edge cases. Use .globl findIndex as the function name. Do not forget to ‘ret’ at the end of your function!
Inputs (you can assume that these are given): a0 = array; a1=length; a2=target.
Examples:
array = [1 3 5 7 9] target = 1 -> return 0
array = [1 3 5 7 9] target = 6 -> return 3
Question 2: Determine if a word or phrase is an isogram. An isogram (also known as a “non-pattern word”) is a word or phrase without a repeating letter.
Examples of isograms:
lumberjacks
background
downstream
Create a RISC-V program that checks whether a given input string (given in register a0) is an isogram. An implementation in C is given below to help you solve this problem. Instead of printing the result, you should return 1 if the input string is an isogram, 0 if it isn’t. You can assume that the input strings only contain characters from the alphabet, so no hyphens, spaces… All input characters are lowercase. No input validation is needed.
The C code is given below:
#include <stdio.h>
#include <string.h>
#include <ctype.h>
int is_isogram(const char *str) {
int char_count[26] = {0}; // Assuming only lowercase letters are considered
for (int i = 0; str[i] != '\0'; i++) {
char c = tolower(str[i]);
if (isalpha(c)) {
int index = c - 'a';
if (char_count[index] > 0) {
return 0; // Not an isogram
}
char_count[index]++;
}
}
return 1; // Isogram
}
int main() {
char input[1000];
printf("Enter a string: ");
fgets(input, sizeof(input), stdin);
// Remove the newline character from fgets
input[strcspn(input, "\n")] = '\0';
if (is_isogram(input)) {
printf("The string is an isogram.\n");
} else {
printf("The string is not an isogram.\n");
}
return 0;
}
Aside from attempting the practice exam, you should redo as many exercises from the previous sessions as possible. These materials will be sufficient for you to pass the test with flying colors. The additional exercises from session 3 are especially recommended. Try to apply your knowledge from session 4 and 5 to those questions if you want to be fully prepared for the exam.
Tip: Don’t neglect the theoretical aspects of this course! Having a solid grasp of the theory will make it a lot easier for you to (re)do the exercises and complete the practice exam.
Writing unit tests
To test the validity of your implementation, it is useful to write simple unit tests and test your program for different cases. To do this, you open a new file and make sure your testfile is in the same directory as your implementation or you assemble all the currently open files by going to Settings
and selecting the box Assemble all files currently open
. You then load the necessary input for your implementation into the correct registers and run your implementation with a jump-and-link instruction. To check if the return value is correct, you can simply put the correct value in a register and check if it matches the return value with a Branch if equal instruction. If the value is correct, load a certain value in a register and another value if it is wrong.
A simple unit test for question 1 above could for example be:
.globl main
.data
array: .space 12
.text
main:
# Making an array [1,2,3] pointed to by a0
la a0, array
li t0, 1
li t1, 2
li t2, 3
sw t0, 0(a0)
sw t1, 4(a0)
sw t2, 8(a0)
# Load the other input into the correct registers
li a1, 3 # a1 = length of the array
li a2, 2 # a2 = target value
# Run your implementation
jal findIndex
# Load the expected value into a register (in this case the expected index is 1)
li t3, 1
# Check if the returned value matches the expected value
beq a0, t3, test_success
bne a0, t3, test_fail
test_success:
li t4, 1 # If success, load 1 into t4
j end
test_fail:
li t4, -1 # If failed, load -1 into t4
end:
li a7, 93 # Exit system call ends program
ecall
After you run the test, register t4 will contain the value 1 if the test was a succes and -1 if the test failed.
Recursion
What happens when a function calls itself? Functions that depend on a simpler or previous version of themselves are called recursive functions. This also means we have to be very careful when considering the calling convetions mentioned before.
Exercise Recursion
Consider the following recursive function which calculates n!
.
unsigned int fact(unsigned int n) {
if (n < 2) return 1;
return n*fact(n-1);
}
- Convert this function to RISC-V.
- Consider the call
fact(3)
. What is the state of stack when it reaches its maximum size (at the deepest level of recursion)? - In exercise 3 of the first session you implemented an iterative factorial function. Compare both factorial implementations in terms of memory usage. Which implementation do you prefer?
Solution
.globl main
fact:
addi sp, sp, -8 # Make space for 2 words on the stack
sw ra, 4(sp) # Store the (caller-save) return address
sw a0, 0(sp) # Store the (caller-save) register a0
# if (n < 2)
slti t0, a0, 2 # Set t0 to 1 if a0 (n) < 2
beqz t0, _recurse # Branch to _recurse if a0 > 1 (because then t0 would be 0 now)
# return 1;
li a0, 1 # Return 1 if a0 <= 1
addi sp, sp, 8 # Free 2 words of space on stack
ret
_recurse:
#fact(n-1)
addi a0, a0, -1 # Decrement $a0
jal fact # Execute fact
mv t0, a0 # Move fact(n-1) result to t0
lw a0, 0(sp) # Restore initial a0 (old n value)
lw ra, 4(sp) # Restore initial ra
addi sp, sp, 8 # Free 2 words of space on stack
#n*fact(n-1);
mul a0, a0, t0 # a0 = a0 (n) * t0 (fact(n-1))
ret # return n * fact(n-1)
main:
li a0, 5 # Input will be 5
jal fact # a0 = fact(5);
Excursion: Tail recursion
A tail call occurs whenever the last instruction of a subroutine (before the return) calls a different subroutine. Compilers can take advantage of tail calls to reduce memory usage. This is because for tail calls, no additional stack frame needs to be entered. Instead, we can simply overwrite the function parameters, jump to the function and execute from there by reusing the original function stack frame. This is possible since we do not expect to be returned to and instead refer to our original caller that is on our stack frame. Thus, when the (tail-) called function returns, it will not return to us but directly to the original code that called us.
The benefit of tail calls is that they are very light on stack usage. While non-tail recursion adds a stack frame for each recursion depth, tail recursion only uses a single stack frame for any recursion depth.
The call
fact(n-1)
in the previous exercise is not a tail call. Why not?
Solution
The multiplication must be performed after the recursive function returns. Thus, the recursive function call is not the last instruction in the function.
Excursion exercise
We have converted the factorial program to use tail recursion. Translate this program to RISC-V. Try to avoid using the call stack during the fact_tail
implementation. Why is this possible?
unsigned int fact_tail(unsigned int n, unsigned int result) {
if (n <= 1) return result;
return fact_tail(n - 1, n * result);
}
unsigned int fact(unsigned int n) {
return fact_tail(n, 1);
}
int main(void) {
int n = 5;
int r;
r = fact(n);
}
Solution
.globl main
fact_tail:
# if (n < 2)
slti t0, a0, 2 # Set t0 to 1 if a0 (n) < 2
beqz t0, _recurse # Branch to _recurse if a0 > 1 (because then t0 would be 0 now)
# return result;
mv a0, a1
ret
_recurse:
#fact_tail(n - 1, n * result);
mul a1, a0, a1 # a1 = n * result
addi a0, a0, -1 # a0 = n - 1
j fact_tail # a0 = fact_tail(n - 1, n * result)
# Note: NO JAL here -> ra is not overwritten
# We don't need to use jal since there is no code to be executed after the tail call!
# Thus no need to back-up ra on the stack at all.
fact:
li a1, 1
addi sp, sp, -4
sw ra, 0(sp)
jal fact_tail
lw ra, 0(sp)
addi sp, sp, 4
ret
main:
li a0, 5 # Input will be 5
jal fact # a0 = fact(5);
Re-examination
The re-examination will take place in the third exam period. It will consist of only one test, and will cover the full material of the previous two tests (1 through 8). The duration is 2 hours. The questions will have a format and difficulty similar to those of test 1 and 2.