

دانشگاه صنعتی شریف دانشکده مهندسی کامپیوتر

عنوان:

# طراحی و پیادهسازی پروتکل ارتباطی سریال شبه UART برای پردازندهی Logisim

Design and Implementation of a UART-Inspired Serial Communication Protocol for a MIPS Processor in Logisim Environment

نگارش

نرگس کاری، ثنا نیرومند، یاسمین رجبی ریسه و حسنا شاه حیدری

استاد درس

دكتر حسين اسدى

دستیار آموزشی مسئول پروژه مهدی بهرامیان

تابستان ۲۰۴۴

چکیده:در این پروژه، یک پروتکل ارتباطی سریال ساده با الهام از پروتکل UART طراحی و پیادهسازی شده است. این پیادهسازی در بستر یک پردازنده MIPS تکچرخهای و با استفاده از محیط شبیهسازی Logisim انجام گرفته است. هدف اصلی، ایجاد پلی میان دنیای سختافزار و نرمافزار بوده است، به گونهای که پردازنده بتواند با دستگاههای ورودی/خروجی نظیر صفحه کلید، نمایشگر متنی TTY و هاLED ارتباط برقرار کند. برای تحقق این هدف، مجموعهای از دستورهای سفارشی ( Instructions Custom ) به فرمت I-type طراحی و به معماری پردازنده افزوده شده شده اند تا امکان ارسال و دریافت داده به/از دستگاههای جانبی فراهم شود. پروتکل طراحی شده شامل بستههای دادهای با ساختار مشخص (شامل بیت شروع، آدرس مقصد و داده) است که از طریق رجیسترهای شیفت دهنده به صورت سریال و از طریق bus تبادل می شوند. در طراحی این پروتکل از مفهوم بیتهای آغاز و پایان که در UART وجود دارد الهام گرفته شده تا انتقال داده به صورت قابل اطمینان انجام شود. کارایی سیستم طراحی شده با چهار برنامه ی تستی شامل کنترل به صورت قابل اطمینان انجام شود. کارایی سیستم طراحی شده با چهار برنامه ی تستی شامل کنترل به صورت قابل اطمینان انجام شود. کارایی سیستم طراحی شده با چهار برنامه ی تستی شامل کنترل عملکرد صحیح آن در تبادل داده ها تأیید شده است.

واژههای کلیدی: پردازنده ،MIPS ارتباط سریال، ،UART رابط ورودی/خروجی، ،MIPS رجیستر شیفت دهنده، سیستمهای نهفته، معماری تکچرخه، دستور سفارشی، صفحه نمایش ،TTY ورودی کیبورد

#### ۱ مقدمه

#### ۱-۱ تعریف مساله

در پردازنده ها، برای برقراری ارتباط با دنیای بیرون نیاز به استفاده از رابطهای ورودی/خروجی (I/O) وجود دارد. این رابطها امکان تبادل داده بین پردازنده و دستگاه های جانبی مانند صفحه کلید، نمایشگر، حافظه خارجی و غیره را فراهم می کنند. یکی از رایج ترین روش ها برای انتقال داده، استفاده از پروتکل های سریال مانند UART است. در این پروژه، با الهام از پروتکل یک پروتکل ساده و سفارشی طراحی شده است که بر بستر معماری MIPS پیاده سازی شده و هدف آن ایجاد یک بستر ارتباطی بین رجیسترهای داخلی پردازنده و دستگاه های جانبی (مانند LED و صفحه نمایش TTY) است.

## ۱-۲ اهمیت موضوع

درک عمیق از نحوه طراحی و پیادهسازی پروتکلهای ارتباطی، یکی از مباحث کلیدی در معماری کامپیوتر و طراحی سیستمهای نهفته است. این پروژه با فراهم آوردن بستری برای تعامل میان سخت افزار و نرمافزار، زمینه مناسبی برای درک بهتر مفاهیمی مانند طراحی پردازنده، نحوه انتقال داده، طراحی رجیسترهای کنترلی و کاربرد پروتکلهای سریال فراهم می سازد. پیاده سازی این سیستم در محیطهایی مانند Logisim همچنین دانش عملی دانشجو را در زمینه طراحی دیجیتال افزایش می دهد.

