# 計算機科学実験3A

ユーザーズマニュアル

チーム4: 竹田原俊介 Chung Mung Tim 2022-06-02

# 第1章 概要

この実験の目的は、SIMPLE (SIxteen-bit MicroProcessor for Laboratory Experiment) と呼ばれるコンピュータを設計することである。本レポートでは、SIMPLE の命令セットアーキテクチャの詳細と、各命令に対する私たちが設計した CPU 全体としての振る舞いを紹介する.

# 第2章 使用説明書

この章では、IPコア(ソフトコアプロセッサ)として提供することを想定し、わたしたちが設計したプロセッサを使用する上で必要十分な情報を記載する.一般に、ユーザーが実行したい16ビット命令をボードの主記憶に入力すると、命令を実行することができる.命令の書き方については、「第4章 命令セットアーキテクチャ」を参照してください.ここでは、ユーザーがボードに直接に触れる方法について説明する.

## reset ボタン

リセットボタンを押せば、プロセッサは命令の実行を最初からやり直す.

#### exec ボタン

exec ボタンには 2 つの機能がある. 停止状態にある時に押すと, 命令の実行を開始する. SIMPLE が以前に停止していた場合は, 停止したところから開始される. SIMPLE が実行状態にある時に押すと, その時点で実行中の命令を完了してかた停止する. reset ボタンと exec ボタンの場所については図 2.1 を参照してください.



図 2.1 reset ボタンと exec ボタン

## IN 命令

提供している命令の1つは入力 IN 命令で、ユーザーはボードから16桁の2進数を入力し、それをレジスタに保存することができる. FPGA ボードからの入力はディップスイッチを使って入力する. 16 ビットの入力を図2.2 のように桁を対応して入力することができる. ただし、ディップスイッチは負論理を使用している.



図 2.2 ボードからの入力

### OUT 命令

また、レジスタに格納された 16 ビットの値を二つボードに LED などとして出力する OUT 命令も提供されている. 私が設計した SIMPLE では、その二つの 16 ビットの値をそれぞれ 4 桁 16 進数として表 2.3 のように Power Medusa EC6S ボードの 8 桁 7SEG LED に映ることができる。また、命令中の 3 ビットフィールド(d)を使って、出力の場所を指定できるように設計した(図 2.4 参照).

表 2.3 7SEG LED で 16 進数の表示

| 16 進数 | 7SEG LED 表示 | 16 進数 | 7SEG LED 表示 |
|-------|-------------|-------|-------------|
| 0     | 8.          | 8     | 8.          |
| 1     |             | 9     | 8.          |
| 2     | 8.          | A     | 8.          |
| 3     | 8.          | В     | 8.          |
| 4     |             | С     |             |
| 5     | 8.          | D     | 8.          |
| 6     | 8.          | E     | 8.          |
| 7     |             | F     | 8.          |

## 図 2.4 出力 OUT 命令



# 第3章 性能と特長

プロセッサ性能は以下である:

サイクル数: 命令数の5倍周波数:(最大周波数)5MHz

• 面積(Logical elements): 1152/28,848(4%)

### 3.2 特長

- ALU とシフタを一体化すること
- プログラムカウンターと加算器を一体化すること
- 汎用レジスタは書き込みイネブラーが持つこと
- レジスタは出力イネブラーを持つこと
- 全てのレジスタは同じクロックを共用していること
- 即値オペランドが強化されていること
- 出力命令が強化されていること

## 第4章 命令セットアーキテクチャ

**SIMPLE** の命令は全て 1 語 16 ビットの固定長である. **SIMPLE** は以下 4 種類の命令形式がある. その形式と機能については, 4.1 節から 4.4 節で紹介する.

## 4.1 演算/入出力命令

| 11 | l  | Rs |    | Ro | ì |   | op3 |   | f |   | d |   |
|----|----|----|----|----|---|---|-----|---|---|---|---|---|
| 15 | 14 | 13 | 11 | 10 | 8 | 7 |     | 4 | 3 | 2 |   | 0 |

- *I*<sub>15:14</sub> (11) 操作コード
- *I*<sub>Is:11</sub> (Rs) ソースレジスタ番号
- *I*<sub>108</sub> (Rd) デスティネーションレジスタ番号
- *I*<sub>7:4</sub> (op3) 操作コード (0000~1111)
- *I*₃(f) オペランドフラグ
- *I*<sub>20</sub> (d) シフト桁数

#### ただし、

