This document describes the processor simulators that accompany the presentation of the Y86-64 processor architectures in Chapter 4 of Computer Systems: A Programmer’s Perspective, Third Edition.
The original (official) simulator, written in C has difficulty adapting to too many modifications on the seq, seq+ and pipe HCL, leading to a limited range of lab assignments. This project aims to provide a more flexible and extensible simulator for the Y86-64 processor, and is employed in Peking U's ICS: Introduction to Computer System in 2024.
This project is written in Rust, so you'd have your Rust toolchain installed. If you haven't, please execute the following command to install rustup:
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
You can verify the installation by executing the command rustup
.
Install and the Rust toolchain by executing the following command (by the time of writing, the latest stable version is 1.81):
rustup install 1.81
rustup default 1.81
Simply execute cargo build
to build all the binaries in the project. After running this command, a folder named target
will be created to store the output binaries and other intermediate files. The output executables are
target/debug/yas
: Y86-64 Assemblertarget/debug/yis
: Y86-64 ISA Simulatortarget/debug/ysim
: Y86-64 Pipline Simulatortarget/debug/ydb
: Y86-64 Debugger Server
To build the release version, execute cargo build --release
. The release version is optimized for performance. The released version executables locate in the target/release
folder (target/release/{yas,yis,ysim,ydb}
).
To assemble a Y86-64 assembly file, execute the following command:
./target/debug/yas [input_file].ys
The default output filename is [input_file].yo
. You can specify the output filename by adding the -o
option. For example, given the following y86 assembly file swap.ys
:
# Swap nums if the former one >= the latter one
.pos 0
irmovq stack, %rsp
irmovq nums, %rdi
mrmovq (%rdi), %rdx
mrmovq 8(%rdi), %rcx
rrmovq %rdx, %rbp
subq %rcx, %rbp # $rbp <= $rcx ?
# if so, then do not swap
jle done
rmmovq %rdx, 8(%rdi)
rmmovq %rcx, (%rdi)
done:
halt
nop
nop
nop
.align 8
nums:
.quad 0xcba
.quad 0xbca
.pos 0x200
stack:
By running ./target/debug/yas swap.ys
, the assembler will generate a binary file swap.yo
:
| # Swap nums if the former one >= the latter one
0x0000: | .pos 0
0x0000: 30f40002000000000000 | irmovq stack, %rsp
|
0x000a: 30f75000000000000000 | irmovq nums, %rdi
0x0014: 50270000000000000000 | mrmovq (%rdi), %rdx
0x001e: 50170800000000000000 | mrmovq 8(%rdi), %rcx
0x0028: 2025 | rrmovq %rdx, %rbp
0x002a: 6115 | subq %rcx, %rbp # $rbp <= $rcx ?
| # if so, then do not swap
0x002c: 714900000000000000 | jle done
0x0035: 40270800000000000000 | rmmovq %rdx, 8(%rdi)
0x003f: 40170000000000000000 | rmmovq %rcx, (%rdi)
0x0049: | done:
0x0049: 00 | halt
0x004a: 10 | nop
0x004b: 10 | nop
0x004c: 10 | nop
|
0x0050: | .align 8
0x0050: | nums:
0x0050: ba0c000000000000 | .quad 0xcba
0x0058: ca0b000000000000 | .quad 0xbca
|
0x0200: | .pos 0x200
0x0200: | stack:
|
To simulate a Y86-64 assembly file w.r.t. the Y86 ISA specification, you can execute the following command:
./target/debug/yis [input_file].yo
For example, by running ./target/debug/yis swap.yo
, the simulator will print the following information:
0x0000 icode: 0x3 (IRMOVQ), ifun: 0, rA: RNONE, rB: RSP, V: 0x200
0x000a icode: 0x3 (IRMOVQ), ifun: 0, rA: RNONE, rB: RDI, V: 0x50
0x0014 icode: 0x5 (MRMOVQ), ifun: 0, rA: RDX, rB: RDI
0x001e icode: 0x5 (MRMOVQ), ifun: 0, rA: RCX, rB: RDI
0x0028 icode: 0x2 (CMOVX), ifun: 0, rA: RDX, rB: RBP
0x002a icode: 0x6 (OPQ), ifun: 1, rA: RCX, rB: RBP
0x002c icode: 0x7 (JX), ifun: 1, V: 0x49
0x0035 icode: 0x4 (RMMOVQ), ifun: 0, rA: RDX, rB: RDI
0x003f icode: 0x4 (RMMOVQ), ifun: 0, rA: RCX, rB: RDI
0x0049 icode: 0x0 (HALT), ifun: 0
total instructions: 10
ax 0000000000000000 bx 0000000000000000 cx 0000000000000bca dx 0000000000000cba
si 0000000000000000 di 0000000000000050 sp 0000000000000200 bp 00000000000000f0
0x0050: ba0c000000000000 -> ca0b000000000000
0x0058: ca0b000000000000 -> ba0c000000000000
To simulate a Y86-64 assembly file over the default architecture (seq_std
), execute the following command:
./target/debug/ysim [input_file].ys
This will print the state of the processor at each cycle to the standard output. If you want to read tht output from start to end, you can pipe a less
command to the output (To quit less
, press q
):
./target/debug/ysim [input_file].ys | less
To print more information you can use the -v
option, which will display the value of each signal in each stage of the cycle:
./target/debug/ysim [input_file].ys -v
We provide different architectures for the simulator. To view available architectures, you can run
./target/debug/ysim --help
To specify an architecture, you can use the --arch
option. For example, to run the simulator with the seq_plus_std
architecture, you can run:
./target/debug/ysim [input_file].ys --arch seq_plus_std
You can also inspect an architecture via -I
option:
./target/debug/ysim --arch seq_plus_std -I
Its output will be like:
propagate order:
lv.1: pc
lv.2: imem align icode ifun instr_valid mem_read mem_write need_regids need_valC set_cc alufun
lv.3: ialign pc_inc dstM srcA srcB valC valP
lv.4: reg_read aluA mem_data aluB
lv.5: alu valE mem_addr
lv.6: reg_cc dmem cc valM stat prog_term
lv.7: cond cnd dstE
lv.8: reg_write
dependency graph visualization is generated at: seq_plus_std_dependency_graph.html
Here an HTML file is generated to visualize the dependency graph of the architecture. You can open the HTML file in a browser to view the dependency graph. In the graph, blue blocks are hardware components, and red blocks are signals. You can drag each block to change its position.
To provide a friendly coding experience, we develop a debugger server for the Y86 assembly language. This debugger server is used along with the y86-debugger
VSCode extension.
First you should install y86-debugger
extension from assets/y86-debugger-0.2.0.vsix
(choose "Install from VSIX..."). To install the extension on SSH remote servers, you can first install it on your local machine, then open a remote host VSCode window and choose "Install in SSH:..." to install the extension on the remote server.
Then you should start the debugger server:
./target/debug/ydb -p 2345 # -p specifies the port number
After that you can start debugging in VSCode. You can set breakpoints, step through the code, and inspect the registers and memory. Click the debug icon at the right side of the menu bar to start debugging.
By default, your assembly file is simulated with the seq_std
architecture. If you want to change the architecture, you may use the --arch
option:
# specify another architecture to debug
# both builtin and extra architectures are supported
./target/debug/ydb -p 2345 --arch seq_plus_std
Refer to y86-debugger for more information.
Please refer to this attachment for detailed description of the HCL-rs syntax.