## ۱-۳ ادبیات موضوع

پروتکل UART یکی از پرکاربردترین پروتکلهای ارتباط سریال است که در بسیاری از سیستمهای نهفته، دستگاههای الکترونیکی و پردازندهها برای ارسال و دریافت داده استفاده می شود. در ،UART نهفته، دستگاه الکترونیکی و پردازندهها برای ارسال و دریافت ( RX و TX ) بین دو دستگاه داده ها به صورت سریال و با استفاده از رجیسترهای انتقال و دریافت ( RX و RX ) بین دو دستگاه تبادل می شوند. پروژه حاضر با ساده سازی و شبیه سازی این پروتکل در محیط آموزشی و با استفاده از یک پردازنده ساده تک چرخه ای ( MIPS single-cycle ) گامی در جهت آموزش عملی مفاهیم پایه ای معماری و ارتباطات برداشته است.

## ۱-۴ اهداف پروژه

اهداف اصلی این پروژه عبارتاند از:

طراحی و پیادهسازی یک ماژول ارتباطی سریال ساده بر پایه پروتکل UART

شبیه سازی پروتکل پیاده سازی شده در محیط Logisim

برقراری ارتباط بین رجیسترهای داخلی پردازنده و دستگاههای جانبی مانند LED و TTY افزایش توانمندی دانشجو در درک مفاهیم پایهای معماری پردازنده و ارتباط با دنیای خارج

# ٢ مفاهيم اوليه

## ۱-۲ پردازنده MIPS

پردازنده مورد استفاده در این پروژه از نوع single-cycle است و قابلیت اجرای دستورات پایه را دارد که در تمرینات عملی درس پیادهسازی شده و تست شده بود. برای ارتباط با دستگاههای جانبی، دستورات I/O اختصاصی به مانند فرمت I-type اضافه شدهاند. اکنون به اختصار به شرح دستوراتی که در پردازنده پشتیبانی میشوند میپردازیم:

| Instruction | Туре   | Opcode | Funct  |      |                                     | Notes                                             |
|-------------|--------|--------|--------|------|-------------------------------------|---------------------------------------------------|
| add         | R-Type | 000000 | 100    | 0000 | rd =                                | rs + rt                                           |
| addi        | I-Type | 001000 | _      |      | rt =                                | rs + imm (sign-extended)                          |
| sub         | R-Type | 000000 | 100    | 0010 | rd =                                | rs - rt                                           |
| or          | R-Type | 000000 | 100    | 101  | rd =                                | rs   rt (bitwise OR)                              |
| and         | R-Type | 000000 | 100    | 100  | rd =                                | rs & rt (bitwise AND)                             |
| xor         | R-Type | 000000 | 100    | 110  | rd =                                | rs ^ rt (bitwise XOR)                             |
| sll         | R-Type | 000000 | 000    | 100  | rd =                                | rs << rt (logical shift left)                     |
| srl         | R-Type | 000000 | 000    | 110  | rd = rs >> rt (logical shift right) |                                                   |
| sra         | R-Type | 000000 | 000    | 111  | rd =                                | rs >>> rt (arithmetic shift right, sign-extended) |
| Instruction | Туре   | Орс    | ode    | Fu   | nct                                 | Notes                                             |
| div         | R-Typ  | e 0000 | 900    | 011  | 1010                                | hi = rs % rt; lo = rs / rt                        |
| sll         | R-Typ  | e 0000 | 900    | 000  | 9000                                | rd = rt << shamt (logical shift left)             |
| mfhi        | R-Typ  | e 0000 | 900    | 016  | 9000                                | rd = hi                                           |
| mflo        | R-Typ  | e 0000 | 000000 |      | 9010                                | rd = lo                                           |
| SW          | І-Тур  | e 1010 | 101011 |      |                                     | *(int*)(offset+rs)=rt (sign-extended)             |
| lw          | І-Тур  | e 1000 | 100011 |      |                                     | rt=*(int*)(offset+rs) (sign-extended)             |
| bne         | І-Тур  | e 0000 | 000101 |      |                                     | if(rs!=rt) pc+=offset (sign-extended)             |
| slti        | І-Тур  | e 0010 | 001010 |      |                                     | rt=rs <imm (sign-extended)<="" td=""></imm>       |
| jmp         | І-Тур  | e 0000 | 910    |      |                                     | pc=target                                         |