- fが0の場合, r[Rd]とr[Rs]をオペランドとして使用して計算する.
- f が 1 の場合, r[Rd]と sign\_ext(conc(Rs,d))をオペランドとして使って計算する. ここで, conc(Rs,d)は Rs フィールドの 3 ビットと d フィルドの 3 ビット を結合した値を表す. sign\_ext(conc(Rs,d))のようにして得られるオペランドを sw(f, Rs, d)と表記する

### 4.1.1 演算命令

SIMPLE の演算命令を表 2.1.1 に示す. 演算命令では結果に基づく条件コード S, Z, C, V が設定される.

1. 算術演算

レジスタ Rd と Rs の加算(ADD)または減算(SUB)の結果を Rd に格納し、条件コードを設定する.

2. 論理演算

レジスタ Rd  $\geq Rs$  の,ビットごとの論理積(AND),論理和(OR),または排他的論理和(XOR)の結果を Rd に格納し,条件コードを設定する.

#### 3. 比較演算 (CMP)

レジスタ Rd から Rs を減算し、結果に基づく条件コード設定のみを行う.

#### 4. 移動演算 (MOV)

レジスタ Rd に Rs の値を単に格納し、Rd の値に基づき条件コードを設定する.

### 5. シフト演算

レジスタ Rd の値を、以下のようにシフトした値を Rd に格納し、条件コードを設定する。

## • SLL (shift left logical)

左論理シフト. 左シフト後, 空いた部分に0を入れる

#### • SLR (shift left rotate)

左循環シフト. 左シフト後, 空いた部分にシフトアウトされたビット列を 入れる.

## • SRL (shift right logical)

右論理シフト・右シフト後,空いた部分に0を入れる.

## • SRA (shift right arithmetic)

右算術シフト・右シフト後,空いた部分に符号ビットの値を入れる.

シフト桁数は即値d(0~15)である.

表 4.1.1: SIMPLE の演算命令

