CS 572 Laboratory Assignment 1: Instruction Set Architectures solution

$24.99

Original Work ?
Category: You will Instantly receive a download link for .ZIP solution file upon Payment

Description

5/5 - (10 votes)

0. Introduction
This lab assignment aims to re-acquaint you with the MIPS assembly language. First, you will have to simulate two simple machines. Next, you are required to design binary encodings for two simple instruction set architectures. After designing these encodings and comparing the results with the MIPS Instruction Set Architecture (ISA), you will get a better understanding of the issues of ISA. Last, but not least, you need to write an assembler to automatically translate the source codes to binary, and rewrite the simulators to use (“execute”) the binary format.

Make Your Code Readable
It is very important for you to write well-documented and readable code in this programming assignment. The reason for making your code clear and readable is two-fold. First, subsequent labs will make use of the components developed in this lab. Second, it will be a lot easier for the Grader to grade your programming assignments if you provide well-commented code.

Since there exist a variety of ways to organize and document your code, you are allowed to make use of any particular coding style for this programming assignment. It is believed that reading other people’s code is a way of learning how to writing readable code. Importantly, when you write code, please pay attention to comments which are used to explain what is going on in your simulated systems.

Here Are Some General Tips For Writing Better Code:
A little time spent thinking up better names for variables can make debugging a lot easier. Use descriptive names for variables and procedures.
Group related items together, whether they are variable declarations, lines of code, or functions.
Watch out for uninitialized variables.
Split large functions that span multiple pages. Break large functions down! Keep functions simple.
Always prefer legibility over elegance or conciseness. Note that brevity is often the enemy of legibility.
Code that is sparsely commented is hard to maintain. Comments should describe the programmer’s intent, not the actual mechanics of the code. A comment which says “Find a free disk block” is much more informative than one that says “Find first non-zero element of array.”
Backing up your code as you work is difficult to remember to do sometimes. As soon as your code works, back it up. You always should be able to revert to working code if you accidentally paint yourself into a corner during a “bad day.”
Part 1. Begin Your Assignment
MIPS Assembly
First, download the article – “Assemblers, Linkers, and the SPIM Simulator”, and read Section A.9 on pp. A38-A47 to get familiar with the SPIM simulator. Second, download and save the sample program. This program is intended to evaluate a quadratic. Third, since the sample program contains a few bugs, your will need to find them and correct the code so that it can be properly executed. Leave the bugs in your corrected source, but of course, comment them out. You must add comments to explain what your corrections do. You will need to submit your corrected code.
An Example of a Simulated Machine
Throughout our laboratory assignments, we will use xspim for our example of a simulated machine with general purpose registers that uses loads and stores to access memory. xspim is a self-contained simulator that will run MIPS32 assembly language programs. It reads and executes assembly language programs written for this processor. It also provides a simple debugger and minimal set of operating system services. xspim does NOT execute binary (compiled) programs. Detailed information about xspim can be found at http://www.cs.wisc.edu/~larus/spim.html

Downloading SPIM
Platform

Program

Form

File

Unix or Linux system
Mac OS X

spim
xspim

Source code

http://www.cs.wisc.edu/~larus/SPIM/spim.tar.Z or
http://www.cs.wisc.edu/~larus/SPIM/spim.tar.gz

Linux

spim
xspim

Binary RPM for Fedora

http://www.cs.wisc.edu/cbi/downloads/

Microsoft Windows
(Windows NT, 2000, XP)

(spim 7.0 and later versions no longer run on Windows 95/98. Use version 6.5 or earlier.)

spim
PCSpim

Executable

http://www.cs.wisc.edu/~larus/SPIM/pcspim.zip

Source code

http://www.cs.wisc.edu/~larus/SPIM/pcspim_src.zip

Part 2. Develop Two Simple Simulated Machines
Now that you are in a position to develop two simulated machines: a stack-based machine, and an accumulator-based machine. You can use any language such as C or java.
First, you need to think about the basic components and operations in any machine.
Whether it’s stack, accumulator or GPR-based (General Purpose Register), a program must be stored in memory. Hence, you should simulate a memory system: given an address, it will need to return the contents of that address. Please note that the contents are either instructions or some data. You could use an array, and then the address would just be the array index. However, I’d like you to use the MIPS memory model and 32-bit addresses in your code, so that really isn’t a practical solution.
Other better choices include:

using one array for each memory area (user text, kernel text, user data, the stack, and kernel data). The base address of each area would then be a constant that needs to be added to the index of each array element to form the actual address.
using some form of linked list or heap, if you think code might need to be scattered throughout memory (in which case the array approach would be very wasteful).
hashing, which may be a good choice because it is a primitive in some languages.
Your first module, then, should be something that takes an address and returns the contents of that address. Of course, that only takes care of reads (loads). You also need to take care of writes (stores), and you also need some sort of all-powerful initialization routine that simulates the action of the operating system when it loads all the relevant areas of memory prior to handing execution off to a user program. So that’s actually three modules to start with. The third module (the “loader”) will need some sort of file format to read from – for source code files I’d suggest you use a layout similar to MIPS assembly source files. We’ll get to binaries later…
Now that you can load, read and write memory…
You will have to create two separate simulators – one stack-based, and the other accumulator-based.
Let’s start with the stack-based machine. The instructions you’ll need for your quadratic evaluation code are PUSH, POP, ADD and MULT. Something like END will be handy too.

PUSH x will read the contents of memory at address x and push the value onto the stack.
POP y will pop one word off the top of the stack and write the value to memory at address y.
ADD will remove the top two words from the stack, add them, and leave the result on the top of the stack.
MULT behaves similarly.
END just signals the end of the code. We could think about implementing a return, but that’s overkill for now.
Neither ADD nor MULT have explicit operands – both operate implicitly on the top of the stack.
Your simulator will just be a big case statement inside a loop that reads source code statements one by one and simulates the action(s) of each instruction. There will be one “case” for each type of instruction, so for the stack-based machine you will have a loop containing five cases. Of course, before dropping into the loop you have to “load” your source code into memory and perform any necessary initializations of the data areas. Then you need to give the starting address of the code to the case loop, and start it up. (Hey, that sounds like you need to have something like a program counter, right? Plus some way of automatically incrementing it… note that we’re not handling branches in this lab.)

Recap: Simulator outline
Open source code file
Load source code and data into memory (using all-powerful initialization routine)
Initialize PC
user_mode = true;
while ( user_mode ) {
read memory at PC
increment PC
case PUSH: do something; break;
case POP: do something; break;
case ADD: do something; break;
case MULT: do something; break;
case END: user_mode = false; break;
}
Print summary statistics and selected memory locations
Once you have this simulator ready, write the code for the quadratic evaluator running on the stack-based machine. You will have to demo this to the Grader and convince him that your simulator is behaving correctly (as well as producing the correct final result). This is one of the reasons I’d prefer that you use C or java: you could use the debugger in a very convincing demo.
That’s all there is to the stack-based machine. Um – except for the stack, right? You should know how to build that.
The accumulator-based machine is very similar – the only real change will be the specifics of the cases in the main loop. And of course, the source code that you write for your “machine” to run will be quite different. You only have a single “register” in the CPU, which is of course the accumulator. You can load a value into the accumulator from memory, and store the current value back to memory. The arithmetic operations all have the accumulator as one implicit operand, and an explicit memory location as the other operand. I suppose it would be possible to build an accumulator-based machine where the destination of an arithmetic operation is a specified location in memory, but for this lab let’s make the destination implicitly the accumulator. All this means you’ll have the following instructions:
LOAD x will read the contents of memory at address x and put the value into the accumulator.
STO y will write the contents of the accumulator to memory at address y.
ADD x will read the contents of memory at address x, add the value to the accumulator, and leave the result in the accumulator.
MULT x behaves similarly.
END just signals the end of the code.
A README file will have to be submitted along with your source code. Your README file must include compilation instructions, and instructions on how to use your program. In addition, a discussion of any design issues you ran into while implementing this project and a description of how the program works (or any parts that don’t work) is also appropriate content for the README file.