| Instruction | Туре   | Opcode | Funct  | Notes                                 |
|-------------|--------|--------|--------|---------------------------------------|
| mul         | R-Type | 011100 | 000010 | rd = rs * rt (signed-multiply)        |
| jr          | R-Type | 000000 | 001000 | pc = rs                               |
| jal         | J-Type | 000011 | _      | r31(ra) = pc+1 & pc=addr              |
| beq         | I-Type | 000100 | _      | if(rs==rt) pc+=offset (sign-extended) |

#### I/O دستورات ویژه I/O

- led \$rt •
- tty \$rt •
- kb \$rt •

#### I/O فرمت دستورات T-1-T

- Opcode: مقدار 111111 برای فعالسازی سیگنال isIO
- آدرس دستگاه: ۵ بیت بعدی که مقصد دستور را مشخص میکند:
  - 0 تا 6: آدرس هاLED
  - 7: آدرس TTY (صفحه نمایش)
    - **-** 8: آدرس کیبورد
- 11: علاوه بر این دستورات، با آدرس ۱۱۱۱ میتوان عدد موجود در رجیستر مشخص شده را به صورت کامل با همهی LED ها نمایش داد.
- رجیستر داده: ۵ بیت بعدی مشخصکننده رجیستری است که داده در آن ذخیره یا از آن خوانده می شود. در لامپهای LED علاوه بر آدرس لامپ نیاز به داده ی صفر یا یک در جایگاه i ام عدد داریم که بگوید لامپ i ام روشن یا خاموش شود. در صفحه ی نمایشگر TTY نیاز به کد ASCII هفت بیتی یک کاراکتر داریم تا روی صفحه ی نمایشگر چاپ شود. برای این منظور یک رجیستر باید مشخص کنیم در دستور ورودی تا این داده در آن ذخیره شود. در نتیجه شماره ی این رجیستر را در پنج بیت بعدی ذخیره میکنیم.

۲-۱-۳ واحد کنترلی

در اینجا دستور I/O تشخیص داده شده و سیگنال آن در واحد کنترلی تولید می شود.



#### **۲-۱-۲** واحد حافظه

لازم است که تغییرات اندکی به فرمت حافظه در پروژه ی خود بدهیم تا بتوانیم دستورات خودمان را در حافظه ذخیره کنیم و به عنوان ورودی به برنامه بدهیم. در تمرینهای عملی از فرمت JTAG استفاده شده بود تا حافظه ی دستور و داده توسط برنامه ی judge پر شوند. از آنجایی که در این پروژه دستورات را دستی وارد می کنیم نیازی به این ارتباطات نیست و در حافظه ی D-mem کافی است

فقط این اتصالات را مطابق تصویر حذف کنیم.



به جای I-mem هم برای اینکه خودمان در برنامه دستورات را دستی وارد کنیم یک ROM میگذاریم که از بلوکهای ۳۲ بیتی تشکیل شده و هنگام سیموله کردن برنامه درون آن را با برنامهی مدنظر پر میکنیم.



## Y-۲ پروتکل UART