| mnemonic  |        | op3  | function                            |
|-----------|--------|------|-------------------------------------|
| ADD       | Rd, Rs | 0000 | r[Rd] = r[Rd] + sw(f, Rs, d)        |
| SUB       | Rd, Rs | 0001 | r[Rd] = r[Rd] - sw(f, Rs, d)        |
| AND       | Rd, Rs | 0010 | r[Rd] = r[Rd] & sw(f, Rs, d)        |
| OR        | Rd, Rs | 0011 | $r[Rd] = r[Rd] \mid sw(f, Rs, d)$   |
| XOR       | Rd, Rs | 0100 | $r[Rd] = r[Rd] \wedge sw(f, Rs, d)$ |
| CMP       | Rd, Rs | 0101 | r[Rd] - $sw(f, Rs, d)$              |
| MOV       | Rd, Rs | 0110 | r[Rd] = r[Rs]                       |
| (reserved |        | 0111 | /                                   |

| SLL | Rd, d | 1000 | r[Rd] = shift_left_logical ( r[Rd], d )     |
|-----|-------|------|---------------------------------------------|
| SLR | Rd, d | 1001 | r[Rd] = shift_left_rotate ( r[Rd], d )      |
| SRL | Rd, d | 1010 | r[Rd] = shift_right_logical ( r[Rd], d )    |
| SRA | Rd, d | 1011 | r[Rd] = shift_right_arithmetic ( r[Rd], d ) |

### 4.1.2 入出力・制御命令

• IN (input) :機器から入力した値をレジスタ Rd に格納する.

• OUT (output) : レジスタ Rs の値を機器に出力する.

• HLT (HLT) : SIMPLE を停止させる.

表 4.1.3: SIMPLE の入出力・制御命令

| mnemo      | nic | op3  | function                   |
|------------|-----|------|----------------------------|
| IN         | Rd  | 1100 | r[Rd] = input              |
| OUT        | Rs  | 1101 | output = (r[Rs], r[Rd], d) |
| (reserved) |     | 1110 | /                          |
| HLT        | /   | 1111 | halt()                     |

第2章で説明したように、dフィールドの値を指定することによって出力の場所を変わることができる。また、RsとRdで指定されたレジスタの値を同時に出力することができる。Rdが指定されていない時は0000を出力する。

# <u>4.2 ロード/ストア命令</u>

15 14 13 11 10 8 7 4 3 0

op1 Ra Rb d

- I<sub>15:14</sub> (op1) 操作コード (00/01)
- *I*<sub>Isti</sub> (Ra) ソース・デスティネーションのレジスタ番号
- *I*<sub>10:8</sub> (Rb) ベースレジスタ番号

## • I<sub>7:0</sub> (d) 変位

SIMPLE のロード命令とストア命令の機能を表 2に示す.ソース/デスティネーションは,フィールド Ra で指定されたレジスタ Ra である.また実行アドレスはベース・レジスタ・アドレス指定により,フィールド Rb で指定されたレジスタ Rb と,フィールド d を符号拡張した  $sign_ext(d)$ を加算して求める.

LD (load) : 主記憶アドレス(r[Rb] + sign\_ext(d))にある 16 ビットの値を
 Ra にロードする

• ST (store) : Ra にある 16 ビットの値を主記憶アドレス(r[Rb] + sign ext(d))のところに書き込む・

| _        | <u>17.7.2</u> | DIIVII LI | 7 42 E 1 24 1 7 HI II             |
|----------|---------------|-----------|-----------------------------------|
| mnemonic | ;             | op1       | function                          |
| LD       | Ra, d(Rb)     | 00        | $r[Ra] = *(r[Rb] + sign\_ext(d))$ |
| ST       | Ra, d(Rb)     | 01        | $*(r[Rb] + sign\_ext(d)) = r[Ra]$ |

表 4.2 SIMPLE のロード・ストア命令

#### 4.3 即値ロード/無条件分岐命令

| 10 | 0  | op2 |    | Rb |   |   | d   |   |  |
|----|----|-----|----|----|---|---|-----|---|--|
| 15 | 14 | 13  | 11 | 10 | 8 | 7 | 4 3 | 0 |  |

- I<sub>15:14</sub> (10) 操作コード
- I<sub>13:11</sub> (op2) 操作コード (000~110)
- $I_{108}$  (Rb) ソース・デスティネーション・ベースのレジスタ番号
- I<sub>7:0</sub> (d) 即値・変位

• LI(load immediate) : 即値 sign\_ext(d)をレジスタ Rb に格納する

• B (branch) : d を符号拡張した値 sign\_ext(d)を変位として, PC

相対アドレス指定による分岐を行う

表 4.3 SIMPLE の即値ロード・無条件分岐命令

| mnemonic   |       | op1 | function                    |  |
|------------|-------|-----|-----------------------------|--|
| LI         | Rb, d | 000 | $r[Rb] = sign\_ext(d)$      |  |
| (reserved) |       | 001 | /                           |  |
| (reserved) |       | 010 | /                           |  |
| (reserved) |       | 011 | /                           |  |
| В          | d     | 100 | $PC = PC + 1 + sign_ext(d)$ |  |
| (reserved) |       | 101 | /                           |  |
| (reserved) |       | 110 | /                           |  |
| (条件分岐命令)   |       | 111 | 表 4.4 参照                    |  |

## 4.4 条件分岐命令形式

| 15 | 0 | 111 | 10  |   | <br> |  |
|----|---|-----|-----|---|------|--|
|    | U | 111 | con | a | a    |  |

- 1. I15:14 (10) 操作コード
- 2. I1311 (111) 操作コード
- 3. *I*<sub>10:8</sub> (cond) 分岐条件
- 4. I<sub>7:0</sub> (d) 変位

SIMPLE の条件分岐命令は以下の表に示すように、フィールド cond で定められる分岐条件が成り立てば PC 相対アドレスによる分岐を行い、成り立たなければ単に次の命令に移行する。各命令の分岐条件は以下の通り:

• BE (branch on equal to) : Z が 1

• BLT (branch on less than) : S^V が 1

• BLE (branch on less than or equal to) : Zまたは (S^V) が1

• BNE (branch on not equal to) : Z が 0

表 4.4 SIMPLE の条件分岐命令

| mnemonic   |   | cond | function                                                  |
|------------|---|------|-----------------------------------------------------------|
| BE         | d | 000  | if (Z) $PC = PC + 1 + sign_ext(d)$                        |
| BLT        | d | 001  | if $(S \wedge V) PC = PC + 1 + sign_ext(d)$               |
| BLE        | d | 010  | if $(Z \parallel (S \land V)) PC = PC + 1 + sign\_ext(d)$ |
| BNE        | d | 011  | if $(!Z) PC = PC + 1 + sign_ext(d)$                       |
| (reserved) |   | 100  | /                                                         |
| (reserved) |   | 101  | /                                                         |
| (reserved) |   | 110  | /                                                         |
| (reserved) |   | 110  | /                                                         |

## 4.5 ボードから直接入力される命令

reset (リセット信号)

• FPGA ボード上のプッシュスイッチを用いて、スイッチを押すと1が、離すと0が供給されるようにする. reset が1になると、PC 等をクリアし、初期 状態に移行する.

### exec (起動/停止信号)

- FPGA ボード上のプッシュスイッチを用いて、スイッチを押すと1が、離すと0が供給されるようにする.
- SIMPLE が停止状態にある時に exec が 0 から 1 に変化すると, SIMPLE は命令の実行を開始する.
- SIMPLE が実行状態にある時に exec が 0 から 1 に変化すると、その時点で実行中の命令が完了してから停止する.

# 第5章 構造と動作

この章では、さまざまな命令を受け取ったときの CPU の振る舞いについて説明する。同じような動作をする命令は、繰り返しを避けるためにまとめてある。第5章の説明については、図 5.1 と図 5.2 を参照してください。





#### ADD, SUB, AND, OR, XOR

- 1. PC (Program Counter) が保持するアドレスの命令を命令主記憶 inst\_memory からフェッチし、IR (Instruction Register) に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が ALU 制御部に必要な演算に対応するコードを送る.
  - 1. 命令の中のフラグ f が O ならば、マルチプレクサ MUX 3、MUX 2 と MUX 9 のセレクタ信号として O が制御部から与えられ、 IR が保持する命令の Rs フィールドと Rd フィールドで指定される汎用レジスタの アドレスにある値を読み出し、それぞれレジスタ AR と BR に格納する.
  - 2. フラグ f が 1 ならば、conc(Rs,d) (つまり Rs フィールドの 3 ビットと d フィルドの 3 ビットを結合した値) を符号拡張し、MUX 9 のセレクタ 信号が 1 にし、符号拡張した conc(Rs,d)が AR に格納する. 一方、MUX 3 のセレクタ信号が 0 で、Rd フィールドの値が BR に格納する.
- 2. AR と BR にある値が ALU の入力になる. ALU により、命令が定める演算を行い、その結果をレジスタ DR (Data Register) に格納する. 条件コード S, Z, C. V をセットする.
- 3. DR にある値を命令の Rd フィールドに指定される汎用レジスタに書き込む. (MUX4のセレクタ信号は0で、MUX8と MUX5のセレクタ信号は1である.)

#### CMP 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が ALU 制御部 に必要な演算に対応するコードを送る.
  - 1. 命令の中のフラグ f が 0 ならば、マルチプレクサ MUX 3、MUX 2 と MUX 9 のセレクタ信号として 0 が制御部から与えられ、 IR が保持する命令の Rs フィールドと Rd フィールドで指定される汎用レジスタの アドレスにある値を読み出し、それぞれレジスタ AR と BR に格納する.
  - 2. フラグ f が 1 ならば、conc(Rs,d)(つまり Rs フィールドの 3 ビットと d フィルドの 3 ビットを結合した値)を符号拡張し、MUX 9 のセレクタ

信号が1にし、符号拡張した conc(Rs,d)がAR に格納する.一方、MUX3のセレクタ信号が0で、Rd フィールドの値がBR に格納する.

2. **AR** と **BR** にある値が **ALU** の入力になる. **ALU** により、命令が定める演算を行うが、結果を出力しない、条件コード **S**. **Z**. **C**. **V** をセットする.

## MOV 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が ALU 制御部 に必要な演算に対応するコードを送る.
- 2. マルチプレクサ MUX 3, MUX 2 と MUX 9 のセレクタ信号として 0 が制御部 から与えられ, IR が保持する命令の Rs フィールドと Rd フィールドで指定 される汎用レジスタのアドレスにある値を読み出し, それぞれレジスタ AR と BR に格納する.
- 4. DR にある値を命令の Rd フィールドに指定される汎用レジスタに書き込む.  $(MUX4 \, のセレクタ信号は \, 0 \, で, \, MUX8 \, と \, MUX5 \, のセレクタ信号は \, 1 \, である. )$

## SLL, SLR, SRL, SRA 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が ALU 制御部 に必要な演算に対応するコードを送る.
- 2. IR が保持する命令の、Rd フィールドで指定される汎用レジスタのアドレス にある値を読み出し、レジスタ BR に格納する(MUX3のセレクタ信号は 0 である).
- 3. MUX2のセレクタ信号は1でMUX9のセレクタ信号は0で、符号拡張した 即値 d が AR に格納する.
- 4. AR と BR にある値が ALU の入力になる. ALU により、命令が定める演算を行い、その結果をレジスタ DR に格納する. 条件コード S, Z, C, V をセットする.

5. DR にある値を命令の Rd フィールドに指定される汎用レジスタに書き込む. (MUX4のセレクタ信号は0で、MUX8と MUX5のセレクタ信号は1である. )

## IN 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が制御信号を発信する.
- 2. 外部入力(ディップスイッチ)の値が MDR に格納される. (ここで MUX 7 のセレクタは 1 になる. )
- 3. MDR にある値を命令の Rd フィールドに指定される汎用レジスタに書き込む. (MUX4のセレクタ信号は1で、MUX8と MUX5のセレクタ信号は1である.)

## OUT 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が制御信号を発信する.
- 2. フィールド Rs と Rd で指定されたレジスタにある値が seven というデコーダ の部品に通して、7 SEG LED などの外部出力へ表示される. (ここで MUX10 のセレクタは1になる.) Rd が指定されていない時は4桁の 0000 が 表示される.

#### HLT 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える.
- 2. IR にある命令が制御部に届き、制御部が全ての部品を停止させる.

#### LD 命令

1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が ALU 制御部に必要な演算に対応するコード(加算 ADD)を送る.

- 2. IR が保持する命令の、Rb フィールドで指定される汎用レジスタのアドレス にある値を読み出し、レジスタ BR に格納する. (ここで MUX7 のセレクタ は 0 になる.)
- 3. MUX2のセレクタ信号は1で MUX9のセレクタ信号が0で、符号拡張した 即値dがARに格納する.
- 4. **AR** と **BR** にある値が **ALU** の入力になる. **ALU** により, 加算を行い, その結果をレジスタ **DR** に格納する.
- 5. DR にある値をアドレスとして、PC に j\_addr(jump address)として入力し、データ主記憶をアクセスする. 読み出した値をレジスタ MDR(Memory Data Register)に格納する. (ここで MUX 7 のセレクタは 0 になる.)
- 6. MUX 4 と MUX 8 のセレクタが 1 になり、MDR にある値を命令の Ra フィールドに指定される汎用レジスタに書き込む.(MUX 5 のセレクタは 0 になる.)

## ST 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が ALU 制御部 に必要な演算に対応するコード(加算 ADD)を送る.
- 2. IR が保持する命令の、Rb フィールドで指定される汎用レジスタのアドレス にある値を読み出し、レジスタ BR に格納する. (ここで MUX3 のセレクタ は 0 になる.)
- 3. ここで MUX2のセレクタは1になり、MUX9のセレクタは0になり、符号拡張した即値 d が AR に格納する.
- 4. AR と BR にある値が ALU の入力になる. ALU により, 加算を行い, その結果をレジスタ DR に格納する.
- 5. DR にある値をアドレスとしてデータ主記憶をアクセスする. AR にある値が そのアドレスにストアする.

#### LI 命令

1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える.

2. MUX8 のセレクタ信号が 0 になり、書き込みデータとして符号拡張した d を 汎用レジスタに書き込む、書き込みアドレスを決めるために、MUX5 のセレクタ信号が 1 になる.

## B, BE, BLT, BLE, BNE 命令

- 1. PC が保持するアドレスの命令を主記憶からフェッチし、IR に格納すると共に、PC に 1 を加える. IR にある命令が制御部に届き、制御部が ALU 制御部 に必要な演算に対応するコード(加算 ADD)を送る.
- 2. MUX3 のセレクタは1で、PC+1 の値が BR に格納する。MUX2 のセレクタは1で MUX9 のセレクタは0になり、符号拡張した d の値が AR に格納する.
- 3. **AR** と **BR** にある値が **ALU** の入力になる. **ALU** のもう一つの入力になる. **ALU** が加算を行い、その結果をレジスタ **DR** に格納する.
- 4. DR にある値を分岐先アドレスとして PC に直接に書き込む.

### reset(リセット信号)

- 1. 制御部へ1ビットの reset 信号が届く.
- 2. 制御部で SIMPLE を初期状態にさせる.

## exec(起動/停止信号)

- 1. 制御部へ1ビットの exec 信号が届く.
- 2. 制御部でフラグを使って、その時点で実行中の命令が完了してから SIMPLE を停止する.