

به نام خدا

## گزارشکار آز single\_cycle میکرورپرسوسور

۴۰۲۱۰۲۵۵۱

سید محمد حسن میرشفیعی

کد وریلاغ تکمیل شده همراه این فایل فرستاده شده است\*\*\*

در این آز باید بخش control unit از یک پردازنده میپس سینگل سایکل را پیاده سازی کنیم؛ قسمت های خواسته شده برای پیاده سازی در زیر آمده اند:

**R format:** add, sub, addu, subu, and, or, xor, nor,slt, sltu

**I format:** addi, addiu, slti, sltiu, andi, ori, xori, lui, beq, bne, lw, sw

دستورات R-format با op صفر در واقع با funct از instruction کنترل میشوند و بقیه دستورات با همان op کنترل می شوند (دلیل اینکه تمامی دستورات داخل فیلد op قرار نگرفته محدودیت بیت برای فرمت I و L است که چون در دستورات R-format چون از ۳۲ بیت خالی (function) داریم میتوان از آن برای کنترل این دسته از دستورات استفاده کرد). در ادامه به کد مربوطه می پردازیم:

خروجی های اصلی کنترل یونیت شامل موارد زیر هستند:

برای پرش در دستورات bne و beq PCSrc: •

Sign Extend فعال سازی SEn: •

نتخاب عملیات ALUOP: •

نتخاب بین عملوند دوم ALUSrc: •

نتخاب بین rd/rt برای نوشت RegDst: •

نتخاب منبع نوشتند در رجیستر MemtoReg: •

نوشتند در رجیستر RegWrite: •

نوشتند در حافظه MemWrite: •

```

67   6'b000000: begin // R-type
68     RegDst    = 1'b1;
69     ALUSrc    = 1'b0;
70     MemtoReg  = 1'b0;
71     RegWrite  = 1'b1;
72     MemWrite  = 1'b0;
73     SZEen     = 1'bx;
74     case(func)
75       6'b100000: AluOP = `ADD;
76       6'b100001: AluOP = `ADD;
77       6'b100010: AluOP = `SUB;
78       6'b100011: AluOP = `SUB;
79       6'b100100: AluOP = `AND;
80       6'b100101: AluOP = `OR;
81       6'b100110: AluOP = `XOR;
82       6'b100111: AluOP = `NOR;
83       6'b101010: AluOP = `SLT;
84       6'b101011: AluOP = `SLTU;
85     default : AluOP = 4'bxxxx;
86   endcase
87 end

```

| Control Signal | R-type (add, sub, and, or, slt...) |
|----------------|------------------------------------|
| RegDst         | 1                                  |
| ALUSrc         | 0                                  |
| MemtoReg       | 0                                  |
| RegWrite       | 1                                  |
| MemWrite       | 0                                  |
| SZEen          | X                                  |
| AluOP          | Derived                            |
| PCSrc          | 0                                  |

مطابق با جدول بالا باید برای این تایپ از دستورات سیگنال کنترلی را اساین کنیم؛ در قسمت AluOP بسته به دستور func خروجی لازم ست شده است. تنها چیزی که مقدار صریح برای آن ست نشده است که چون سیگنال از نوع reg نیست لذا اگر به درستی مقداردهی نشود ممکن است باعث جامپ به جای پرت شود که برای این مشکل راه حل را در ادامه خواهیم گفت.

از دستورات R-type تنها یکی را بررسی خواهیم کرد  $\text{lw}$  (load word).

```

89  6'b100011: begin // lw
90    RegDst    = 1'b0;
91    ALUSrc    = 1'b1;
92    MemtoReg  = 1'b1;
93    RegWrite  = 1'b1;
94    MemWrite  = 1'b0;
95    SZEen     = 1'b1;
96    AluOP     = `ADD;
97  end

```

| Control Signal | lw (Load Word) |
|----------------|----------------|
| RegDst         | 0              |
| ALUSrc         | 1              |
| MemtoReg       | 1              |
| RegWrite       | 1              |
| MemWrite       | 0              |
| SZEen          | 1              |
| AluOP          | ADD            |
| PCSrc          | 0              |



Figure 7.15 Critical path for  $\text{lw}$  instruction

رجیستر RegDst باید صفر باشد چون:

مقصود (رجیستر rt) در فرمت I-type در بیت‌های [۱۶:۰] قرار دارد، و نه [۱۵:۱۱] مانند دستورهای R-type.

رجیستر AluSrc باید یک باشد چون:

یکی از ورودی‌های ALU باید مقدار offset (ثابت) باشد و نه رجیستر دوم. بنابراین از Immediate استفاده می‌شود.

رجیستر MemtoReg باید یک باشد چون:

داده‌ای که از حافظه خوانده شده باید به عنوان خروجی به رجیستر بنویسیم، نه خروجی ALU.

رجیستر RegWrite باید یک باشد چون:

چون داده‌ای که از حافظه می‌خوانیم باید در رجیستر مقصود (rt) نوشته شود، باید این سیگنال فعال باشد.

رجیستر MemWrite باید صفر باشد چون:

این دستور چیزی در حافظه نمی‌نویسد، بلکه از آن می‌خواند. بنابراین نوشتمن در حافظه غیرضروری است.

رجیستر SLEN باید یک باشد چون:

در بسیاری از پیاده‌سازی‌ها برای استفاده از Immediate مقدار آن باید sign-extend شود؛ چون ممکن است منفی باشد.

وایر PCSrc باید صفر باشد چون:

چون این دستور جهشی (branch) نیست، نباید PC به مقدار دیگر تغییر یابد. مسیر عادی ادامه دارد.

رجیستر AluOP باید در حالت جمع ADD باشد چون:

برای محاسبه آدرس مؤثر (offset + rs) از جمع استفاده می‌شود ALU. باید عمل جمع انجام دهد.

```
54 // CONTROLLER COMES HERE
55 assign PCSrc = (op == 6'b000100 && AluZero) || (op == 6'b000101 && ~AluZero);
```

برای کنترل کردن بیت کنترلی PCSrc نیز از اساینمنت فوق استفاده می‌کنیم. که با دستورات beq, bne مقدار PCSrc را برای جامپ کردن سمت می‌کند.