بعد از این تغییرات نوبت به رد و بدل کردن اطلاعات با I/O و انجام عملیات مدنظر روی آنها میباشد. به این صورت که طبق خواستهی صورت پروژه نمی توانیم مستقیم داده ی خوانده شده از رجیستر یا حتی آدرس دستگاه جانبی مورد هدف را با یک سیم به سمت I/O بفرستیم و بایستی یک پروتکل سریال برای انتقال اطلاعات در نظر بگیریم. برای این منظور، ما یک پروتکل شبه UART پیاده سازی کرده ایم. پروتکل T مطابق تصویر شامل دو سیم برای تبادل داده می باشد که با استفاده از هر دو سمت می توان اطلاعات گرفت و دریافت کرد.



نکته ی مهم در مورد پروتکل UART این است که آسنکرون عمل میکند و با استفاده از کلاک داده را ارسال نمیکند. در نتیجه برای انجام این پیاده سازی از دو مکانیزم استفاده میکند: یکی تنظیم rate baud برای ارسال داده با سرعت مشخص و از پیش تعیین شده و دیگری استفاده از بیت start و و start برای ارسال داده و ارسال کل این بسته بندی به سمت گیرنده. نکته ی مهم برای ما در و و ارسال کل این بسته بندی به سمت گیرنده. نکته ی مهم برای ما در مورد این پروتکل مورد دوم یا همان نحوه ی ساختن packet و بسته بندی کردن داده با استفاده از بیت شروع و پایان است.



#### packet 1-1-1

ما از این ایده استفاده کردهایم و برای مشخص شدن شروع ارسال اول هر داده بیت برابر با یک ارسال میکنیم.

با توجه به توضیحات داده شده مربوط به UART پروژهی ما به همانگونه پیادهسازی شده است.

با این تفاوت که اندازه ی هر بسته بندی داده یا همان packet مشخص است. هر packet ما شامل ۴ بیت آدرس I/O مقصد و ۷ بیت داده ایست که باید به آن داده شود و آن را از رجیستر موجود در دستور ورودی خوانده ایم. ابتدای packet هم یک بیت برابر با عدد یک برای نشان دادن پایان فعالیت می گذاریم.

#### ۲-۲-۲ ماژول فرستنده و گیرنده داده

مورد بعدی که در مورد این پروتکل در پروژهی ما تاثیر گذاشته load موازی دادهها در هر ماژول فرستنده ی پروتکل است که بعد داده را بیت به بیت جدا میکند و روی bus به گیرنده ارسال میکند و بعد از دریافت، دوباره در ماژول گیرنده داده به صورت موازی به خارج پروتکل فرستاده می شود. هر ماژول فرستنده یا گیرنده از تعداد برابر بیتهای هر packet یا همان ۱۲ رجیستر تشکیل شده که با هم یک شیفت رجیستر را تشکیل می دهند. این شیفت رجیستر از مکانیزم PIPO) output parallel می با هم یک سیفت رجیستر از مکانیزم ورود و خروج داده از پروتکل استفاده می کند. ابتدا داده به صورت موازی داخل همه ی رجیسترها ریخته می شود. سپس آنقدر بیت به بیت این داده شیفت پیدا می کنند و از طریق bus به گیرنده رسیده باشد. نکته ی مهم این است که از مور پروتکل به تعداد بیت packet یک اندازه ی ثابت و مشخص کجا متوجه می شویم که انتقال کامل انجام شده؟ جوابش دقیقا با همان ایده ی بیت و مشخص در پروتکل TART می باشد. اما در اینجا چون تعداد بیت packet یک اندازه ی ثابت و مشخص در پروتکل TART می بیت نشانگر پایان فعالیت استفاده کنیم و پروتکل هم با روشن کردن سیگنال Load با استفاده از سیگنال Ioad نام با استفاده از سیگنال Ioad نام با استفاده از سیگنال IsiO با استفاده از سیگنال آن را با شروع فعالیت پروتکل یک می کنیم و آنقدر شیفت بیت داده برای انتقال، یک بیت در ابتدای آن را با شروع فعالیت پروتکل یک می کنیم و آنقدر شیفت می دهیم تا این بیت به اولین رجیستر در سمت گیرنده برسد که خود نشان دهنده ی پایان انتقال است.