Part 3. Binary Instruction Encoding
It should be noted that having ASCII-encoded instructions in memory is not realistic. Even in case when very large semiconductor memories are available, one of the key problems is that memory cycle times are much slower than CPU cycle times – and getting slower (in relative terms). One way to solve this problem is to encode instructions and data as densely as possible – so that means we need dense, binary encodings for the instruction sets of both simulated machines.
We’ve only been using five instructions, so you could potentially use a 3-bit opcode. In order to have an apples-to-apples comparison with the code density of the MIPS ISA, I’ll add one “extraneous” condition: your instruction set encoding must be able to accommodate 140 instructions. Other than that, the conditions your encodings must satisfy are:

addresses in main memory are 32-bit quantities. That means any address fields in your instructions must be 32 bits long, unless you can find some other way to specify a reasonable range of addresses (at minimum, you must be able to get to the base addresses of all five regions in the MIPS memory model: user text, kernel text, user data, kernel data and the stack).
you must invent one new instruction for each architecture that manages an immediate value, either pushing it onto the top of the stack in the stack-based machine, or loading it into the accumulator of the accumulator-based machine.
There is no programming in this part of the lab assignment. Your task is to invent separate binary encodings for the stack-based and accumulator-based machines. Then you have to write your quadratic evaluators in binary (please use hexadecimal notation though!). Finally, record the number of bytes required for your programs (both code and data) and compare that to the size of the MIPS program (not including the trap handler, of course).
Part 4. Using the Binary Encodings in the Simulations
Now you have two simulators that take ASCII input, and you have also developed the rules for translating that ASCII input into binary. The next step would be to write the assemblers to automatically translate the source codes to binary, and rewrite the simulators to use (“execute”) the binary format. That actually wouldn’t be too tough, as we only have five instructions to deal with.
The more important thing that this brings us face-to-face with is the need to define the layout of the file containing an executable. This issue doesn’t become “visible” until you start grappling with the use of binary files containing executables, because ASCII files have lots of clues that make them easy to parse (.text and .data, for example). If you pick up something that is just a sequence of anonymous bits, though, how would you know what’s code and what’s data, for example?

The answer is that you need a defined format. Recent versions of Linux use something called ELF (executable and linking format). You can find lots of information about ELF using google. If you take a peek at the specification, the thing that should jump out at you is that there is something called an “ELF Header”, and this header is always the first thing in an ELF file. The header specifies the type of file and the target architecture, specifies the position in the file of other headers and sections, and gives the location of the first executable instruction in the file.

You are not required to use binaries in this course. It just seemed that the process of working through the lab and building the simulators brings us to a point where it’s easy to motivate something that isn’t often discussed (i.e. the need for defined formats for executables), and that as a result you might see quite easily what the issue is, and how important it is.

Lab Assignment 1 Submission
The following six files must be submit for this lab assignment.
quadratic_eval.s – It contains your debugged / commented code from Part 1.
stackSim – It should contain your simulator for the stack-based machine.
stackCode – It should contain your code for the quadratic evaluator, to run on your stack simulator.
accumSim – It should contain the simulator for the accumulator-based machine.
accumCode – It contains the code for the quadratic evaluator to run on it.
README – It should contain compilation instructions, and instructions on how to use your program. In addition, a discussion of any design issues you ran into while implementing this project and a description of how the program works (or any parts that don’t work) is also appropriate content for the README file.
You must hand in an electronic document with the answers for the questions in Part 3. This will consist of:
diagrams giving your binary format for each of the instructions on the two machines,
the assembled versions of both programs in a two-column format, with the source code in the left-hand column and the binary translation in the right-hand column (binary version in hexadecimal, please), and
the size in bytes of the binary versions of your stack and accumulator quadratic evaluation codes, and the size in bytes of the user code plus user data segments of the MIPS version of the quadratic evaluator.
E-mail the six files and the document for Part3 to chinmayk93@gmail.com and txie@mail.sdsu.edu with the subject line of “CS 572] Lab Assignment 1”. In your Email, please indicate your name and your student ID. Note that this assignment counts 10% towards your course grade.