From 19a7e4ec2d2206261761521842a66a3936c83691 Mon Sep 17 00:00:00 2001
From: "github-classroom[bot]"
<66690702+github-classroom[bot]@users.noreply.github.com>
Date: Wed, 20 Mar 2024 22:21:48 +0000
Subject: [PATCH] Initial commit
---
.gitignore | 10 +++
Makefile | 23 ++++++
README.md | 189 +++++++++++++++++++++++++++++++++++++++++++++++++
bit_utils.c | 29 ++++++++
bit_utils.h | 30 ++++++++
instructions.c | 51 +++++++++++++
instructions.h | 91 ++++++++++++++++++++++++
lab04.c | 91 ++++++++++++++++++++++++
8 files changed, 514 insertions(+)
create mode 100644 .gitignore
create mode 100644 Makefile
create mode 100644 README.md
create mode 100644 bit_utils.c
create mode 100644 bit_utils.h
create mode 100644 instructions.c
create mode 100644 instructions.h
create mode 100644 lab04.c
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..7e1824b
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,10 @@
+# Build files
+*.o
+main
+
+# macOS
+.DS_Store
+
+# IDE
+.idea
+.vscode
diff --git a/Makefile b/Makefile
new file mode 100644
index 0000000..55705e2
--- /dev/null
+++ b/Makefile
@@ -0,0 +1,23 @@
+CC=gcc
+CFLAGS=-c -Wall -Werror -g
+
+all: lab04
+
+lab04: bit_utils.o instructions.o lab04.o
+ ${CC} $^ -o lab04
+
+bit_utils.o: bit_utils.c bit_utils.h
+ $(CC) $(CFLAGS) bit_utils.c
+
+instructions.o: instructions.c instructions.h bit_utils.h
+ $(CC) $(CFLAGS) instructions.c
+
+lab04.o: lab04.c instructions.h
+ $(CC) $(CFLAGS) lab04.c
+
+run: lab04
+ ./lab04
+
+clean:
+ rm -f lab04 *.o
+
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..b202eb1
--- /dev/null
+++ b/README.md
@@ -0,0 +1,189 @@
+
+# Lab 4
+
+In this lab, you'll gain experience working with MIPS instructions by coding a MIPS VM/simulator. Your simulator will
+
+1. convert an integer value into a 32-bit machine (i.e., binary) instruction,
+2. determine the type of instruction (R or I),
+3. identify the field values (i.e., opcode, rs, rt, rd, sa, func, immediate, etc.), and
+4. execute the instruction using a simulated set of registers.
+
+In particular, this lab has four goals:
+
+In this lab, you will apply the `bit_select` function in Lab 3 (the code is given here, in case you need it) to identify the fields in a binary instruction. You will develop a program that closely simulates a real problem and gain experience with C structs, enums, pointers, and `malloc`.
+
+Contents
+
+- [Lab 4](#lab-4)
+ - [Pre-lab knowledge](#pre-lab-knowledge)
+ - [Background reading](#background-reading)
+ - [Fixed-width integer types](#fixed-width-integer-types)
+ - [Lab structure](#lab-structure)
+ - [Part 0](#part-0)
+ - [Part 1](#part-1)
+ - [Part 2](#part-2)
+ - [Examples/testing](#examplestesting)
+ - [Submit your assignment](#submit-your-assignment)
+
+2N - 1
since they are unsigned. You can read more about these types in this [article](https://www.badprog.com/c-type-what-are-uint8-t-uint16-t-uint32-t-and-uint64-t) or this [manual](https://www.gnu.org/software/libc/manual/html_node/Integers.html).
+
+## Lab structure
+
+```text
+.
+├── Makefile
+├── README.md
+├── bit_utils.c - Do not modify (functions from previous lab).
+├── bit_utils.h - Do not modify (from previous lab).
+├── instructions.c - Contains functions to be implemented in Part 1.
+├── instructions.h - Do not modify and please read carefully comments (important definitions and information).
+└── lab04.c - Contains functions to be implemented in Part 2.
+```
+
+There are comments in the `lab04.c` and `instructions.c` source files as well as in the `instructions.h` header file. Read the comments carefully, as they provide information needed to complete this assignment.
+
+## Part 0
+
+Take a look through each source file and read the comments to have a better understanding of the assignment and the functions you will need to implement.
+
+## Part 1
+
+In this part, you will edit `instructions.c`.
+
+First, implement `get_type_of_instruction`. This function determines the instruction type given an unsigned 32-bit integer by returning an `instruction_type` enum value. For example, if the input is the following 32-bit integer:
+
+```text
+00000001000000110001000000100010
+```
+
+the function should be able to recognize it as an R-type instruction and return `R_TYPE`. Although J-type instructions also exist in MIPS, you need only distinguish between R-type or I-type in this lab.
+
+Then, implement `create_r_instruction`, `create_i_instruction`, which serve as "constructors" for the `r_instruction` and `i_instruction` structs, which are defined as follows in `instructions.h`:
+
+```c
+typedef struct {
+ uint8_t rs;
+ uint8_t rt;
+ uint8_t rd;
+ uint8_t shamt;
+ uint8_t func;
+} r_instruction;
+
+typedef struct {
+ uint8_t opcode;
+ uint8_t rs;
+ uint8_t rt;
+ uint16_t immediate;
+} i_instruction;
+```
+
+`create_r_instruction` and `create_i_instruction` are given the instruction as a 32-bit integer and return a pointer to a `malloc`'d `r_instruction`/`i_instruction` with all fields initialized.
+
+In the next part of this lab, these structures will be used for execution.
+
+To complete this part, reference the top left of the MIPS cheat sheet, which details the position of the fields that each instruction type has:
+
+
+
+
MIPS instruction format
+ +`instructions.h` has several constants that you should use to implement these functions. Specifically, once you have identified whether the instruction is R-type or I-type, you need only extract the bits at the correct positions to initialize the instruction struct. For example, bits 5-0 are the funct bits for an R-type instruction, so the constants `FUNC_START_BIT` and `FUNC_END_BIT` are defined as 5 and 0 and given in `instructions.h`. + +## Part 2 + +In `lab04.c`, implement `execute_r_instruction` and `execute_i_instruction`. `execute_r_instruction` accepts a pointer to an `r_instruction`, while `execute_i_instruction` accepts a pointer to an `i_instruction`. Each function first determines the specific instruction to execute (e.g., if R-type, is the instruction `add`, `sub`, etc.) and then executes it. + +Each instruction modifies a register, so update the `registers` global array variable as appropriate. For example, for the MIPS instruction + +```asm +and $4, $1, $2 +``` + +Register 4 would be updated with the contents of the bitwise AND of register #1 and register #2, as follows: + +```c +registers[4] = registers[1] & registers[2]; +``` + +The following MIPS instructions need to be implemented: `sll`, `sra`, `add`, `sub`, `and`, `or`, `nor`, `addi`, `andi`, `ori`. + +**Note**: There are some nuances in MIPS architecture that we will ignore in this lab. For example, `$0`/`$zero` (the "zero register") is usually read-only and always contains the value 0. However, for this lab, all 32 registers can be considered general registers. Further, do not need to consider the signed extension of immediate values. Each update to a register should be fairly straightforward, similar to above. + +## Examples/testing + +Consider the following sequence of MIPS instructions: + +```asm +addi $0, $0, 21834 # r[0] = r[0] + 21834 +add $1, $0, $0 # r[1] = r[0] + r[0] +sub $2, $1, $0 # r[2] = r[1] - r[0] +ori $3, $2, 1 # r[3] = r[2] | 1 +``` + +In order to execute this series of instructions, each instruction will need to be translated to its binary equivalent by using an assembler. Using your [MIPS cheatsheet](https://sakai.unc.edu/access/content/group/167842e9-e6e0-4d16-81bd-842fcf59831e/Supplemental/mips_cheat_sheet.pdf), you can do this translation yourself. + +```text +00100000000000000101010101001010 +00000000000000000000100000100000 +00000000001000000001000000100010 +00110100010000110000000000000001 +``` + +For this lab, each instruction is accepted as an integer through standard input, so each binary instruction will need to undergo another conversion: + +```text +536892746 +2080 +2101282 +876806145 +``` + +You can then enter each integer into the simulator. To exit the simulator, you can enter the integer `4294967295`. The `main` method handles this special instruction, so you don't need to implement it. + +```text +Please enter your instruction as a 32-bit integer: 536892746 +Current register status: +[21834, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0] +Please enter your instruction as a 32-bit integer: 2080 +Current register status: +[21834, 43668, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0] +Please enter your instruction as a 32-bit integer: 2101282 +Current register status: +[21834, 43668, 21834, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0] +Please enter your instruction as a 32-bit integer: 876806145 +Current register status: +[21834, 43668, 21834, 21835, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0] +Please enter your instruction as a 32-bit integer: 4294967295 +``` + +## Submit your assignment + +1. Use git to push your finished code to this GitHub repository. +2. Go to the COMP 211 course in Gradescope and click on the assignment called **Lab 04**. +3. Click on the option to **Submit Assignment** and choose GitHub as the submission method. +4. You should see a list of your public repositories. Select the one named **lab-04-yourname** and submit it. +5. Your assignment should be autograded within a few seconds, and you will receive feedback for the autograded portion. +6. If you receive all the points, then you have completed this lab! Otherwise, you are free to keep pushing commits to your GitHub repository and submit for regrading up until the deadline of the lab. diff --git a/bit_utils.c b/bit_utils.c new file mode 100644 index 0000000..021764a --- /dev/null +++ b/bit_utils.c @@ -0,0 +1,29 @@ +// Do not edit this file + +#include "bit_utils.h" + +unsigned int mask(unsigned int num, unsigned int bits) { + return num & bits; +} + +unsigned int set(unsigned int num, unsigned int bits) { + return num | bits; +} + +unsigned int inverse(unsigned int num, unsigned int bits) { + return num ^ bits; +} + +unsigned int bit_select(unsigned int num, + unsigned int startbit, + unsigned int endbit) { + startbit++; + num <<= (SIZE - startbit); + num >>= (SIZE - startbit + endbit); + return num; +} + +unsigned int barrel_shift(unsigned int num, unsigned int shamt) { + unsigned int right_bits = num << (sizeof(unsigned int) * 8 - shamt); + return (num >> shamt) | right_bits; +} diff --git a/bit_utils.h b/bit_utils.h new file mode 100644 index 0000000..1c8e378 --- /dev/null +++ b/bit_utils.h @@ -0,0 +1,30 @@ +// Do not edit this file + +#include