## ۲-۳ چالش ها

یک چالشی که در پیادهسازی پروتکلها از اهمیت بسیاری برخوردار است این است که نباید به اشتباه هیچ سیگنالی در دو طرف به جز از طریق bus پروتکل با هم در ارتباط باشند. به عنوان مثال در اینجا که از پروتکل برای ارتباط پردازنده و دستگاههای I/O استفاده کردهایم نباید هیچ یک از سیگنالهای پردازنده مثل isIO در سمت I/O به کار گرفته شوند مگر این که از طریق bus ارسال شوند. اشتباهی که در ابتدای پیادهسازی کردیم این بود که بیت شروع نداشتیم و با استفاده از counter و شمردن تعداد شیفتهای انجام شده انتهای انتقال را بررسی میکردیم و برای شروع شمارنده هم از سیگنال

isIO استفاده کردیم بدون گذر از bus چرا که شمارنده در سمت I/O کار میکرد و پایان فعالیت را در ماژول گیرنده بررسی میکرد اما این سیگنال در سمت پردازنده تولید شده بود. به این منظور شد که از بیت نشانگر پایان فعالیت موجود در UART ایده گرفتیم و مکانیزم پروتکل خود را بر مبنای آن پیادهسازی کردیم.

#### ۲-۳-۲ شمای کل و ماژول بن*د*ی



این شکل شمای کلی و ماژولبندی شده ی پروتکل را نمایش می دهد. به این صورت که به ماژول فرستنده یا همان سمت CPU سیگنال فعال شدن isIO داده می شود و با آن کار خود را آغاز می کند. علاوه بر آن آدرس مقصد داده شده که طبق فرمت دستورات I-type در پردازنده ی MIPS و با اطلاع به اینکه در ۵ بیت بعد از opcode قرار می گیرد می توان آن را از تونلی به نام rs گرفت که در پیاده سازی خود انجام داده داده اداده ایم و متصل به پنج بیت مدنظر در دستور است.

| MIPS I-Type Instruction Format                                       |            |              |                                               |  |
|----------------------------------------------------------------------|------------|--------------|-----------------------------------------------|--|
| Bit Range                                                            | Field Name | Width (bits) | Description                                   |  |
| 31 – 26                                                              | opcode     | 6            | Operation code (determines instruction type). |  |
| 25 – 21                                                              | rs         | 5            | Source register operand.                      |  |
| 20 – 16                                                              | rt         | 5            | Destination/target register operand.          |  |
| 15 – 0                                                               | immediate  | 16           | Immediate value or offset.                    |  |
| 31 26 25 21 20 16 15 0<br>++<br> opcode  rs   rt   immediate  <br>++ |            |              |                                               |  |



I/O با استفاده از ماژول فرستنده I/O با استفاده از ماژول فرستنده

داده ی مورد نیاز در سمت I/O را هم از لیبل IDread در شکل زیر می گیریم چرا که همچنان طبق گفته های قبلمان آدرس رجیستری که داده مان در آن قرار دارد از پنج بیت بعد از rs می آید که در پیاده سازی ما یکسان با لیبل rt است که رجیستر فایل بعد از خواندن داده ی موجود در رجیستر rt آن را در IDread قرار می دهد.



در خروجی هم bus داریم که محل اتصال دو ماژول فرستنده و گیرنده است. دقت کنید که فعلا در حال توضیح بخش مربوط به ارسال داده به I/O هستیم. میدانیم که در هنگام کار با کیبورد نیاز داریم از I/O داده دریافت کنیم که بخشهای ورودی و خروجی دیگر روی ماژول که فعلا توضیح داده نمی شوند مربوط به این عملیات هستند. در حال حاضر روی ارسال داده به I/O تمرکز میکنیم.

در سمت ماژول گیرنده هم که خروجیهای فعال کردن یا نکردن I/O های مقصد و دادههای ارسالی به آنها مشخص هستند.



شکل زیر پیادهسازی های سمت ماژول فرستنده را نمایش می دهد. در ورودی load مکانیزمی برای اینکه فقط یک کلاک روشن باشد و ورودی موازی بگیرد و بعد خاموش شود صورت گرفته است. به این شکل که از رجیستر برای ذخیره یک مقدار در هر کلاک استفاده می کنیم به این صورت که اگر در کلاک قبل isIO روشن بوده ورودی صفر خواهد بود (در کلاک قبلی داده گرفته شده و bad خاموش می شود) در غیر این حالت ورودی از isIO می آید. برای اینکه در هنگام اجرای دو دستور I/O پشت هم مشکل ایجاد نشود یک رجیستر دیگر هم این وسط اضافه می کنیم تا با یک کلاک اضافی در آن وسط مشکلات تغییر isIO حل شود. برای اتمام دستور isIO هم یک سیگنال در نظر گرفته شده که با انجام آن PC افزایش پیدا می کند و گرنه در قبل از آن چون سینگل سایکل داریم با این همه کلاکی که خورده شده PC بارها افزایش و به دستور بعد رفته بود.

در هنگام استفاده از IOIsDone برای reset کردن هم کلاک اضافی قرار میدهیم تا در شیفت رجیستر زیر با خودش race ایجاد نشود.

#### I/O گرفتن و پردازش داده در سمت Z

در مرحلهی آخر داده ی انتقال یافته را پردازش میکنیم تا عملیاتهای مورد نظر در سمت I/O انجام شود. به این صورت که داده ها در سمت ماژول گیرنده دریافت می شوند و در نتیجه با آدرس مقصد



و یک دیکودر I/O مقصد را فعال میکنیم و داده ی لازم را که در ۷msb قرار دارد به مقصد می دهیم. حواسمان هست که با دریافت اطلاعات و رسیدن بیت نشانگر پایان شیفت را متوقف میکنیم و در کلاک بعدی هم برای اجرای دستور بعدی کلا شیفت رجیستر را reset میکنیم.



با استفاده از بیت i ام داده ی دریافت شده و بر حسب یک (روشن) و صفر (خاموش) بودن آن

وضعیت LED ام مشخص می شود. دقت کنید که برای پایدار بودن وضعیت لامپ تا زمانی که تغییر بعدی روی آن رخ دهد یک رجیستر در نظر می گیریم که تنها در زمانی که آدرس I/O مقصد لامپ موردنظر بود WE آن را فعال کند و تغییرش دهد. در غیر این حالت وضعیت قبلی لامپ را نگه می دارد. با استفاده از آدرس ۱۱۱۱ می توان همه ی لامپها را با توجه به مقدار باینری موجود در رجیستر مدنظر روشن کرد. برای TTY هم هفت بیت کاراکتر ورودی داده می شود و روی صفحه چاپ می شود.

نحوه ارسال داده KeyBoard و پردازش و گرفتن آن از سمت پردازنده و اما میرسیم به بحث کیبورد. در کیبورد علاوه بر دریافت اطلاعات از پردازنده مبنی بر اینکه میخواهیم با کیبورد کار کنیم و همچنین اطلاعات آن در چه رجیستری دخیره شود، باید کارکتر موجود در کیبورد را دریافت کرده و در رجیستر مقصد درون پردازنده ذخیره کنیم. این یعنی یک برگشت دیگر از سمت I/O به پردازنده. برای این منظور مشابه قسمت قبل عمل میکنیم و دو تا شیفت رجیستر دیگر یعنی به عبارت دیگر دو ماژول فرستنده و گیرنده ی دیگر قرار میدهیم، با این تفاوت که این دفعه فرستنده در سمت پردازنده.



عملیات کیبورد بدین صورت عمل میکند که با فعال کردن کیبورد و دریافت داده ی آن، به صورت موازی آن را به شیفت رجیستر میدهیم تا آن به سمت پردازنده ارسال کند. برای این منظور

یک سیگنال آغاز ارسال پیام به پردازنده هم مطابق شکل مشابه isIO برای این ماژول میگذاریم و با شیفت داده را به سمت پردازنده ارسال میکنیم. این سیگنال sendMessage که علاوه بر برطرف کردن نیاز کیبورد، میتوان از آن برای نشان دادن پایان کار پردازنده در دستور I/O استفاده کرد و با توجه به سیگنال IOIsDone که معرفی کردیم PC را افزایش داد و به دستور بعدی رفت.





# ۳ آزمون و تست کردن طراحی

برای بررسی عملکرد مدار و دستگاه طراحی شده از چهار برنامهی زیر استفاده میکنیم. // این دستورات در ROM برنامه ذخیره میشوند.

## ۳-۰-۱ آزمون اول: روشن و خاموش کردن LED

| Instruction        | Machine Code                            |
|--------------------|-----------------------------------------|
| addi \$r2, \$r2, 4 | 001000001000010000000000000000100       |
| led \$r2           | 111111000100001000000000000000000000000 |
| addi \$r2, \$r0, 0 | 001000000000010000000000000000000000000 |
| led \$r2           | 111111000100001000000000000000000000000 |

## TTY آزمون دوم: نمایش کاراکتر موجود در رجیستر مورد نظر روی ----

کاراکتر a با کد اسکی ۹۷ روی صفحه نمایش داده می شود

| Instruction         | Machine Code                            |
|---------------------|-----------------------------------------|
| addi \$r2, \$r2, 97 | 00100000010000100000000001100001        |
| tty \$r2            | 111111001110001000000000000000000000000 |

## TTY قرمون سوم: نمایش کل کلمه ینوشته شده روی کیبورد روی صفحه $-\infty$

| Instruction        | Machine Code                            |
|--------------------|-----------------------------------------|
| kb \$r2            | 111111010000001000000000000000000000000 |
| tty \$r2           | 111111001110001000000000000000000000000 |
| bne \$r2, \$r0, kb | 000101000100000011111111111111111111111 |

حلقه تا وقتی که به کاراکتر null با کد اسکی صفر نرسیدهایم، تکرار میشود.

## ۳-۰-۴ آزمون چهارم: تولید توالی فیبوناچی روی LED ها

این سری را تا جایی که ۷ بیت لامپ میتواند نمایش دهد (یعنی عدد ۵۵) تولید و با نور آنها به شکل عدد باینری درآورده و به نمایش میگذاریم.

| Instruction          | Machine Code                            |
|----------------------|-----------------------------------------|
| xor \$r1, \$r1, \$r1 | 000000000100001000010000100110          |
| xor \$r2, \$r2, \$r2 | 000000001000010000100000100110          |
| addi \$r2, \$r2, 1   | 001000001000010000000000000000000000000 |
| xor \$r4, \$r4, \$r4 | 0000000100001000010000000100110         |
| addi \$r4, \$r4, 89  | 00100000100001000000000001011001        |
| xor \$r5, \$r5, \$r5 | 0000000101001010010100000100110         |
| led \$r2             | 111111111111000100000000000000000000000 |
| add \$r3, \$r2, \$r1 | 0000000001000100001100000100000         |
| add \$r1, \$r2, \$r5 | 0000000101000100000100000100000         |
| add \$r2, \$r3, \$r5 | 0000000101000110001000000100000         |
| bne \$r2, \$r4, fibo | 0001010001000100111111111111111111111   |